123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241 |
- // Copyright (c) 2020 Slack Technologies, Inc.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #include "shell/common/v8_value_serializer.h"
- #include <utility>
- #include <vector>
- #include "base/memory/raw_ptr.h"
- #include "gin/converter.h"
- #include "shell/common/api/electron_api_native_image.h"
- #include "shell/common/gin_helper/microtasks_scope.h"
- #include "skia/public/mojom/bitmap.mojom.h"
- #include "third_party/blink/public/common/messaging/cloneable_message.h"
- #include "third_party/blink/public/common/messaging/web_message_port.h"
- #include "ui/gfx/image/image_skia.h"
- #include "v8/include/v8.h"
- namespace electron {
- namespace {
- enum SerializationTag {
- kNativeImageTag = 'i',
- kTrailerOffsetTag = 0xFE,
- kVersionTag = 0xFF
- };
- } // namespace
- class V8Serializer : public v8::ValueSerializer::Delegate {
- public:
- explicit V8Serializer(v8::Isolate* isolate)
- : isolate_(isolate), serializer_(isolate, this) {}
- ~V8Serializer() override = default;
- bool Serialize(v8::Local<v8::Value> value, blink::CloneableMessage* out) {
- gin_helper::MicrotasksScope microtasks_scope(
- isolate_, isolate_->GetCurrentContext()->GetMicrotaskQueue(),
- v8::MicrotasksScope::kDoNotRunMicrotasks);
- WriteBlinkEnvelope(19);
- serializer_.WriteHeader();
- bool wrote_value;
- if (!serializer_.WriteValue(isolate_->GetCurrentContext(), value)
- .To(&wrote_value)) {
- isolate_->ThrowException(v8::Exception::Error(
- gin::StringToV8(isolate_, "An object could not be cloned.")));
- return false;
- }
- DCHECK(wrote_value);
- std::pair<uint8_t*, size_t> buffer = serializer_.Release();
- DCHECK_EQ(buffer.first, data_.data());
- out->encoded_message = base::make_span(buffer.first, buffer.second);
- out->owned_encoded_message = std::move(data_);
- out->sender_agent_cluster_id =
- blink::WebMessagePort::GetEmbedderAgentClusterID();
- return true;
- }
- // v8::ValueSerializer::Delegate
- void* ReallocateBufferMemory(void* old_buffer,
- size_t size,
- size_t* actual_size) override {
- DCHECK_EQ(old_buffer, data_.data());
- data_.resize(size);
- *actual_size = data_.capacity();
- return data_.data();
- }
- void FreeBufferMemory(void* buffer) override {
- DCHECK_EQ(buffer, data_.data());
- data_ = {};
- }
- v8::Maybe<bool> WriteHostObject(v8::Isolate* isolate,
- v8::Local<v8::Object> object) override {
- api::NativeImage* native_image;
- if (gin::ConvertFromV8(isolate, object, &native_image)) {
- // Serialize the NativeImage
- WriteTag(kNativeImageTag);
- gfx::ImageSkia image = native_image->image().AsImageSkia();
- std::vector<gfx::ImageSkiaRep> image_reps = image.image_reps();
- serializer_.WriteUint32(image_reps.size());
- for (const auto& rep : image_reps) {
- serializer_.WriteDouble(rep.scale());
- const SkBitmap& bitmap = rep.GetBitmap();
- std::vector<uint8_t> bytes =
- skia::mojom::InlineBitmap::Serialize(&bitmap);
- serializer_.WriteUint32(bytes.size());
- serializer_.WriteRawBytes(bytes.data(), bytes.size());
- }
- return v8::Just(true);
- } else {
- return v8::ValueSerializer::Delegate::WriteHostObject(isolate, object);
- }
- }
- void ThrowDataCloneError(v8::Local<v8::String> message) override {
- isolate_->ThrowException(v8::Exception::Error(message));
- }
- private:
- void WriteTag(SerializationTag tag) { serializer_.WriteRawBytes(&tag, 1); }
- void WriteBlinkEnvelope(uint32_t blink_version) {
- // Write a dummy blink version envelope for compatibility with
- // blink::V8ScriptValueSerializer
- WriteTag(kVersionTag);
- serializer_.WriteUint32(blink_version);
- }
- raw_ptr<v8::Isolate> isolate_;
- std::vector<uint8_t> data_;
- v8::ValueSerializer serializer_;
- };
- class V8Deserializer : public v8::ValueDeserializer::Delegate {
- public:
- V8Deserializer(v8::Isolate* isolate, base::span<const uint8_t> data)
- : isolate_(isolate),
- deserializer_(isolate, data.data(), data.size(), this) {}
- V8Deserializer(v8::Isolate* isolate, const blink::CloneableMessage& message)
- : V8Deserializer(isolate, message.encoded_message) {}
- v8::Local<v8::Value> Deserialize() {
- v8::EscapableHandleScope scope(isolate_);
- auto context = isolate_->GetCurrentContext();
- uint32_t blink_version;
- if (!ReadBlinkEnvelope(&blink_version))
- return v8::Null(isolate_);
- bool read_header;
- if (!deserializer_.ReadHeader(context).To(&read_header))
- return v8::Null(isolate_);
- DCHECK(read_header);
- v8::Local<v8::Value> value;
- if (!deserializer_.ReadValue(context).ToLocal(&value))
- return v8::Null(isolate_);
- return scope.Escape(value);
- }
- v8::MaybeLocal<v8::Object> ReadHostObject(v8::Isolate* isolate) override {
- uint8_t tag = 0;
- if (!ReadTag(&tag))
- return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
- switch (tag) {
- case kNativeImageTag:
- if (api::NativeImage* native_image = ReadNativeImage(isolate))
- return native_image->GetWrapper(isolate);
- break;
- }
- // Throws an exception.
- return v8::ValueDeserializer::Delegate::ReadHostObject(isolate);
- }
- private:
- bool ReadTag(uint8_t* tag) {
- const void* tag_bytes = nullptr;
- if (!deserializer_.ReadRawBytes(1, &tag_bytes))
- return false;
- *tag = *reinterpret_cast<const uint8_t*>(tag_bytes);
- return true;
- }
- bool ReadBlinkEnvelope(uint32_t* blink_version) {
- // Read a dummy blink version envelope for compatibility with
- // blink::V8ScriptValueDeserializer
- uint8_t tag = 0;
- if (!ReadTag(&tag) || tag != kVersionTag)
- return false;
- if (!deserializer_.ReadUint32(blink_version))
- return false;
- static constexpr uint32_t kMinWireFormatVersionWithTrailer = 21;
- if (*blink_version >= kMinWireFormatVersionWithTrailer) {
- // In these versions, we expect kTrailerOffsetTag (0xFE) followed by an
- // offset and size. See details in
- // third_party/blink/renderer/core/v8/serialization/serialization_tag.h.
- uint8_t trailer_offset_tag = 0;
- if (!ReadTag(&trailer_offset_tag) ||
- trailer_offset_tag != kTrailerOffsetTag)
- return false;
- const void* trailer_offset_and_size_bytes = nullptr;
- static constexpr size_t kTrailerOffsetDataSize =
- sizeof(uint64_t) + sizeof(uint32_t);
- if (!deserializer_.ReadRawBytes(kTrailerOffsetDataSize,
- &trailer_offset_and_size_bytes))
- return false;
- }
- return true;
- }
- api::NativeImage* ReadNativeImage(v8::Isolate* isolate) {
- gfx::ImageSkia image_skia;
- uint32_t num_reps = 0;
- if (!deserializer_.ReadUint32(&num_reps))
- return nullptr;
- for (uint32_t i = 0; i < num_reps; i++) {
- double scale = 0.0;
- if (!deserializer_.ReadDouble(&scale))
- return nullptr;
- uint32_t bitmap_size_bytes = 0;
- if (!deserializer_.ReadUint32(&bitmap_size_bytes))
- return nullptr;
- const void* bitmap_data = nullptr;
- if (!deserializer_.ReadRawBytes(bitmap_size_bytes, &bitmap_data))
- return nullptr;
- SkBitmap bitmap;
- if (!skia::mojom::InlineBitmap::Deserialize(bitmap_data,
- bitmap_size_bytes, &bitmap))
- return nullptr;
- image_skia.AddRepresentation(gfx::ImageSkiaRep(bitmap, scale));
- }
- gfx::Image image(image_skia);
- return new api::NativeImage(isolate, image);
- }
- raw_ptr<v8::Isolate> isolate_;
- v8::ValueDeserializer deserializer_;
- };
- bool SerializeV8Value(v8::Isolate* isolate,
- v8::Local<v8::Value> value,
- blink::CloneableMessage* out) {
- return V8Serializer(isolate).Serialize(value, out);
- }
- v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
- const blink::CloneableMessage& in) {
- return V8Deserializer(isolate, in).Deserialize();
- }
- v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
- base::span<const uint8_t> data) {
- return V8Deserializer(isolate, data).Deserialize();
- }
- } // namespace electron
|