v8_util.cc 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. // Copyright (c) 2020 Slack Technologies, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/common/v8_util.h"
  5. #include <cstdint>
  6. #include <utility>
  7. #include <vector>
  8. #include "base/memory/raw_ptr.h"
  9. #include "gin/converter.h"
  10. #include "shell/common/api/electron_api_native_image.h"
  11. #include "shell/common/gin_helper/microtasks_scope.h"
  12. #include "skia/public/mojom/bitmap.mojom.h"
  13. #include "third_party/blink/public/common/messaging/cloneable_message.h"
  14. #include "third_party/blink/public/common/messaging/web_message_port.h"
  15. #include "ui/gfx/image/image_skia.h"
  16. #include "v8/include/v8.h"
  17. namespace electron {
  18. namespace {
  19. constexpr uint8_t kNativeImageTag = 'i';
  20. constexpr uint8_t kTrailerOffsetTag = 0xFE;
  21. constexpr uint8_t kVersionTag = 0xFF;
  22. } // namespace
  23. class V8Serializer : public v8::ValueSerializer::Delegate {
  24. public:
  25. explicit V8Serializer(v8::Isolate* isolate)
  26. : isolate_(isolate), serializer_(isolate, this) {}
  27. ~V8Serializer() override = default;
  28. bool Serialize(v8::Local<v8::Value> value, blink::CloneableMessage* out) {
  29. gin_helper::MicrotasksScope microtasks_scope{
  30. isolate_->GetCurrentContext(), false,
  31. v8::MicrotasksScope::kDoNotRunMicrotasks};
  32. WriteBlinkEnvelope(19);
  33. serializer_.WriteHeader();
  34. bool wrote_value;
  35. if (!serializer_.WriteValue(isolate_->GetCurrentContext(), value)
  36. .To(&wrote_value)) {
  37. isolate_->ThrowException(v8::Exception::Error(
  38. gin::StringToV8(isolate_, "An object could not be cloned.")));
  39. return false;
  40. }
  41. DCHECK(wrote_value);
  42. const auto [data_bytes, data_len] = serializer_.Release();
  43. DCHECK_EQ(std::data(data_), data_bytes);
  44. DCHECK_GE(std::size(data_), data_len);
  45. data_.resize(data_len);
  46. out->owned_encoded_message = std::move(data_);
  47. out->encoded_message = out->owned_encoded_message;
  48. out->sender_agent_cluster_id =
  49. blink::WebMessagePort::GetEmbedderAgentClusterID();
  50. return true;
  51. }
  52. // v8::ValueSerializer::Delegate
  53. void* ReallocateBufferMemory(void* old_buffer,
  54. size_t size,
  55. size_t* actual_size) override {
  56. DCHECK_EQ(old_buffer, data_.data());
  57. data_.resize(size);
  58. *actual_size = data_.capacity();
  59. return data_.data();
  60. }
  61. void FreeBufferMemory(void* buffer) override {
  62. DCHECK_EQ(buffer, data_.data());
  63. data_ = {};
  64. }
  65. v8::Maybe<bool> WriteHostObject(v8::Isolate* isolate,
  66. v8::Local<v8::Object> object) override {
  67. api::NativeImage* native_image;
  68. if (gin::ConvertFromV8(isolate, object, &native_image)) {
  69. // Serialize the NativeImage
  70. WriteTag(kNativeImageTag);
  71. gfx::ImageSkia image = native_image->image().AsImageSkia();
  72. std::vector<gfx::ImageSkiaRep> image_reps = image.image_reps();
  73. serializer_.WriteUint32(image_reps.size());
  74. for (const auto& rep : image_reps) {
  75. serializer_.WriteDouble(rep.scale());
  76. const SkBitmap& bitmap = rep.GetBitmap();
  77. std::vector<uint8_t> bytes =
  78. skia::mojom::InlineBitmap::Serialize(&bitmap);
  79. serializer_.WriteUint32(bytes.size());
  80. serializer_.WriteRawBytes(bytes.data(), bytes.size());
  81. }
  82. return v8::Just(true);
  83. } else {
  84. return v8::ValueSerializer::Delegate::WriteHostObject(isolate, object);
  85. }
  86. }
  87. void ThrowDataCloneError(v8::Local<v8::String> message) override {
  88. isolate_->ThrowException(v8::Exception::Error(message));
  89. }
  90. private:
  91. void WriteTag(const uint8_t tag) { serializer_.WriteRawBytes(&tag, 1U); }
  92. void WriteBlinkEnvelope(uint32_t blink_version) {
  93. // Write a dummy blink version envelope for compatibility with
  94. // blink::V8ScriptValueSerializer
  95. WriteTag(kVersionTag);
  96. serializer_.WriteUint32(blink_version);
  97. }
  98. raw_ptr<v8::Isolate> isolate_;
  99. std::vector<uint8_t> data_;
  100. v8::ValueSerializer serializer_;
  101. };
  102. class V8Deserializer : public v8::ValueDeserializer::Delegate {
  103. public:
  104. V8Deserializer(v8::Isolate* isolate, base::span<const uint8_t> data)
  105. : isolate_(isolate),
  106. deserializer_(isolate, data.data(), data.size(), this) {}
  107. V8Deserializer(v8::Isolate* isolate, const blink::CloneableMessage& message)
  108. : V8Deserializer(isolate, message.encoded_message) {}
  109. v8::Local<v8::Value> Deserialize() {
  110. v8::EscapableHandleScope scope(isolate_);
  111. auto context = isolate_->GetCurrentContext();
  112. uint32_t blink_version;
  113. if (!ReadBlinkEnvelope(&blink_version))
  114. return v8::Null(isolate_);
  115. bool read_header;
  116. if (!deserializer_.ReadHeader(context).To(&read_header))
  117. return v8::Null(isolate_);
  118. DCHECK(read_header);
  119. v8::Local<v8::Value> value;
  120. if (!deserializer_.ReadValue(context).ToLocal(&value))
  121. return v8::Null(isolate_);
  122. return scope.Escape(value);
  123. }
  124. v8::MaybeLocal<v8::Object> ReadHostObject(v8::Isolate* isolate) override {
  125. uint8_t tag = 0;
  126. if (!ReadTag(&tag))
  127. return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
  128. switch (tag) {
  129. case kNativeImageTag:
  130. if (api::NativeImage* native_image = ReadNativeImage(isolate))
  131. return native_image->GetWrapper(isolate);
  132. break;
  133. }
  134. // Throws an exception.
  135. return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
  136. }
  137. private:
  138. bool ReadTag(uint8_t* tag) {
  139. const void* tag_bytes = nullptr;
  140. if (!deserializer_.ReadRawBytes(1, &tag_bytes))
  141. return false;
  142. *tag = *reinterpret_cast<const uint8_t*>(tag_bytes);
  143. return true;
  144. }
  145. bool ReadBlinkEnvelope(uint32_t* blink_version) {
  146. // Read a dummy blink version envelope for compatibility with
  147. // blink::V8ScriptValueDeserializer
  148. uint8_t tag = 0;
  149. if (!ReadTag(&tag) || tag != kVersionTag)
  150. return false;
  151. if (!deserializer_.ReadUint32(blink_version))
  152. return false;
  153. static constexpr uint32_t kMinWireFormatVersionWithTrailer = 21;
  154. if (*blink_version >= kMinWireFormatVersionWithTrailer) {
  155. // In these versions, we expect kTrailerOffsetTag (0xFE) followed by an
  156. // offset and size. See details in
  157. // third_party/blink/renderer/core/v8/serialization/serialization_tag.h.
  158. uint8_t trailer_offset_tag = 0;
  159. if (!ReadTag(&trailer_offset_tag) ||
  160. trailer_offset_tag != kTrailerOffsetTag)
  161. return false;
  162. const void* trailer_offset_and_size_bytes = nullptr;
  163. static constexpr size_t kTrailerOffsetDataSize =
  164. sizeof(uint64_t) + sizeof(uint32_t);
  165. if (!deserializer_.ReadRawBytes(kTrailerOffsetDataSize,
  166. &trailer_offset_and_size_bytes))
  167. return false;
  168. }
  169. return true;
  170. }
  171. api::NativeImage* ReadNativeImage(v8::Isolate* isolate) {
  172. gfx::ImageSkia image_skia;
  173. uint32_t num_reps = 0;
  174. if (!deserializer_.ReadUint32(&num_reps))
  175. return nullptr;
  176. for (uint32_t i = 0; i < num_reps; i++) {
  177. double scale = 0.0;
  178. if (!deserializer_.ReadDouble(&scale))
  179. return nullptr;
  180. uint32_t bitmap_size_bytes = 0;
  181. if (!deserializer_.ReadUint32(&bitmap_size_bytes))
  182. return nullptr;
  183. const void* bitmap_data = nullptr;
  184. if (!deserializer_.ReadRawBytes(bitmap_size_bytes, &bitmap_data))
  185. return nullptr;
  186. SkBitmap bitmap;
  187. if (!skia::mojom::InlineBitmap::Deserialize(bitmap_data,
  188. bitmap_size_bytes, &bitmap))
  189. return nullptr;
  190. image_skia.AddRepresentation(gfx::ImageSkiaRep(bitmap, scale));
  191. }
  192. gfx::Image image(image_skia);
  193. return new api::NativeImage(isolate, image);
  194. }
  195. raw_ptr<v8::Isolate> isolate_;
  196. v8::ValueDeserializer deserializer_;
  197. };
  198. bool SerializeV8Value(v8::Isolate* isolate,
  199. v8::Local<v8::Value> value,
  200. blink::CloneableMessage* out) {
  201. return V8Serializer(isolate).Serialize(value, out);
  202. }
  203. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  204. const blink::CloneableMessage& in) {
  205. return V8Deserializer(isolate, in).Deserialize();
  206. }
  207. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  208. base::span<const uint8_t> data) {
  209. return V8Deserializer(isolate, data).Deserialize();
  210. }
  211. namespace util {
  212. /**
  213. * SAFETY: There is not yet any v8::ArrayBufferView API that passes the
  214. * UNSAFE_BUFFER_USAGE test, so let's isolate the unsafe API here.
  215. *
  216. * Where possible, Electron should use spans returned here instead of
  217. * |v8::ArrayBufferView::Buffer()->Data()|,
  218. * |v8::ArrayBufferView::ByteOffset()|,
  219. * |v8::ArrayBufferView::ByteLength()|.
  220. */
  221. base::span<uint8_t> as_byte_span(v8::Local<v8::ArrayBufferView> val) {
  222. uint8_t* data = UNSAFE_BUFFERS(static_cast<uint8_t*>(val->Buffer()->Data()) +
  223. val->ByteOffset());
  224. const size_t size = val->ByteLength();
  225. return UNSAFE_BUFFERS(base::span{data, size});
  226. }
  227. } // namespace util
  228. } // namespace electron