3160edf011
Detailed list of changes: https://docs.google.com/document/d/15i4-SZDzFDW7FfclIYuZEhFn-q-KpobCBy23x9zZZLc/edit?usp=sharing Bug: v8:11111 Change-Id: I931003bd4552cf91d57de95af04a427a9e6d6ac9 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2814259 Reviewed-by: Igor Sheludko <ishell@chromium.org> Reviewed-by: Michael Lippautz <mlippautz@chromium.org> Reviewed-by: Jakob Kummerow <jkummerow@chromium.org> Reviewed-by: Shu-yu Guo <syg@chromium.org> Reviewed-by: Leszek Swirski <leszeks@chromium.org> Commit-Queue: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#74459}
431 lines
13 KiB
JavaScript
431 lines
13 KiB
JavaScript
// Copyright 2021 the V8 project authors. All rights reserved.
|
||
// Use of this source code is governed by a BSD-style license that can be
|
||
// found in the LICENSE file.
|
||
|
||
// Flags: --harmony-rab-gsab --allow-natives-syntax
|
||
|
||
"use strict";
|
||
|
||
function resizeHelper(ab, value) {
|
||
const return_value = ab.resize(value);
|
||
assertEquals(undefined, return_value);
|
||
assertEquals(value, ab.byteLength);
|
||
}
|
||
|
||
function growHelper(ab, value) {
|
||
const return_value = ab.grow(value);
|
||
assertEquals(undefined, return_value);
|
||
assertEquals(value, ab.byteLength);
|
||
}
|
||
|
||
(function TestRABBasics() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
assertTrue(rab instanceof ResizableArrayBuffer);
|
||
assertFalse(rab instanceof GrowableSharedArrayBuffer);
|
||
assertFalse(rab instanceof ArrayBuffer);
|
||
assertFalse(rab instanceof SharedArrayBuffer);
|
||
assertEquals(10, rab.byteLength);
|
||
assertEquals(20, rab.maxByteLength);
|
||
})();
|
||
|
||
(function TestRABCtorByteLengthEqualsMax() {
|
||
const rab = new ResizableArrayBuffer(10, 10);
|
||
assertEquals(10, rab.byteLength);
|
||
assertEquals(10, rab.maxByteLength);
|
||
})();
|
||
|
||
(function TestRABCtorByteLengthZero() {
|
||
const rab = new ResizableArrayBuffer(0, 10);
|
||
assertEquals(0, rab.byteLength);
|
||
assertEquals(10, rab.maxByteLength);
|
||
})();
|
||
|
||
(function TestRABCtorByteLengthAndMaxZero() {
|
||
const rab = new ResizableArrayBuffer(0, 0);
|
||
assertEquals(0, rab.byteLength);
|
||
assertEquals(0, rab.maxByteLength);
|
||
})();
|
||
|
||
(function TestRABCtorNoMaxByteLength() {
|
||
assertThrows(() => { new ResizableArrayBuffer(10); }, RangeError);
|
||
// But this is fine; undefined is converted to 0.
|
||
const rab = new ResizableArrayBuffer(0);
|
||
assertEquals(0, rab.byteLength);
|
||
assertEquals(0, rab.maxByteLength);
|
||
})();
|
||
|
||
(function TestAllocatingOutrageouslyMuchThrows() {
|
||
assertThrows(() => { new ResizableArrayBuffer(0, 2 ** 100);}, RangeError);
|
||
})();
|
||
|
||
(function TestRABCtorOperationOrder() {
|
||
let log = '';
|
||
const mock_length = {valueOf: function() {
|
||
log += 'valueof length, '; return 10; }};
|
||
const mock_max_length = {valueOf: function() {
|
||
log += 'valueof max_length, '; return 10; }};
|
||
new ResizableArrayBuffer(mock_length, mock_max_length);
|
||
|
||
assertEquals('valueof length, valueof max_length, ', log);
|
||
})();
|
||
|
||
(function TestGSABCtorOperationOrder() {
|
||
let log = '';
|
||
const mock_length = {valueOf: function() {
|
||
log += 'valueof length, '; return 10; }};
|
||
const mock_max_length = {valueOf: function() {
|
||
log += 'valueof max_length, '; return 10; }};
|
||
new ResizableArrayBuffer(mock_length, mock_max_length);
|
||
|
||
assertEquals('valueof length, valueof max_length, ', log);
|
||
})();
|
||
|
||
(function TestByteLengthGetterReceiverChecks() {
|
||
const name = 'byteLength';
|
||
const ab_getter = Object.getOwnPropertyDescriptor(
|
||
ArrayBuffer.prototype, name).get;
|
||
const sab_getter = Object.getOwnPropertyDescriptor(
|
||
SharedArrayBuffer.prototype, name).get;
|
||
const rab_getter = Object.getOwnPropertyDescriptor(
|
||
ResizableArrayBuffer.prototype, name).get;
|
||
const gsab_getter = Object.getOwnPropertyDescriptor(
|
||
GrowableSharedArrayBuffer.prototype, name).get;
|
||
|
||
const ab = new ArrayBuffer(40);
|
||
const sab = new SharedArrayBuffer(40);
|
||
const rab = new ResizableArrayBuffer(40, 40);
|
||
const gsab = new GrowableSharedArrayBuffer(40, 40);
|
||
|
||
assertEquals(40, ab_getter.call(ab));
|
||
assertEquals(40, sab_getter.call(sab));
|
||
assertEquals(40, rab_getter.call(rab));
|
||
assertEquals(40, gsab_getter.call(gsab));
|
||
|
||
assertThrows(() => { ab_getter.call(sab);});
|
||
assertThrows(() => { ab_getter.call(rab);});
|
||
assertThrows(() => { ab_getter.call(gsab);});
|
||
|
||
assertThrows(() => { sab_getter.call(ab);});
|
||
assertThrows(() => { sab_getter.call(rab);});
|
||
assertThrows(() => { sab_getter.call(gsab);});
|
||
|
||
assertThrows(() => { rab_getter.call(ab);});
|
||
assertThrows(() => { rab_getter.call(sab);});
|
||
assertThrows(() => { rab_getter.call(gsab);});
|
||
|
||
assertThrows(() => { gsab_getter.call(ab);});
|
||
assertThrows(() => { gsab_getter.call(sab);});
|
||
assertThrows(() => { gsab_getter.call(rab);});
|
||
})();
|
||
|
||
(function TestMaxByteLengthGetterReceiverChecks() {
|
||
const name = 'maxByteLength';
|
||
const rab_getter = Object.getOwnPropertyDescriptor(
|
||
ResizableArrayBuffer.prototype, name).get;
|
||
const gsab_getter = Object.getOwnPropertyDescriptor(
|
||
GrowableSharedArrayBuffer.prototype, name).get;
|
||
|
||
const ab = new ArrayBuffer(40);
|
||
const sab = new SharedArrayBuffer(40);
|
||
const rab = new ResizableArrayBuffer(20, 40);
|
||
const gsab = new GrowableSharedArrayBuffer(20, 40);
|
||
|
||
assertEquals(40, rab_getter.call(rab));
|
||
assertEquals(40, gsab_getter.call(gsab));
|
||
|
||
assertThrows(() => { rab_getter.call(ab);});
|
||
assertThrows(() => { rab_getter.call(sab);});
|
||
assertThrows(() => { rab_getter.call(gsab);});
|
||
|
||
assertThrows(() => { gsab_getter.call(ab);});
|
||
assertThrows(() => { gsab_getter.call(sab);});
|
||
assertThrows(() => { gsab_getter.call(rab);});
|
||
})();
|
||
|
||
(function TestResizeAndGrowReceiverChecks() {
|
||
const rab_resize = ResizableArrayBuffer.prototype.resize;
|
||
const gsab_grow = GrowableSharedArrayBuffer.prototype.grow;
|
||
|
||
const ab = new ArrayBuffer(40);
|
||
const sab = new SharedArrayBuffer(40);
|
||
const rab = new ResizableArrayBuffer(10, 40);
|
||
const gsab = new GrowableSharedArrayBuffer(10, 40);
|
||
|
||
rab_resize.call(rab, 20);
|
||
gsab_grow.call(gsab, 20);
|
||
assertThrows(() => { rab_resize.call(ab, 30);});
|
||
assertThrows(() => { rab_resize.call(sab, 30);});
|
||
assertThrows(() => { rab_resize.call(gsab, 30);});
|
||
|
||
assertThrows(() => { gsab_grow.call(ab, 30);});
|
||
assertThrows(() => { gsab_grow.call(sab, 30);});
|
||
assertThrows(() => { gsab_grow.call(rab, 30);});
|
||
})();
|
||
|
||
(function TestRABResizeToMax() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
resizeHelper(rab, 20);
|
||
})();
|
||
|
||
(function TestRABResizeToSameSize() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
resizeHelper(rab, 10);
|
||
})();
|
||
|
||
(function TestRABResizeToSmaller() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
resizeHelper(rab, 5);
|
||
})();
|
||
|
||
(function TestRABResizeToZero() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
resizeHelper(rab, 0);
|
||
})();
|
||
|
||
(function TestRABResizeZeroToZero() {
|
||
const rab = new ResizableArrayBuffer(0, 20);
|
||
resizeHelper(rab, 0);
|
||
})();
|
||
|
||
(function TestRABGrowBeyondMaxThrows() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
assertEquals(10, rab.byteLength);
|
||
assertThrows(() => {rab.grow(21)});
|
||
assertEquals(10, rab.byteLength);
|
||
})();
|
||
|
||
(function TestRABResizeMultipleTimes() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
const sizes = [15, 7, 7, 0, 8, 20, 20, 10];
|
||
for (let s of sizes) {
|
||
resizeHelper(rab, s);
|
||
}
|
||
})();
|
||
|
||
(function TestRABResizeParameters() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
rab.resize('15');
|
||
assertEquals(15, rab.byteLength);
|
||
rab.resize({valueOf: function() { return 16; }});
|
||
assertEquals(16, rab.byteLength);
|
||
rab.resize(17.9);
|
||
assertEquals(17, rab.byteLength);
|
||
})();
|
||
|
||
(function TestRABResizeInvalidParameters() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
assertThrows(() => { rab.resize(-1) }, RangeError);
|
||
assertThrows(() => { rab.resize({valueOf: function() {
|
||
throw new Error('length param'); }})});
|
||
assertEquals(10, rab.byteLength);
|
||
|
||
// Various non-numbers are converted to NaN which is treated as 0.
|
||
rab.resize('string');
|
||
assertEquals(0, rab.byteLength);
|
||
rab.resize();
|
||
assertEquals(0, rab.byteLength);
|
||
})();
|
||
|
||
(function TestRABResizeDetached() {
|
||
const rab = new ResizableArrayBuffer(10, 20);
|
||
%ArrayBufferDetach(rab);
|
||
assertThrows(() => { rab.resize(15) }, TypeError);
|
||
})();
|
||
|
||
(function DetachInsideResizeParameterConversion() {
|
||
const rab = new ResizableArrayBuffer(40, 80);
|
||
|
||
const evil = {
|
||
valueOf: () => { %ArrayBufferDetach(rab); return 20; }
|
||
};
|
||
|
||
assertThrows(() => { rab.resize(evil); });
|
||
})();
|
||
|
||
(function ResizeInsideResizeParameterConversion() {
|
||
const rab = new ResizableArrayBuffer(40, 80);
|
||
|
||
const evil = {
|
||
valueOf: () => { rab.resize(10); return 20; }
|
||
};
|
||
|
||
rab.resize(evil);
|
||
assertEquals(20, rab.byteLength);
|
||
})();
|
||
|
||
|
||
(function TestRABNewMemoryAfterResizeInitializedToZero() {
|
||
const maybe_page_size = 4096;
|
||
const rab = new ResizableArrayBuffer(maybe_page_size, 2 * maybe_page_size);
|
||
const i8a = new Int8Array(rab);
|
||
rab.resize(2 * maybe_page_size);
|
||
for (let i = 0; i < 2 * maybe_page_size; ++i) {
|
||
assertEquals(0, i8a[i]);
|
||
}
|
||
})();
|
||
|
||
(function TestRABMemoryInitializedToZeroAfterShrinkAndGrow() {
|
||
const maybe_page_size = 4096;
|
||
const rab = new ResizableArrayBuffer(maybe_page_size, 2 * maybe_page_size);
|
||
const i8a = new Int8Array(rab);
|
||
for (let i = 0; i < maybe_page_size; ++i) {
|
||
i8a[i] = 1;
|
||
}
|
||
rab.resize(maybe_page_size / 2);
|
||
rab.resize(maybe_page_size);
|
||
for (let i = maybe_page_size / 2; i < maybe_page_size; ++i) {
|
||
assertEquals(0, i8a[i]);
|
||
}
|
||
})();
|
||
|
||
(function TestGSABBasics() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertFalse(gsab instanceof ResizableArrayBuffer);
|
||
assertTrue(gsab instanceof GrowableSharedArrayBuffer);
|
||
assertFalse(gsab instanceof ArrayBuffer);
|
||
assertFalse(gsab instanceof SharedArrayBuffer);
|
||
assertEquals(10, gsab.byteLength);
|
||
assertEquals(20, gsab.maxByteLength);
|
||
})();
|
||
|
||
(function TestGSABCtorByteLengthEqualsMax() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 10);
|
||
assertEquals(10, gsab.byteLength);
|
||
assertEquals(10, gsab.maxByteLength);
|
||
})();
|
||
|
||
(function TestGSABCtorByteLengthZero() {
|
||
const gsab = new GrowableSharedArrayBuffer(0, 10);
|
||
assertEquals(0, gsab.byteLength);
|
||
assertEquals(10, gsab.maxByteLength);
|
||
})();
|
||
|
||
(function TestGSABCtorByteLengthAndMaxZero() {
|
||
const gsab = new GrowableSharedArrayBuffer(0, 0);
|
||
assertEquals(0, gsab.byteLength);
|
||
assertEquals(0, gsab.maxByteLength);
|
||
})();
|
||
|
||
(function TestGSABCtorNoMaxByteLength() {
|
||
assertThrows(() => { new GrowableSharedArrayBuffer(10); }, RangeError);
|
||
// But this is fine; undefined is converted to 0.
|
||
const gsab = new GrowableSharedArrayBuffer(0);
|
||
assertEquals(0, gsab.byteLength);
|
||
assertEquals(0, gsab.maxByteLength);
|
||
})();
|
||
|
||
(function TestAllocatingOutrageouslyMuchThrows() {
|
||
assertThrows(() => { new GrowableSharedArrayBuffer(0, 2 ** 100);},
|
||
RangeError);
|
||
})();
|
||
|
||
(function TestGSABGrowToMax() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
growHelper(gsab, 20);
|
||
})();
|
||
|
||
(function TestGSABGrowToSameSize() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
growHelper(gsab, 10);
|
||
})();
|
||
|
||
(function TestGSABGrowToSmallerThrows() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
assertThrows(() => {gsab.grow(5)});
|
||
assertEquals(10, gsab.byteLength);
|
||
})();
|
||
|
||
(function TestGSABGrowToZeroThrows() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
assertThrows(() => {gsab.grow(0)});
|
||
assertEquals(10, gsab.byteLength);
|
||
})();
|
||
|
||
(function TestGSABGrowBeyondMaxThrows() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
assertThrows(() => {gsab.grow(21)});
|
||
assertEquals(10, gsab.byteLength);
|
||
})();
|
||
|
||
(function TestGSABGrowMultipleTimes() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
const sizes = [15, 7, 7, 0, 8, 20, 20, 10];
|
||
for (let s of sizes) {
|
||
const current_size = gsab.byteLength;
|
||
if (s < gsab.byteLength) {
|
||
assertThrows(() => {gsab.grow(s)});
|
||
assertEquals(current_size, gsab.byteLength);
|
||
} else {
|
||
growHelper(gsab, s);
|
||
}
|
||
}
|
||
})();
|
||
|
||
(function TestGSABGrowParameters() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
gsab.grow('15');
|
||
assertEquals(15, gsab.byteLength);
|
||
gsab.grow({valueOf: function() { return 16; }});
|
||
assertEquals(16, gsab.byteLength);
|
||
gsab.grow(17.9);
|
||
assertEquals(17, gsab.byteLength);
|
||
})();
|
||
|
||
(function TestGSABGrowInvalidParameters() {
|
||
const gsab = new GrowableSharedArrayBuffer(0, 20);
|
||
assertThrows(() => { gsab.grow(-1) }, RangeError);
|
||
assertThrows(() => { gsab.grow({valueOf: function() {
|
||
throw new Error('length param'); }})});
|
||
assertEquals(0, gsab.byteLength);
|
||
|
||
// Various non-numbers are converted to NaN which is treated as 0.
|
||
gsab.grow('string');
|
||
assertEquals(0, gsab.byteLength);
|
||
gsab.grow();
|
||
assertEquals(0, gsab.byteLength);
|
||
})();
|
||
|
||
(function TestGSABMemoryInitializedToZeroAfterGrow() {
|
||
const maybe_page_size = 4096;
|
||
const gsab = new GrowableSharedArrayBuffer(maybe_page_size,
|
||
2 * maybe_page_size);
|
||
const i8a = new Int8Array(gsab);
|
||
gsab.grow(2 * maybe_page_size);
|
||
assertEquals(2 * maybe_page_size, i8a.length);
|
||
for (let i = 0; i < 2 * maybe_page_size; ++i) {
|
||
assertEquals(0, i8a[i]);
|
||
}
|
||
})();
|
||
|
||
(function GrowGSABOnADifferentThread() {
|
||
const gsab = new GrowableSharedArrayBuffer(10, 20);
|
||
assertEquals(10, gsab.byteLength);
|
||
function workerCode() {
|
||
function assert(thing) {
|
||
if (!thing) {
|
||
postMessage('error');
|
||
}
|
||
}
|
||
onmessage = function(params) {
|
||
const gsab = params.gsab;
|
||
assert(!(gsab instanceof ResizableArrayBuffer));
|
||
assert(gsab instanceof GrowableSharedArrayBuffer);
|
||
assert(!(gsab instanceof ArrayBuffer));
|
||
assert(!(gsab instanceof SharedArrayBuffer));
|
||
assert(10 == gsab.byteLength);
|
||
gsab.grow(15);
|
||
postMessage('ok');
|
||
}
|
||
}
|
||
const w = new Worker(workerCode, {type: 'function'});
|
||
w.postMessage({gsab: gsab});
|
||
assertEquals('ok', w.getMessage());
|
||
assertEquals(15, gsab.byteLength);
|
||
})();
|