#ifndef SRC_NODE_UNION_BYTES_H_ #define SRC_NODE_UNION_BYTES_H_ #if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS // A union of const uint8_t* or const uint16_t* data that can be // turned into external v8::String when given an isolate. #include "v8.h" namespace node { class NonOwningExternalOneByteResource : public v8::String::ExternalOneByteStringResource { public: explicit NonOwningExternalOneByteResource(const uint8_t* data, size_t length) : data_(data), length_(length) {} ~NonOwningExternalOneByteResource() override = default; const char* data() const override { return reinterpret_cast(data_); } size_t length() const override { return length_; } NonOwningExternalOneByteResource(const NonOwningExternalOneByteResource&) = delete; NonOwningExternalOneByteResource& operator=( const NonOwningExternalOneByteResource&) = delete; private: const uint8_t* data_; size_t length_; }; class NonOwningExternalTwoByteResource : public v8::String::ExternalStringResource { public: explicit NonOwningExternalTwoByteResource(const uint16_t* data, size_t length) : data_(data), length_(length) {} ~NonOwningExternalTwoByteResource() override = default; const uint16_t* data() const override { return data_; } size_t length() const override { return length_; } NonOwningExternalTwoByteResource(const NonOwningExternalTwoByteResource&) = delete; NonOwningExternalTwoByteResource& operator=( const NonOwningExternalTwoByteResource&) = delete; private: const uint16_t* data_; size_t length_; }; // Similar to a v8::String, but it's independent from Isolates // and can be materialized in Isolates as external Strings // via ToStringChecked. The data pointers are owned by the caller. class UnionBytes { public: UnionBytes(const uint16_t* data, size_t length) : one_bytes_(nullptr), two_bytes_(data), length_(length) {} UnionBytes(const uint8_t* data, size_t length) : one_bytes_(data), two_bytes_(nullptr), length_(length) {} UnionBytes(const UnionBytes&) = default; UnionBytes& operator=(const UnionBytes&) = default; UnionBytes(UnionBytes&&) = default; UnionBytes& operator=(UnionBytes&&) = default; bool is_one_byte() const { return one_bytes_ != nullptr; } const uint16_t* two_bytes_data() const { CHECK_NOT_NULL(two_bytes_); return two_bytes_; } const uint8_t* one_bytes_data() const { CHECK_NOT_NULL(one_bytes_); return one_bytes_; } v8::Local ToStringChecked(v8::Isolate* isolate) const { if (is_one_byte()) { NonOwningExternalOneByteResource* source = new NonOwningExternalOneByteResource(one_bytes_data(), length_); return v8::String::NewExternalOneByte(isolate, source).ToLocalChecked(); } else { NonOwningExternalTwoByteResource* source = new NonOwningExternalTwoByteResource(two_bytes_data(), length_); return v8::String::NewExternalTwoByte(isolate, source).ToLocalChecked(); } } size_t length() { return length_; } private: const uint8_t* one_bytes_; const uint16_t* two_bytes_; size_t length_; }; } // namespace node #endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS #endif // SRC_NODE_UNION_BYTES_H_