123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- /*
- * Copyright (c) 2020-2023, Linus Groh <linusg@serenityos.org>
- * Copyright (c) 2021-2022, Jamie Mansfield <jmansfield@cadixdev.org>
- * Copyright (c) 2021, Idan Horowitz <idan.horowitz@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <AK/Function.h>
- #include <LibJS/Runtime/AbstractOperations.h>
- #include <LibJS/Runtime/ArrayBufferConstructor.h>
- #include <LibJS/Runtime/ArrayBufferPrototype.h>
- #include <LibJS/Runtime/GlobalObject.h>
- namespace JS {
- GC_DEFINE_ALLOCATOR(ArrayBufferPrototype);
- ArrayBufferPrototype::ArrayBufferPrototype(Realm& realm)
- : PrototypeObject(realm.intrinsics().object_prototype())
- {
- }
- void ArrayBufferPrototype::initialize(Realm& realm)
- {
- auto& vm = this->vm();
- Base::initialize(realm);
- u8 attr = Attribute::Writable | Attribute::Configurable;
- define_native_accessor(realm, vm.names.byteLength, byte_length_getter, {}, Attribute::Configurable);
- define_native_accessor(realm, vm.names.detached, detached_getter, {}, Attribute::Configurable);
- define_native_accessor(realm, vm.names.maxByteLength, max_byte_length, {}, Attribute::Configurable);
- define_native_accessor(realm, vm.names.resizable, resizable, {}, Attribute::Configurable);
- define_native_function(realm, vm.names.resize, resize, 1, attr);
- define_native_function(realm, vm.names.slice, slice, 2, attr);
- define_native_function(realm, vm.names.transfer, transfer, 0, attr);
- define_native_function(realm, vm.names.transferToFixedLength, transfer_to_fixed_length, 0, attr);
- // 25.1.6.7 ArrayBuffer.prototype [ @@toStringTag ], https://tc39.es/ecma262/#sec-arraybuffer.prototype-@@tostringtag
- define_direct_property(vm.well_known_symbol_to_string_tag(), PrimitiveString::create(vm, vm.names.ArrayBuffer.as_string()), Attribute::Configurable);
- }
- // 25.1.6.1 get ArrayBuffer.prototype.byteLength, https://tc39.es/ecma262/#sec-get-arraybuffer.prototype.bytelength
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::byte_length_getter)
- {
- // 1. Let O be the this value.
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferData]]).
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // NOTE: These steps are done in byte_length()
- // 4. If IsDetachedBuffer(O) is true, return +0𝔽.
- // 5. Let length be O.[[ArrayBufferByteLength]].
- // 6. Return 𝔽(length).
- return Value(array_buffer_object->byte_length());
- }
- // 25.1.6.3 get ArrayBuffer.prototype.detached, https://tc39.es/ecma262/#sec-get-arraybuffer.prototype.detached
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::detached_getter)
- {
- // 1. Let O be the this value.
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferData]]).
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 4. Return IsDetachedBuffer(O).
- return Value(array_buffer_object->is_detached());
- }
- // 25.1.6.4 get ArrayBuffer.prototype.maxByteLength, https://tc39.es/ecma262/#sec-get-arraybuffer.prototype.maxbytelength
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::max_byte_length)
- {
- // 1. Let O be the this value.
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferData]]).
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 4. If IsDetachedBuffer(O) is true, return +0𝔽.
- if (array_buffer_object->is_detached())
- return Value { 0 };
- size_t length = 0;
- // 5. If IsFixedLengthArrayBuffer(O) is true, then
- if (array_buffer_object->is_fixed_length()) {
- // a. Let length be O.[[ArrayBufferByteLength]].
- length = array_buffer_object->byte_length();
- }
- // 6. Else,
- else {
- // a. Let length be O.[[ArrayBufferMaxByteLength]].
- length = array_buffer_object->max_byte_length();
- }
- // 7. Return 𝔽(length).
- return Value { length };
- }
- // 25.1.6.5 get ArrayBuffer.prototype.resizable, https://tc39.es/ecma262/#sec-get-arraybuffer.prototype.resizable
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::resizable)
- {
- // 1. Let O be the this value.
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferData]]).
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 4. If IsFixedLengthArrayBuffer(O) is false, return true; otherwise return false.
- return Value { !array_buffer_object->is_fixed_length() };
- }
- // 25.1.6.6 ArrayBuffer.prototype.resize ( newLength ), https://tc39.es/ecma262/#sec-arraybuffer.prototype.resize
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::resize)
- {
- auto new_length = vm.argument(0);
- // 1. Let O be the this value.
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferMaxByteLength]]).
- if (array_buffer_object->is_fixed_length())
- return vm.throw_completion<TypeError>(ErrorType::FixedArrayBuffer);
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 4. Let newByteLength be ? ToIndex(newLength).
- auto new_byte_length = TRY(new_length.to_index(vm));
- // 5. If IsDetachedBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_detached())
- return vm.throw_completion<TypeError>(ErrorType::DetachedArrayBuffer);
- // 6. If newByteLength > O.[[ArrayBufferMaxByteLength]], throw a RangeError exception.
- if (new_byte_length > array_buffer_object->max_byte_length())
- return vm.throw_completion<RangeError>(ErrorType::ByteLengthExceedsMaxByteLength, new_byte_length, array_buffer_object->max_byte_length());
- // 7. Let hostHandled be ? HostResizeArrayBuffer(O, newByteLength).
- auto host_handled = TRY(vm.host_resize_array_buffer(array_buffer_object, new_byte_length));
- // 8. If hostHandled is handled, return undefined.
- if (host_handled == HandledByHost::Handled)
- return js_undefined();
- // 9. Let oldBlock be O.[[ArrayBufferData]].
- auto const& old_block = array_buffer_object->buffer();
- // 10. Let newBlock be ? CreateByteDataBlock(newByteLength).
- auto new_block = TRY(create_byte_data_block(vm, new_byte_length));
- // 11. Let copyLength be min(newByteLength, O.[[ArrayBufferByteLength]]).
- auto copy_length = min(new_byte_length, array_buffer_object->byte_length());
- // 12. Perform CopyDataBlockBytes(newBlock, 0, oldBlock, 0, copyLength).
- copy_data_block_bytes(new_block.buffer(), 0, old_block, 0, copy_length);
- // 13. NOTE: Neither creation of the new Data Block nor copying from the old Data Block are observable. Implementations may implement this method as in-place growth or shrinkage.
- // 14. Set O.[[ArrayBufferData]] to newBlock.
- array_buffer_object->set_data_block(move(new_block));
- // 15. Set O.[[ArrayBufferByteLength]] to newByteLength.
- // 16. Return undefined.
- return js_undefined();
- }
- // 25.1.6.7 ArrayBuffer.prototype.slice ( start, end ), https://tc39.es/ecma262/#sec-arraybuffer.prototype.slice
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::slice)
- {
- auto& realm = *vm.current_realm();
- auto start = vm.argument(0);
- auto end = vm.argument(1);
- // 1. Let O be the this value.
- // 2. Perform ? RequireInternalSlot(O, [[ArrayBufferData]]).
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 3. If IsSharedArrayBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 4. If IsDetachedBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_detached())
- return vm.throw_completion<TypeError>(ErrorType::DetachedArrayBuffer);
- // 5. Let len be O.[[ArrayBufferByteLength]].
- auto length = array_buffer_object->byte_length();
- // 6. Let relativeStart be ? ToIntegerOrInfinity(start).
- auto relative_start = TRY(start.to_integer_or_infinity(vm));
- double first;
- // 7. If relativeStart is -∞, let first be 0.
- if (Value(relative_start).is_negative_infinity())
- first = 0;
- // 8. Else if relativeStart < 0, let first be max(len + relativeStart, 0).
- else if (relative_start < 0)
- first = max(length + relative_start, 0.0);
- // 9. Else, let first be min(relativeStart, len).
- else
- first = min(relative_start, (double)length);
- // 10. If end is undefined, let relativeEnd be len; else let relativeEnd be ? ToIntegerOrInfinity(end).
- auto relative_end = end.is_undefined() ? length : TRY(end.to_integer_or_infinity(vm));
- double final;
- // 11. If relativeEnd is -∞, let final be 0.
- if (Value(relative_end).is_negative_infinity())
- final = 0;
- // 12. Else if relativeEnd < 0, let final be max(len + relativeEnd, 0).
- else if (relative_end < 0)
- final = max(length + relative_end, 0.0);
- // 13. Else, let final be min(relativeEnd, len).
- else
- final = min(relative_end, (double)length);
- // 14. Let newLen be max(final - first, 0).
- auto new_length = max(final - first, 0.0);
- // 15. Let ctor be ? SpeciesConstructor(O, %ArrayBuffer%).
- auto* constructor = TRY(species_constructor(vm, array_buffer_object, realm.intrinsics().array_buffer_constructor()));
- // 16. Let new be ? Construct(ctor, « 𝔽(newLen) »).
- auto new_array_buffer = TRY(construct(vm, *constructor, Value(new_length)));
- // 17. Perform ? RequireInternalSlot(new, [[ArrayBufferData]]).
- if (!is<ArrayBuffer>(new_array_buffer.ptr()))
- return vm.throw_completion<TypeError>(ErrorType::SpeciesConstructorDidNotCreate, "an ArrayBuffer");
- auto* new_array_buffer_object = static_cast<ArrayBuffer*>(new_array_buffer.ptr());
- // 18. If IsSharedArrayBuffer(new) is true, throw a TypeError exception.
- if (new_array_buffer_object->is_shared_array_buffer())
- return vm.throw_completion<TypeError>(ErrorType::SharedArrayBuffer);
- // 19. If IsDetachedBuffer(new) is true, throw a TypeError exception.
- if (new_array_buffer_object->is_detached())
- return vm.throw_completion<TypeError>(ErrorType::SpeciesConstructorReturned, "a detached ArrayBuffer");
- // 20. If SameValue(new, O) is true, throw a TypeError exception.
- if (same_value(new_array_buffer_object, array_buffer_object))
- return vm.throw_completion<TypeError>(ErrorType::SpeciesConstructorReturned, "same ArrayBuffer instance");
- // 21. If new.[[ArrayBufferByteLength]] < newLen, throw a TypeError exception.
- if (new_array_buffer_object->byte_length() < new_length)
- return vm.throw_completion<TypeError>(ErrorType::SpeciesConstructorReturned, "an ArrayBuffer smaller than requested");
- // 22. NOTE: Side-effects of the above steps may have detached or resized O.
- // 23. If IsDetachedBuffer(O) is true, throw a TypeError exception.
- if (array_buffer_object->is_detached())
- return vm.throw_completion<TypeError>(ErrorType::DetachedArrayBuffer);
- // 24. Let fromBuf be O.[[ArrayBufferData]].
- auto& from_buf = array_buffer_object->buffer();
- // 25. Let toBuf be new.[[ArrayBufferData]].
- auto& to_buf = new_array_buffer_object->buffer();
- // 26. Let currentLen be O.[[ArrayBufferByteLength]].
- auto current_length = array_buffer_object->byte_length();
- // 27. If first < currentLen, then
- if (first < current_length) {
- // a. Let count be min(newLen, currentLen - first).
- auto count = min(new_length, current_length - first);
- // b. Perform CopyDataBlockBytes(toBuf, 0, fromBuf, first, count).
- copy_data_block_bytes(to_buf, 0, from_buf, first, count);
- }
- // 28. Return new.
- return new_array_buffer_object;
- }
- // 25.1.6.8 ArrayBuffer.prototype.transfer ( [ newLength ] ), https://tc39.es/ecma262/#sec-arraybuffer.prototype.transfer
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::transfer)
- {
- // 1. Let O be the this value.
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 2. Return ? ArrayBufferCopyAndDetach(O, newLength, PRESERVE-RESIZABILITY).
- auto new_length = vm.argument(0);
- return TRY(array_buffer_copy_and_detach(vm, array_buffer_object, new_length, PreserveResizability::PreserveResizability));
- }
- // 25.1.6.9 ArrayBuffer.prototype.transferToFixedLength ( [ newLength ] ), https://tc39.es/ecma262/#sec-arraybuffer.prototype.transfertofixedlength
- JS_DEFINE_NATIVE_FUNCTION(ArrayBufferPrototype::transfer_to_fixed_length)
- {
- // 1. Let O be the this value.
- auto array_buffer_object = TRY(typed_this_value(vm));
- // 2. Return ? ArrayBufferCopyAndDetach(O, newLength, FIXED-LENGTH).
- auto new_length = vm.argument(0);
- return TRY(array_buffer_copy_and_detach(vm, array_buffer_object, new_length, PreserveResizability::FixedLength));
- }
- }
|