Avi Drissman | 4e1b7bc3 | 2022-09-15 14:03:50 | [diff] [blame] | 1 | // Copyright 2022 The Chromium Authors |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
Richard He | 555151a5 | 2022-10-20 04:52:22 | [diff] [blame] | 4 | #include "content/public/browser/android/message_payload.h" |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 5 | |
Arthur Sonzogni | c686e8f | 2024-01-11 08:36:37 | [diff] [blame] | 6 | #include <optional> |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 7 | #include <string> |
| 8 | #include <utility> |
Victor Hugo Vianna Silva | 94fdad2 | 2025-03-18 19:51:13 | [diff] [blame] | 9 | #include <variant> |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 10 | |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 11 | #include "base/android/jni_android.h" |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 12 | #include "base/android/jni_array.h" |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 13 | #include "base/android/jni_string.h" |
| 14 | #include "base/android/scoped_java_ref.h" |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 15 | #include "base/containers/span.h" |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 16 | #include "base/notreached.h" |
Ho Cheung | f47d98d9 | 2025-06-06 14:33:16 | [diff] [blame] | 17 | #include "third_party/abseil-cpp/absl/functional/overload.h" |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 18 | #include "third_party/blink/public/common/messaging/string_message_codec.h" |
| 19 | #include "third_party/blink/public/common/messaging/transferable_message.h" |
| 20 | |
Andrew Grieve | ecb885bb | 2024-05-29 18:14:19 | [diff] [blame] | 21 | // Must come after all headers that specialize FromJniType() / ToJniType(). |
| 22 | #include "content/public/android/content_jni_headers/MessagePayloadJni_jni.h" |
| 23 | |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 24 | namespace { |
| 25 | |
| 26 | // An ArrayBufferPayload impl for Browser (Java) to JavaScript message, the |
| 27 | // ArrayBuffer payload data is stored in a Java byte array. |
| 28 | class JavaArrayBuffer : public blink::WebMessageArrayBufferPayload { |
| 29 | public: |
| 30 | explicit JavaArrayBuffer(const base::android::JavaRef<jbyteArray>& array) |
| 31 | : length_(base::android::SafeGetArrayLength( |
| 32 | base::android::AttachCurrentThread(), |
| 33 | array)), |
| 34 | array_(array) {} |
| 35 | |
| 36 | size_t GetLength() const override { return length_; } |
| 37 | |
Shu-yu Guo | 6fdcfb2 | 2022-12-22 19:55:39 | [diff] [blame] | 38 | // Java ArrayBuffers are always fixed-length. |
| 39 | bool GetIsResizableByUserJavaScript() const override { return false; } |
| 40 | |
| 41 | size_t GetMaxByteLength() const override { return length_; } |
| 42 | |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 43 | // Due to JNI limitation, Java ByteArray cannot be converted into base::span |
| 44 | // trivially. |
Arthur Sonzogni | c686e8f | 2024-01-11 08:36:37 | [diff] [blame] | 45 | std::optional<base::span<const uint8_t>> GetAsSpanIfPossible() |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 46 | const override { |
Arthur Sonzogni | c686e8f | 2024-01-11 08:36:37 | [diff] [blame] | 47 | return std::nullopt; |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 48 | } |
| 49 | |
| 50 | void CopyInto(base::span<uint8_t> dest) const override { |
| 51 | base::android::JavaByteArrayToByteSpan(base::android::AttachCurrentThread(), |
| 52 | array_, dest); |
| 53 | } |
| 54 | |
| 55 | private: |
| 56 | size_t length_; |
| 57 | base::android::ScopedJavaGlobalRef<jbyteArray> array_; |
| 58 | }; |
| 59 | } // namespace |
| 60 | |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 61 | namespace content::android { |
| 62 | |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 63 | base::android::ScopedJavaLocalRef<jobject> ConvertWebMessagePayloadToJava( |
| 64 | const blink::WebMessagePayload& payload) { |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 65 | JNIEnv* env = base::android::AttachCurrentThread(); |
Victor Hugo Vianna Silva | 94fdad2 | 2025-03-18 19:51:13 | [diff] [blame] | 66 | return std::visit( |
Ho Cheung | f47d98d9 | 2025-06-06 14:33:16 | [diff] [blame] | 67 | absl::Overload{ |
Joe Mason | 194907d | 2022-08-24 00:08:58 | [diff] [blame] | 68 | [env](const std::u16string& str) { |
| 69 | return Java_MessagePayloadJni_createFromString( |
| 70 | env, base::android::ConvertUTF16ToJavaString(env, str)); |
| 71 | }, |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 72 | [env](const std::unique_ptr<blink::WebMessageArrayBufferPayload>& |
| 73 | array_buffer) { |
| 74 | // Data is from renderer process, copy it first before use. |
| 75 | base::android::ScopedJavaLocalRef<jbyteArray> j_byte_array; |
| 76 | |
| 77 | auto span_optional = array_buffer->GetAsSpanIfPossible(); |
| 78 | if (span_optional) { |
| 79 | j_byte_array = |
| 80 | base::android::ToJavaByteArray(env, span_optional.value()); |
| 81 | } else { |
| 82 | // The ArrayBufferPayload impl does not support |GetArrayBuffer|. |
| 83 | // Fallback to allocate a temporary buffer and copy the data. |
| 84 | std::vector<uint8_t> data(array_buffer->GetLength()); |
| 85 | array_buffer->CopyInto(data); |
| 86 | j_byte_array = base::android::ToJavaByteArray(env, data); |
| 87 | } |
| 88 | |
| 89 | return Java_MessagePayloadJni_createFromArrayBuffer(env, |
| 90 | j_byte_array); |
| 91 | }}, |
Joe Mason | 194907d | 2022-08-24 00:08:58 | [diff] [blame] | 92 | payload); |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 93 | } |
| 94 | |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 95 | blink::WebMessagePayload ConvertToWebMessagePayloadFromJava( |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 96 | const base::android::ScopedJavaLocalRef<jobject>& java_message) { |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 97 | CHECK(java_message); |
| 98 | JNIEnv* env = base::android::AttachCurrentThread(); |
| 99 | const MessagePayloadType type = static_cast<MessagePayloadType>( |
| 100 | Java_MessagePayloadJni_getType(env, java_message)); |
| 101 | switch (type) { |
| 102 | case MessagePayloadType::kString: { |
| 103 | return base::android::ConvertJavaStringToUTF16( |
| 104 | Java_MessagePayloadJni_getAsString(env, java_message)); |
| 105 | } |
| 106 | case MessagePayloadType::kArrayBuffer: { |
| 107 | auto byte_array = |
| 108 | Java_MessagePayloadJni_getAsArrayBuffer(env, java_message); |
Richard He | a3bd95f | 2022-11-24 01:23:15 | [diff] [blame] | 109 | return std::make_unique<JavaArrayBuffer>(byte_array); |
Richard He | 81e959c | 2022-08-02 02:58:40 | [diff] [blame] | 110 | } |
| 111 | case MessagePayloadType::kInvalid: |
| 112 | break; |
| 113 | } |
Peter Boström | fc7ddc18 | 2024-10-31 19:37:21 | [diff] [blame] | 114 | NOTREACHED() << "Unsupported or invalid Java MessagePayload type."; |
Richard He | 57e4ed82 | 2022-07-09 08:29:53 | [diff] [blame] | 115 | } |
| 116 | |
| 117 | } // namespace content::android |