v8_value_serializer.cc 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  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/gin_helper/microtasks_scope.h"
  9. #include "third_party/blink/public/common/messaging/cloneable_message.h"
  10. #include "v8/include/v8.h"
  11. namespace electron {
  12. namespace {
  13. const uint8_t kVersionTag = 0xFF;
  14. } // namespace
  15. class V8Serializer : public v8::ValueSerializer::Delegate {
  16. public:
  17. explicit V8Serializer(v8::Isolate* isolate)
  18. : isolate_(isolate), serializer_(isolate, this) {}
  19. ~V8Serializer() override = default;
  20. bool Serialize(v8::Local<v8::Value> value, blink::CloneableMessage* out) {
  21. gin_helper::MicrotasksScope microtasks_scope(
  22. isolate_, v8::MicrotasksScope::kDoNotRunMicrotasks);
  23. WriteBlinkEnvelope(19);
  24. serializer_.WriteHeader();
  25. bool wrote_value;
  26. if (!serializer_.WriteValue(isolate_->GetCurrentContext(), value)
  27. .To(&wrote_value)) {
  28. isolate_->ThrowException(v8::Exception::Error(
  29. gin::StringToV8(isolate_, "An object could not be cloned.")));
  30. return false;
  31. }
  32. DCHECK(wrote_value);
  33. std::pair<uint8_t*, size_t> buffer = serializer_.Release();
  34. DCHECK_EQ(buffer.first, data_.data());
  35. out->encoded_message = base::make_span(buffer.first, buffer.second);
  36. out->owned_encoded_message = std::move(data_);
  37. return true;
  38. }
  39. // v8::ValueSerializer::Delegate
  40. void* ReallocateBufferMemory(void* old_buffer,
  41. size_t size,
  42. size_t* actual_size) override {
  43. DCHECK_EQ(old_buffer, data_.data());
  44. data_.resize(size);
  45. *actual_size = data_.capacity();
  46. return data_.data();
  47. }
  48. void FreeBufferMemory(void* buffer) override {
  49. DCHECK_EQ(buffer, data_.data());
  50. data_ = {};
  51. }
  52. void ThrowDataCloneError(v8::Local<v8::String> message) override {
  53. isolate_->ThrowException(v8::Exception::Error(message));
  54. }
  55. private:
  56. void WriteTag(uint8_t tag) { serializer_.WriteRawBytes(&tag, 1); }
  57. void WriteBlinkEnvelope(uint32_t blink_version) {
  58. // Write a dummy blink version envelope for compatibility with
  59. // blink::V8ScriptValueSerializer
  60. WriteTag(kVersionTag);
  61. serializer_.WriteUint32(blink_version);
  62. }
  63. v8::Isolate* isolate_;
  64. std::vector<uint8_t> data_;
  65. v8::ValueSerializer serializer_;
  66. };
  67. class V8Deserializer : public v8::ValueDeserializer::Delegate {
  68. public:
  69. V8Deserializer(v8::Isolate* isolate, base::span<const uint8_t> data)
  70. : isolate_(isolate),
  71. deserializer_(isolate, data.data(), data.size(), this) {}
  72. V8Deserializer(v8::Isolate* isolate, const blink::CloneableMessage& message)
  73. : V8Deserializer(isolate, message.encoded_message) {}
  74. v8::Local<v8::Value> Deserialize() {
  75. v8::EscapableHandleScope scope(isolate_);
  76. auto context = isolate_->GetCurrentContext();
  77. uint32_t blink_version;
  78. if (!ReadBlinkEnvelope(&blink_version))
  79. return v8::Null(isolate_);
  80. bool read_header;
  81. if (!deserializer_.ReadHeader(context).To(&read_header))
  82. return v8::Null(isolate_);
  83. DCHECK(read_header);
  84. v8::Local<v8::Value> value;
  85. if (!deserializer_.ReadValue(context).ToLocal(&value))
  86. return v8::Null(isolate_);
  87. return scope.Escape(value);
  88. }
  89. private:
  90. bool ReadTag(uint8_t* tag) {
  91. const void* tag_bytes = nullptr;
  92. if (!deserializer_.ReadRawBytes(1, &tag_bytes))
  93. return false;
  94. *tag = *reinterpret_cast<const uint8_t*>(tag_bytes);
  95. return true;
  96. }
  97. bool ReadBlinkEnvelope(uint32_t* blink_version) {
  98. // Read a dummy blink version envelope for compatibility with
  99. // blink::V8ScriptValueDeserializer
  100. uint8_t tag = 0;
  101. if (!ReadTag(&tag) || tag != kVersionTag)
  102. return false;
  103. if (!deserializer_.ReadUint32(blink_version))
  104. return false;
  105. return true;
  106. }
  107. v8::Isolate* isolate_;
  108. v8::ValueDeserializer deserializer_;
  109. };
  110. bool SerializeV8Value(v8::Isolate* isolate,
  111. v8::Local<v8::Value> value,
  112. blink::CloneableMessage* out) {
  113. return V8Serializer(isolate).Serialize(value, out);
  114. }
  115. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  116. const blink::CloneableMessage& in) {
  117. return V8Deserializer(isolate, in).Deserialize();
  118. }
  119. v8::Local<v8::Value> DeserializeV8Value(v8::Isolate* isolate,
  120. base::span<const uint8_t> data) {
  121. return V8Deserializer(isolate, data).Deserialize();
  122. }
  123. } // namespace electron