123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 |
- /*
- * Copyright (c) 2021, Luke Wilde <lukew@serenityos.org>
- * Copyright (c) 2021, Linus Groh <linusg@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <AK/ByteBuffer.h>
- #include <AK/NumericLimits.h>
- #include <LibJS/Runtime/AbstractOperations.h>
- #include <LibJS/Runtime/ArrayBuffer.h>
- #include <LibJS/Runtime/DataView.h>
- #include <LibJS/Runtime/PropertyKey.h>
- #include <LibJS/Runtime/TypedArray.h>
- #include <LibWeb/Bindings/IDLAbstractOperations.h>
- namespace Web::Bindings::IDL {
- // https://webidl.spec.whatwg.org/#is-an-array-index
- bool is_an_array_index(JS::GlobalObject& global_object, JS::PropertyKey const& property_name)
- {
- // 1. If Type(P) is not String, then return false.
- if (!property_name.is_number())
- return false;
- // 2. Let index be ! CanonicalNumericIndexString(P).
- auto index = JS::canonical_numeric_index_string(global_object, property_name);
- // 3. If index is undefined, then return false.
- if (index.is_undefined())
- return false;
- // 4. If IsInteger(index) is false, then return false.
- // NOTE: IsInteger is the old name of IsIntegralNumber.
- if (!index.is_integral_number())
- return false;
- // 5. If index is −0, then return false.
- if (index.is_negative_zero())
- return false;
- // FIXME: I'm not sure if this is correct.
- auto index_as_double = index.as_double();
- // 6. If index < 0, then return false.
- if (index_as_double < 0)
- return false;
- // 7. If index ≥ 2 ** 32 − 1, then return false.
- // Note: 2 ** 32 − 1 is the maximum array length allowed by ECMAScript.
- if (index_as_double >= NumericLimits<u32>::max())
- return false;
- // 8. Return true.
- return true;
- }
- // https://webidl.spec.whatwg.org/#dfn-get-buffer-source-copy
- Optional<ByteBuffer> get_buffer_source_copy(JS::Object const& buffer_source)
- {
- // 1. Let esBufferSource be the result of converting bufferSource to an ECMAScript value.
- // 2. Let esArrayBuffer be esBufferSource.
- JS::ArrayBuffer* es_array_buffer;
- // 3. Let offset be 0.
- u32 offset = 0;
- // 4. Let length be 0.
- u32 length = 0;
- // 5. If esBufferSource has a [[ViewedArrayBuffer]] internal slot, then:
- if (is<JS::TypedArrayBase>(buffer_source)) {
- auto const& es_buffer_source = static_cast<JS::TypedArrayBase const&>(buffer_source);
- // 1. Set esArrayBuffer to esBufferSource.[[ViewedArrayBuffer]].
- es_array_buffer = es_buffer_source.viewed_array_buffer();
- // 2. Set offset to esBufferSource.[[ByteOffset]].
- offset = es_buffer_source.byte_offset();
- // 3. Set length to esBufferSource.[[ByteLength]].
- length = es_buffer_source.byte_length();
- } else if (is<JS::DataView>(buffer_source)) {
- auto const& es_buffer_source = static_cast<JS::DataView const&>(buffer_source);
- // 1. Set esArrayBuffer to esBufferSource.[[ViewedArrayBuffer]].
- es_array_buffer = es_buffer_source.viewed_array_buffer();
- // 2. Set offset to esBufferSource.[[ByteOffset]].
- offset = es_buffer_source.byte_offset();
- // 3. Set length to esBufferSource.[[ByteLength]].
- length = es_buffer_source.byte_length();
- }
- // 6. Otherwise:
- else {
- // 1. Assert: esBufferSource is an ArrayBuffer or SharedArrayBuffer object.
- auto const& es_buffer_source = static_cast<JS::ArrayBuffer const&>(buffer_source);
- es_array_buffer = &const_cast<JS ::ArrayBuffer&>(es_buffer_source);
- // 2. Set length to esBufferSource.[[ArrayBufferByteLength]].
- length = es_buffer_source.byte_length();
- }
- // 7. If ! IsDetachedBuffer(esArrayBuffer) is true, then return the empty byte sequence.
- if (es_array_buffer->is_detached())
- return ByteBuffer {};
- // 8. Let bytes be a new byte sequence of length equal to length.
- auto bytes = ByteBuffer::create_zeroed(length);
- if (!bytes.has_value())
- return {};
- // 9. For i in the range offset to offset + length − 1, inclusive, set bytes[i − offset] to ! GetValueFromBuffer(esArrayBuffer, i, Uint8, true, Unordered).
- for (u64 i = offset; i <= offset + length - 1; ++i) {
- auto value = es_array_buffer->get_value<u8>(i, true, JS::ArrayBuffer::Unordered);
- (*bytes)[i - offset] = (u8)value.as_u32();
- }
- // 10. Return bytes.
- return bytes;
- }
- }
|