Buffers.cpp 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. /*
  2. * Copyright (c) 2023, Shannon Booth <shannon@serenityos.org>
  3. * Copyright (c) 2023, Matthew Olsson <mattco@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibJS/Runtime/DataView.h>
  8. #include <LibJS/Runtime/TypedArray.h>
  9. #include <LibWeb/WebIDL/Buffers.h>
  10. namespace Web::WebIDL {
  11. GC_DEFINE_ALLOCATOR(BufferableObjectBase);
  12. GC_DEFINE_ALLOCATOR(ArrayBufferView);
  13. GC_DEFINE_ALLOCATOR(BufferSource);
  14. u32 BufferableObjectBase::byte_length() const
  15. {
  16. return m_bufferable_object.visit(
  17. [](GC::Ref<JS::TypedArrayBase> typed_array) {
  18. auto typed_array_record = JS::make_typed_array_with_buffer_witness_record(typed_array, JS::ArrayBuffer::Order::SeqCst);
  19. return JS::typed_array_byte_length(typed_array_record);
  20. },
  21. [](GC::Ref<JS::DataView> data_view) {
  22. auto view_record = JS::make_data_view_with_buffer_witness_record(data_view, JS::ArrayBuffer::Order::SeqCst);
  23. return JS::get_view_byte_length(view_record);
  24. },
  25. [](GC::Ref<JS::ArrayBuffer> array_buffer) { return static_cast<u32>(array_buffer->byte_length()); });
  26. }
  27. u32 BufferableObjectBase::element_size() const
  28. {
  29. return m_bufferable_object.visit(
  30. [](GC::Ref<JS::TypedArrayBase> typed_array) -> u32 {
  31. auto typed_array_record = JS::make_typed_array_with_buffer_witness_record(typed_array, JS::ArrayBuffer::Order::SeqCst);
  32. return typed_array_record.object->element_size();
  33. },
  34. [](GC::Ref<JS::DataView>) -> u32 {
  35. return 1;
  36. },
  37. [](GC::Ref<JS::ArrayBuffer>) -> u32 {
  38. return 1;
  39. });
  40. }
  41. GC::Ref<JS::Object> BufferableObjectBase::raw_object()
  42. {
  43. return m_bufferable_object.visit([](auto const& obj) -> GC::Ref<JS::Object> { return obj; });
  44. }
  45. GC::Ptr<JS::ArrayBuffer> BufferableObjectBase::viewed_array_buffer()
  46. {
  47. return m_bufferable_object.visit(
  48. [](GC::Ref<JS::ArrayBuffer> array_buffer) -> GC::Ptr<JS::ArrayBuffer> { return array_buffer; },
  49. [](auto const& view) -> GC::Ptr<JS::ArrayBuffer> { return view->viewed_array_buffer(); });
  50. }
  51. BufferableObject BufferableObjectBase::bufferable_object_from_raw_object(GC::Ref<JS::Object> object)
  52. {
  53. if (is<JS::TypedArrayBase>(*object))
  54. return GC::Ref { static_cast<JS::TypedArrayBase&>(*object) };
  55. if (is<JS::DataView>(*object))
  56. return GC::Ref { static_cast<JS::DataView&>(*object) };
  57. if (is<JS::ArrayBuffer>(*object))
  58. return GC::Ref { static_cast<JS::ArrayBuffer&>(*object) };
  59. VERIFY_NOT_REACHED();
  60. }
  61. BufferableObjectBase::BufferableObjectBase(GC::Ref<JS::Object> object)
  62. : m_bufferable_object(bufferable_object_from_raw_object(object))
  63. {
  64. }
  65. bool BufferableObjectBase::is_typed_array_base() const
  66. {
  67. return m_bufferable_object.has<GC::Ref<JS::TypedArrayBase>>();
  68. }
  69. bool BufferableObjectBase::is_data_view() const
  70. {
  71. return m_bufferable_object.has<GC::Ref<JS::DataView>>();
  72. }
  73. bool BufferableObjectBase::is_array_buffer() const
  74. {
  75. return m_bufferable_object.has<GC::Ref<JS::ArrayBuffer>>();
  76. }
  77. void BufferableObjectBase::visit_edges(Visitor& visitor)
  78. {
  79. Base::visit_edges(visitor);
  80. m_bufferable_object.visit([&](auto& obj) { visitor.visit(obj); });
  81. }
  82. ArrayBufferView::~ArrayBufferView() = default;
  83. u32 ArrayBufferView::byte_offset() const
  84. {
  85. return m_bufferable_object.visit(
  86. [](GC::Ref<JS::ArrayBuffer>) -> u32 { VERIFY_NOT_REACHED(); },
  87. [](auto& view) -> u32 { return static_cast<u32>(view->byte_offset()); });
  88. }
  89. // https://webidl.spec.whatwg.org/#arraybufferview-write
  90. void ArrayBufferView::write(ReadonlyBytes bytes, u32 starting_offset)
  91. {
  92. // 1. Let jsView be the result of converting view to a JavaScript value.
  93. // 2. Assert: bytes’s length ≤ jsView.[[ByteLength]] − startingOffset.
  94. VERIFY(bytes.size() <= byte_length() - starting_offset);
  95. // 3. Assert: if view is not a DataView, then bytes’s length modulo the element size of view’s type is 0.
  96. if (!m_bufferable_object.has<GC::Ref<JS::DataView>>()) {
  97. auto element_size = m_bufferable_object.get<GC::Ref<JS::TypedArrayBase>>()->element_size();
  98. VERIFY(bytes.size() % element_size == 0);
  99. }
  100. // 4. Let arrayBuffer be the result of converting jsView.[[ViewedArrayBuffer]] to an IDL value of type ArrayBuffer.
  101. auto array_buffer = viewed_array_buffer();
  102. // 5. Write bytes into arrayBuffer with startingOffset set to jsView.[[ByteOffset]] + startingOffset.
  103. array_buffer->buffer().overwrite(byte_offset() + starting_offset, bytes.data(), bytes.size());
  104. }
  105. BufferSource::~BufferSource() = default;
  106. }