/* * 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()); }); } 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()); }); } BufferSource::~BufferSource() = default; }