v8_value_serializer.cc 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  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_value_serializer.h"
  5. #include <utility>
  6. #include <vector>
  7. #include "gin/converter.h"
  8. #include "shell/common/api/electron_api_native_image.h"
  9. #include "shell/common/gin_helper/microtasks_scope.h"
  10. #include "skia/public/mojom/bitmap.mojom.h"
  11. #include "third_party/blink/public/common/messaging/cloneable_message.h"
  12. #include "ui/gfx/image/image_skia.h"
  13. #include "v8/include/v8.h"
  14. namespace electron {
  15. namespace {
  16. enum SerializationTag { kNativeImageTag = 'i', kVersionTag = 0xFF };
  17. } // namespace
  18. class V8Serializer : public v8::ValueSerializer::Delegate {
  19. public:
  20. explicit V8Serializer(v8::Isolate* isolate)
  21. : isolate_(isolate), serializer_(isolate, this) {}
  22. ~V8Serializer() override = default;
  23. bool Serialize(v8::Local<v8::Value> value, blink::CloneableMessage* out) {
  24. gin_helper::MicrotasksScope microtasks_scope(
  25. isolate_, v8::MicrotasksScope::kDoNotRunMicrotasks);
  26. WriteBlinkEnvelope(19);
  27. serializer_.WriteHeader();
  28. bool wrote_value;
  29. if (!serializer_.WriteValue(isolate_->GetCurrentContext(), value)
  30. .To(&wrote_value)) {
  31. isolate_->ThrowException(v8::Exception::Error(
  32. gin::StringToV8(isolate_, "An object could not be cloned.")));
  33. return false;
  34. }
  35. DCHECK(wrote_value);
  36. std::pair<uint8_t*, size_t> buffer = serializer_.Release();
  37. DCHECK_EQ(buffer.first, data_.data());
  38. out->encoded_message = base::make_span(buffer.first, buffer.second);
  39. out->owned_encoded_message = std::move(data_);
  40. return true;
  41. }
  42. // v8::ValueSerializer::Delegate
  43. void* ReallocateBufferMemory(void* old_buffer,
  44. size_t size,
  45. size_t* actual_size) override {
  46. DCHECK_EQ(old_buffer, data_.data());
  47. data_.resize(size);
  48. *actual_size = data_.capacity();
  49. return data_.data();
  50. }
  51. void FreeBufferMemory(void* buffer) override {
  52. DCHECK_EQ(buffer, data_.data());
  53. data_ = {};
  54. }
  55. v8::Maybe<bool> WriteHostObject(v8::Isolate* isolate,
  56. v8::Local<v8::Object> object) override {
  57. api::NativeImage* native_image;
  58. if (gin::ConvertFromV8(isolate, object, &native_image)) {
  59. // Serialize the NativeImage
  60. WriteTag(kNativeImageTag);
  61. gfx::ImageSkia image = native_image->image().AsImageSkia();
  62. std::vector<gfx::ImageSkiaRep> image_reps = image.image_reps();
  63. serializer_.WriteUint32(image_reps.size());
  64. for (const auto& rep : image_reps) {
  65. serializer_.WriteDouble(rep.scale());
  66. const SkBitmap& bitmap = rep.GetBitmap();
  67. std::vector<uint8_t> bytes =
  68. skia::mojom::InlineBitmap::Serialize(&bitmap);
  69. serializer_.WriteUint32(bytes.size());
  70. serializer_.WriteRawBytes(bytes.data(), bytes.size());
  71. }
  72. return v8::Just(true);
  73. } else {
  74. return v8::ValueSerializer::Delegate::WriteHostObject(isolate, object);
  75. }
  76. }
  77. void ThrowDataCloneError(v8::Local<v8::String> message) override {
  78. isolate_->ThrowException(v8::Exception::Error(message));
  79. }
  80. private:
  81. void WriteTag(SerializationTag tag) { serializer_.WriteRawBytes(&tag, 1); }
  82. void WriteBlinkEnvelope(uint32_t blink_version) {
  83. // Write a dummy blink version envelope for compatibility with
  84. // blink::V8ScriptValueSerializer
  85. WriteTag(kVersionTag);
  86. serializer_.WriteUint32(blink_version);
  87. }
  88. v8::Isolate* isolate_;
  89. std::vector<uint8_t> data_;
  90. v8::ValueSerializer serializer_;
  91. };
  92. class V8Deserializer : public v8::ValueDeserializer::Delegate {
  93. public:
  94. V8Deserializer(v8::Isolate* isolate, base::span<const uint8_t> data)
  95. : isolate_(isolate),
  96. deserializer_(isolate, data.data(), data.size(), this) {}
  97. V8Deserializer(v8::Isolate* isolate, const blink::CloneableMessage& message)
  98. : V8Deserializer(isolate, message.encoded_message) {}
  99. v8::Local<v8::Value> Deserialize() {
  100. v8::EscapableHandleScope scope(isolate_);
  101. auto context = isolate_->GetCurrentContext();
  102. uint32_t blink_version;
  103. if (!ReadBlinkEnvelope(&blink_version))
  104. return v8::Null(isolate_);
  105. bool read_header;
  106. if (!deserializer_.ReadHeader(context).To(&read_header))
  107. return v8::Null(isolate_);
  108. DCHECK(read_header);
  109. v8::Local<v8::Value> value;
  110. if (!deserializer_.ReadValue(context).ToLocal(&value))
  111. return v8::Null(isolate_);
  112. return scope.Escape(value);
  113. }
  114. v8::MaybeLocal<v8::Object> ReadHostObject(v8::Isolate* isolate) override {
  115. uint8_t tag = 0;
  116. if (!ReadTag(&tag))
  117. return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
  118. switch (tag) {
  119. case kNativeImageTag:
  120. if (api::NativeImage* native_image = ReadNativeImage(isolate))
  121. return native_image->GetWrapper(isolate);
  122. break;
  123. }
  124. // Throws an exception.
  125. return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
  126. }
  127. private:
  128. bool ReadTag(uint8_t* tag) {
  129. const void* tag_bytes = nullptr;
  130. if (!deserializer_.ReadRawBytes(1, &tag_bytes))
  131. return false;
  132. *tag = *reinterpret_cast<const uint8_t*>(tag_bytes);
  133. return true;
  134. }
  135. bool ReadBlinkEnvelope(uint32_t* blink_version) {
  136. // Read a dummy blink version envelope for compatibility with
  137. // blink::V8ScriptValueDeserializer
  138. uint8_t tag = 0;
  139. if (!ReadTag(&tag) || tag != kVersionTag)
  140. return false;
  141. if (!deserializer_.ReadUint32(blink_version))
  142. return false;
  143. return true;
  144. }
  145. api::NativeImage* ReadNativeImage(v8::Isolate* isolate) {
  146. gfx::ImageSkia image_skia;
  147. uint32_t num_reps = 0;
  148. if (!deserializer_.ReadUint32(&num_reps))
  149. return nullptr;
  150. for (uint32_t i = 0; i < num_reps; i++) {
  151. double scale = 0.0;
  152. if (!deserializer_.ReadDouble(&scale))
  153. return nullptr;
  154. uint32_t bitmap_size_bytes = 0;
  155. if (!deserializer_.ReadUint32(&bitmap_size_bytes))
  156. return nullptr;
  157. const void* bitmap_data = nullptr;
  158. if (!deserializer_.ReadRawBytes(bitmap_size_bytes, &bitmap_data))
  159. return nullptr;
  160. SkBitmap bitmap;
  161. if (!skia::mojom::InlineBitmap::Deserialize(bitmap_data,
  162. bitmap_size_bytes, &bitmap))
  163. return nullptr;
  164. image_skia.AddRepresentation(gfx::ImageSkiaRep(bitmap, scale));
  165. }
  166. gfx::Image image(image_skia);
  167. return new api::NativeImage(isolate, image);
  168. }
  169. v8::Isolate* isolate_;
  170. v8::ValueDeserializer deserializer_;
  171. };
  172. bool SerializeV8Value(v8::Isolate* isolate,
  173. v8::Local<v8::Value> value,
  174. blink::CloneableMessage* out) {
  175. return V8Serializer(isolate).Serialize(value, out);
  176. }
  177. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  178. const blink::CloneableMessage& in) {
  179. return V8Deserializer(isolate, in).Deserialize();
  180. }
  181. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  182. base::span<const uint8_t> data) {
  183. return V8Deserializer(isolate, data).Deserialize();
  184. }
  185. } // namespace electron