| // Copyright 2022 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef CONTENT_BROWSER_ANDROID_APP_WEB_MESSAGE_PORT_H_ |
| #define CONTENT_BROWSER_ANDROID_APP_WEB_MESSAGE_PORT_H_ |
| |
| #include "base/android/jni_android.h" |
| #include "base/android/jni_weak_ref.h" |
| #include "base/android/scoped_java_ref.h" |
| #include "base/memory/scoped_refptr.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "content/common/content_export.h" |
| #include "third_party/blink/public/common/messaging/message_port_descriptor.h" |
| #include "third_party/blink/public/mojom/messaging/transferable_message.mojom.h" |
| |
| namespace mojo { |
| class Connector; |
| class Message; |
| } // namespace mojo |
| |
| namespace content::android { |
| |
| // This is the native side of the java `AppWebMessagePort`, which is present |
| // while the port is still bound to the java object. While bound, the lifetime |
| // of the native side is owned by the java side. Creation: The only way to |
| // create this object is via `content::MessagePort::CreateJavaMessagePort`. The |
| // native object will be created first, and Java object is created within the |
| // constructor of native object. |
| // |
| // Destruction: |
| // 1. Pass the ownership to `blink::MessagePortDescriptor` with |
| // `Release`. This unbinds the message port from the java |
| // object and moves it to `MessagePortDescriptor` to be passed in a message. |
| // Native `AppWebMessagePort` object is deleted as part of this operation. |
| // 2. Close the MessagePort in Java. |
| // 3. The Java object is garbage collected, but `close` is not called. |
| // |
| // Threading: Methods should be called on UI thread only. |
| class CONTENT_EXPORT AppWebMessagePort : public mojo::MessageReceiver { |
| public: |
| // Create `AppWebMessagePort` from a `MessagePortDescriptor`, this takes the |
| // ownership of `MessagePortDescriptor`. `AppWebMessagePort` is created and |
| // returned. |
| static base::android::ScopedJavaLocalRef<jobject> Create( |
| blink::MessagePortDescriptor&& descriptor); |
| |
| // Given an array of `AppWebMessagePort` objects, unwraps them and returns an |
| // equivalent array of `blink::MessagePortDescriptors`. The underlying |
| // instances(Java and Native) will be destroyed. |
| static std::vector<blink::MessagePortDescriptor> Release( |
| JNIEnv* env, |
| const base::android::JavaRef<jobjectArray>& |
| jports /* org.chromium.content.browser.AppWebMessagePort */); |
| |
| // When clean up native, we notify Java instance to release native handle. |
| ~AppWebMessagePort() override; |
| |
| void PostMessage( |
| JNIEnv* env, |
| const base::android::JavaParamRef<jobject>& j_message_payload, |
| const base::android::JavaParamRef<jobjectArray>& j_ports); |
| void SetShouldReceiveMessages(JNIEnv* env, bool should_receive_message); |
| void CloseAndDestroy(JNIEnv* env); |
| |
| private: |
| explicit AppWebMessagePort(blink::MessagePortDescriptor&& descriptor); |
| |
| // Convert `j_obj_` weak reference to a strong local reference. |
| // The Java object is always available until native object destroyed. |
| // This assumption is still true even when the Java object is GC-ed, since |
| // it's strong referenced again by PostTask. |
| base::android::ScopedJavaLocalRef<jobject> GetJavaObj(JNIEnv* env) { |
| auto java_ref = j_obj_.get(env); |
| DCHECK(java_ref); |
| return java_ref; |
| } |
| |
| // Pass out the port descriptor. This port will become invalid after calling |
| // this. This can only be called if `PostMessage` and `SetMessageCallback` are |
| // never called. |
| blink::MessagePortDescriptor PassPort(); |
| |
| // The error handler for the connector. This lets the instrumentation be aware |
| // of the message pipe closing itself due to error, which happens |
| // unconditionally. Note that a subsequent call to |
| // `Connector::PassMessagePipe` should always return an invalid handle at that |
| // point. |
| // TODO(chrisha): Make this an immediate notification that the channel has |
| // been torn down rather than waiting for the owning `MessagePort` to be |
| // cleaned up. |
| void OnPipeError(); |
| void GiveDisentangledHandleIfNeeded(); |
| |
| // mojo::MessageReceiver: |
| bool Accept(mojo::Message* message) override; |
| |
| // UI thread. |
| scoped_refptr<base::SingleThreadTaskRunner> runner_; |
| |
| blink::MessagePortDescriptor descriptor_; |
| JavaObjectWeakGlobalRef j_obj_; |
| |
| // Set when this port is receiving messages. Port should be kept alive |
| // as long as it can still receive messages. |
| base::android::ScopedJavaGlobalRef<jobject> j_strong_obj_; |
| |
| bool connector_errored_ = false; |
| bool is_watching_ = false; |
| |
| std::unique_ptr<mojo::Connector> connector_; |
| }; |
| |
| } // namespace content::android |
| |
| #endif // CONTENT_BROWSER_ANDROID_APP_WEB_MESSAGE_PORT_H_ |