/* * Copyright (c) 2023, Shannon Booth * Copyright (c) 2023, Matthew Olsson * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include namespace Web::WebIDL { GC_DEFINE_ALLOCATOR(BufferableObjectBase); GC_DEFINE_ALLOCATOR(ArrayBufferView); GC_DEFINE_ALLOCATOR(BufferSource); u32 BufferableObjectBase::byte_length() const { return m_bufferable_object.visit( [](GC::Ref typed_array) { auto typed_array_record = JS::make_typed_array_with_buffer_witness_record(typed_array, JS::ArrayBuffer::Order::SeqCst); return JS::typed_array_byte_length(typed_array_record); }, [](GC::Ref data_view) { auto view_record = JS::make_data_view_with_buffer_witness_record(data_view, JS::ArrayBuffer::Order::SeqCst); return JS::get_view_byte_length(view_record); }, [](GC::Ref array_buffer) { return static_cast(array_buffer->byte_length()); }); } u32 BufferableObjectBase::element_size() const { return m_bufferable_object.visit( [](GC::Ref typed_array) -> u32 { auto typed_array_record = JS::make_typed_array_with_buffer_witness_record(typed_array, JS::ArrayBuffer::Order::SeqCst); return typed_array_record.object->element_size(); }, [](GC::Ref) -> u32 { return 1; }, [](GC::Ref) -> u32 { return 1; }); } GC::Ref BufferableObjectBase::raw_object() { return m_bufferable_object.visit([](auto const& obj) -> GC::Ref { return obj; }); } GC::Ptr BufferableObjectBase::viewed_array_buffer() { return m_bufferable_object.visit( [](GC::Ref array_buffer) -> GC::Ptr { return array_buffer; }, [](auto const& view) -> GC::Ptr { return view->viewed_array_buffer(); }); } BufferableObject BufferableObjectBase::bufferable_object_from_raw_object(GC::Ref object) { if (is(*object)) return GC::Ref { static_cast(*object) }; if (is(*object)) return GC::Ref { static_cast(*object) }; if (is(*object)) return GC::Ref { static_cast(*object) }; VERIFY_NOT_REACHED(); } BufferableObjectBase::BufferableObjectBase(GC::Ref object) : m_bufferable_object(bufferable_object_from_raw_object(object)) { } bool BufferableObjectBase::is_typed_array_base() const { return m_bufferable_object.has>(); } bool BufferableObjectBase::is_data_view() const { return m_bufferable_object.has>(); } bool BufferableObjectBase::is_array_buffer() const { return m_bufferable_object.has>(); } void BufferableObjectBase::visit_edges(Visitor& visitor) { Base::visit_edges(visitor); m_bufferable_object.visit([&](auto& obj) { visitor.visit(obj); }); } ArrayBufferView::~ArrayBufferView() = default; u32 ArrayBufferView::byte_offset() const { return m_bufferable_object.visit( [](GC::Ref) -> u32 { VERIFY_NOT_REACHED(); }, [](auto& view) -> u32 { return static_cast(view->byte_offset()); }); } // https://webidl.spec.whatwg.org/#arraybufferview-write void ArrayBufferView::write(ReadonlyBytes bytes, u32 starting_offset) { // 1. Let jsView be the result of converting view to a JavaScript value. // 2. Assert: bytes’s length ≤ jsView.[[ByteLength]] − startingOffset. VERIFY(bytes.size() <= byte_length() - starting_offset); // 3. Assert: if view is not a DataView, then bytes’s length modulo the element size of view’s type is 0. if (!m_bufferable_object.has>()) { auto element_size = m_bufferable_object.get>()->element_size(); VERIFY(bytes.size() % element_size == 0); } // 4. Let arrayBuffer be the result of converting jsView.[[ViewedArrayBuffer]] to an IDL value of type ArrayBuffer. auto array_buffer = viewed_array_buffer(); // 5. Write bytes into arrayBuffer with startingOffset set to jsView.[[ByteOffset]] + startingOffset. array_buffer->buffer().overwrite(byte_offset() + starting_offset, bytes.data(), bytes.size()); } BufferSource::~BufferSource() = default; }