/* * Copyright (c) 2022, Daniel Ehrenberg * Copyright (c) 2022, Andrew Kaster * Copyright (c) 2024, Kenneth Myhra * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include // Structured serialize is an entirely different format from IPC because: // - It contains representation of type information // - It may contain circularities // - It is restricted to JS values namespace Web::HTML { using SerializationRecord = Vector; using SerializationMemory = HashMap, u32>; using DeserializationMemory = JS::MarkedVector; struct TransferDataHolder { Vector data; Vector fds; }; struct SerializedTransferRecord { SerializationRecord serialized; Vector transfer_data_holders; }; struct DeserializedTransferRecord { JS::Value deserialized; Vector> transferred_values; }; struct DeserializedRecord { Optional value; size_t position; }; enum class TransferType : u8 { MessagePort, }; WebIDL::ExceptionOr structured_serialize(JS::VM& vm, JS::Value); WebIDL::ExceptionOr structured_serialize_for_storage(JS::VM& vm, JS::Value); WebIDL::ExceptionOr structured_serialize_internal(JS::VM& vm, JS::Value, bool for_storage, SerializationMemory&); WebIDL::ExceptionOr structured_deserialize(JS::VM& vm, SerializationRecord const& serialized, JS::Realm& target_realm, Optional); WebIDL::ExceptionOr structured_deserialize_internal(JS::VM& vm, ReadonlySpan const& serialized, JS::Realm& target_realm, DeserializationMemory& memory, Optional position = {}); void serialize_boolean_primitive(SerializationRecord& serialized, JS::Value& value); void serialize_number_primitive(SerializationRecord& serialized, JS::Value& value); WebIDL::ExceptionOr serialize_big_int_primitive(JS::VM& vm, SerializationRecord& serialized, JS::Value& value); WebIDL::ExceptionOr serialize_string_primitive(JS::VM& vm, SerializationRecord& serialized, JS::Value& value); void serialize_boolean_object(SerializationRecord& serialized, JS::Value& value); void serialize_number_object(SerializationRecord& serialized, JS::Value& value); WebIDL::ExceptionOr serialize_big_int_object(JS::VM& vm, SerializationRecord& serialized, JS::Value& value); WebIDL::ExceptionOr serialize_string_object(JS::VM& vm, SerializationRecord& serialized, JS::Value& value); void serialize_date_object(SerializationRecord& serialized, JS::Value& value); WebIDL::ExceptionOr serialize_reg_exp_object(JS::VM& vm, SerializationRecord& serialized, JS::Value& value); template requires(IsIntegral || IsFloatingPoint) void serialize_primitive_type(SerializationRecord& serialized, T value) { if constexpr (sizeof(T) < sizeof(u32)) { // NOTE: If the value is smaller than a u32, we can just store it directly. serialized.append(static_cast(value)); return; } serialized.append(bit_cast(&value), sizeof(T) / 4); } template requires(IsEnum) void serialize_enum(SerializationRecord& serialized, T value) { serialize_primitive_type>(serialized, to_underlying(value)); } WebIDL::ExceptionOr serialize_bytes(JS::VM& vm, Vector& vector, ReadonlyBytes bytes); WebIDL::ExceptionOr serialize_string(JS::VM& vm, Vector& vector, DeprecatedFlyString const& string); WebIDL::ExceptionOr serialize_string(JS::VM& vm, Vector& vector, String const& string); WebIDL::ExceptionOr serialize_string(JS::VM& vm, Vector& vector, JS::PrimitiveString const& primitive_string); WebIDL::ExceptionOr serialize_array_buffer(JS::VM& vm, Vector& vector, JS::ArrayBuffer const& array_buffer, bool for_storage); template ViewType> WebIDL::ExceptionOr serialize_viewed_array_buffer(JS::VM& vm, Vector& vector, ViewType const& view, bool for_storage, SerializationMemory& memory); bool deserialize_boolean_primitive(ReadonlySpan const& serialized, size_t& position); double deserialize_number_primitive(ReadonlySpan const& serialized, size_t& position); JS::NonnullGCPtr deserialize_boolean_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); JS::NonnullGCPtr deserialize_number_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); WebIDL::ExceptionOr> deserialize_big_int_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); WebIDL::ExceptionOr> deserialize_string_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); JS::NonnullGCPtr deserialize_date_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); WebIDL::ExceptionOr> deserialize_reg_exp_object(JS::Realm& realm, ReadonlySpan const& serialized, size_t& position); template requires(IsIntegral || IsFloatingPoint || IsEnum) T deserialize_primitive_type(ReadonlySpan const& serialized, size_t& position) { T value; // NOTE: Make sure we always round up, otherwise Ts that are less than 32 bit will end up with a size of 0. auto size = 1 + ((sizeof(value) - 1) / 4); VERIFY(position + size <= serialized.size()); memcpy(&value, serialized.offset_pointer(position), sizeof(value)); position += size; return value; } WebIDL::ExceptionOr deserialize_bytes(JS::VM& vm, ReadonlySpan vector, size_t& position); WebIDL::ExceptionOr deserialize_string(JS::VM& vm, ReadonlySpan vector, size_t& position); WebIDL::ExceptionOr> deserialize_string_primitive(JS::VM& vm, ReadonlySpan vector, size_t& position); WebIDL::ExceptionOr> deserialize_big_int_primitive(JS::VM& vm, ReadonlySpan vector, size_t& position); WebIDL::ExceptionOr structured_serialize_with_transfer(JS::VM& vm, JS::Value value, Vector> const& transfer_list); WebIDL::ExceptionOr structured_deserialize_with_transfer(JS::VM& vm, SerializedTransferRecord&); } namespace IPC { template<> ErrorOr encode(Encoder&, ::Web::HTML::SerializedTransferRecord const&); template<> ErrorOr encode(Encoder&, ::Web::HTML::TransferDataHolder const&); template<> ErrorOr<::Web::HTML::SerializedTransferRecord> decode(Decoder&); template<> ErrorOr<::Web::HTML::TransferDataHolder> decode(Decoder&); }