[fastcall] Add more TypedArray tests for fallback cases

This CL adds the following specific cases, to make sure they are
correctly handled by the slow path:
 - the backing store of a TypedArray gets detached after optimisation
 - passing null instead of a TA doesn't lead to a deopt

Bug: v8:11739, chromium:1052746
Change-Id: I7dfd3da9f535831901998ca6fad854af6e93e9f9
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3320425
Reviewed-by: Michael Stanton <mvstanton@chromium.org>
Commit-Queue: Maya Lekova <mslekova@chromium.org>
Cr-Commit-Position: refs/heads/main@{#78274}
This commit is contained in:
Maya Lekova 2021-12-07 14:12:22 +01:00 committed by V8 LUCI CQ
parent fd23e91464
commit 94f86e6d4e

View File

@ -50,7 +50,7 @@ const fast_c_api = new d8.test.FastCAPI();
// Test with TypedArray. // Test with TypedArray.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
let typed_array = new Uint32Array([1,2,3]); let typed_array = new Uint32Array([1,2,3]);
return fast_c_api.add_all_overload(false /* should_fallback */, return fast_c_api.add_all_overload(false /* should_fallback */,
typed_array); typed_array);
@ -65,7 +65,7 @@ for (let i = 0; i < 100; i++) {
// Non-externalized TypedArray. // Non-externalized TypedArray.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
let typed_array = new Uint32Array(large_array); let typed_array = new Uint32Array(large_array);
return fast_c_api.add_all_overload(false /* should_fallback */, return fast_c_api.add_all_overload(false /* should_fallback */,
typed_array); typed_array);
@ -75,7 +75,7 @@ for (let i = 0; i < 100; i++) {
// Mismatched TypedArray. // Mismatched TypedArray.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
let typed_array = new Float32Array([1.1, 2.2, 3.3]); let typed_array = new Float32Array([1.1, 2.2, 3.3]);
return fast_c_api.add_all_overload(false /* should_fallback */, return fast_c_api.add_all_overload(false /* should_fallback */,
typed_array); typed_array);
@ -85,7 +85,7 @@ for (let i = 0; i < 100; i++) {
// Test with JSArray. // Test with JSArray.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
let js_array = [26, -6, 42]; let js_array = [26, -6, 42];
return fast_c_api.add_all_overload(false /* should_fallback */, js_array); return fast_c_api.add_all_overload(false /* should_fallback */, js_array);
} }
@ -94,7 +94,7 @@ for (let i = 0; i < 100; i++) {
// Test function overloads with undefined. // Test function overloads with undefined.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
return fast_c_api.add_all_overload(false /* should_fallback */, undefined); return fast_c_api.add_all_overload(false /* should_fallback */, undefined);
} }
ExpectSlowCall(overloaded_test, 0); ExpectSlowCall(overloaded_test, 0);
@ -102,7 +102,7 @@ for (let i = 0; i < 100; i++) {
// Test function with invalid overloads. // Test function with invalid overloads.
(function () { (function () {
function overloaded_test(should_fallback = false) { function overloaded_test() {
return fast_c_api.add_all_invalid_overload(false /* should_fallback */, return fast_c_api.add_all_invalid_overload(false /* should_fallback */,
[26, -6, 42]); [26, -6, 42]);
} }
@ -124,13 +124,13 @@ for (let i = 0; i < 100; i++) {
assertEquals(0, fast_c_api.fast_call_count()); assertEquals(0, fast_c_api.fast_call_count());
})(); })();
//----------- Test different TypedArray functions. ----------- // ----------- Test different TypedArray functions. -----------
// ----------- add_all_<TYPE>_typed_array ----------- // ----------- add_all_<TYPE>_typed_array -----------
// `add_all_<TYPE>_typed_array` have the following signature: // `add_all_<TYPE>_typed_array` have the following signature:
// double add_all_<TYPE>_typed_array(bool /*should_fallback*/, FastApiTypedArray<TYPE>) // double add_all_<TYPE>_typed_array(bool /*should_fallback*/, FastApiTypedArray<TYPE>)
(function () { (function () {
function int32_test(should_fallback = false) { function int32_test() {
let typed_array = new Int32Array([-42, 1, 2, 3]); let typed_array = new Int32Array([-42, 1, 2, 3]);
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */, return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
typed_array); typed_array);
@ -139,7 +139,7 @@ for (let i = 0; i < 100; i++) {
})(); })();
(function () { (function () {
function uint32_test(should_fallback = false) { function uint32_test() {
let typed_array = new Uint32Array([1, 2, 3]); let typed_array = new Uint32Array([1, 2, 3]);
return fast_c_api.add_all_uint32_typed_array(false /* should_fallback */, return fast_c_api.add_all_uint32_typed_array(false /* should_fallback */,
typed_array); typed_array);
@ -148,7 +148,7 @@ for (let i = 0; i < 100; i++) {
})(); })();
(function () { (function () {
function float32_test(should_fallback = false) { function float32_test() {
let typed_array = new Float32Array([1.3, 2.4, 3.5]); let typed_array = new Float32Array([1.3, 2.4, 3.5]);
return fast_c_api.add_all_float32_typed_array(false /* should_fallback */, return fast_c_api.add_all_float32_typed_array(false /* should_fallback */,
typed_array); typed_array);
@ -161,7 +161,7 @@ for (let i = 0; i < 100; i++) {
})(); })();
(function () { (function () {
function float64_test(should_fallback = false) { function float64_test() {
let typed_array = new Float64Array([1.3, 2.4, 3.5]); let typed_array = new Float64Array([1.3, 2.4, 3.5]);
return fast_c_api.add_all_float64_typed_array(false /* should_fallback */, return fast_c_api.add_all_float64_typed_array(false /* should_fallback */,
typed_array); typed_array);
@ -173,8 +173,10 @@ for (let i = 0; i < 100; i++) {
} }
})(); })();
// ----------- TypedArray tests in various conditions -----------
// Detached backing store.
(function () { (function () {
function detached_typed_array_test(should_fallback = false) { function detached_typed_array_test() {
let typed_array = new Int32Array([-42, 1, 2, 3]); let typed_array = new Int32Array([-42, 1, 2, 3]);
%ArrayBufferDetach(typed_array.buffer); %ArrayBufferDetach(typed_array.buffer);
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */, return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
@ -183,8 +185,9 @@ for (let i = 0; i < 100; i++) {
ExpectSlowCall(detached_typed_array_test, 0); ExpectSlowCall(detached_typed_array_test, 0);
})(); })();
// Detached, non-externalised backing store.
(function () { (function () {
function detached_non_ext_typed_array_test(should_fallback = false) { function detached_non_ext_typed_array_test() {
let typed_array = new Int32Array(large_array); let typed_array = new Int32Array(large_array);
%ArrayBufferDetach(typed_array.buffer); %ArrayBufferDetach(typed_array.buffer);
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */, return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
@ -193,8 +196,41 @@ for (let i = 0; i < 100; i++) {
ExpectSlowCall(detached_non_ext_typed_array_test, 0); ExpectSlowCall(detached_non_ext_typed_array_test, 0);
})(); })();
// Detaching the backing store after the function is optimised.
(function() {
let typed_array = new Int32Array([-42, 1, 2, 3]);
let expected = -36;
function detached_later_typed_array_test() {
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
typed_array);
}
%PrepareFunctionForOptimization(detached_later_typed_array_test);
let result = detached_later_typed_array_test();
assertEquals(expected, result);
fast_c_api.reset_counts();
%OptimizeFunctionOnNextCall(detached_later_typed_array_test);
result = detached_later_typed_array_test();
assertEquals(expected, result);
assertOptimized(detached_later_typed_array_test);
assertEquals(1, fast_c_api.fast_call_count());
assertEquals(0, fast_c_api.slow_call_count());
fast_c_api.reset_counts();
// Detaching the backing store after the function was optimised should
// not lead to a deopt, but rather follow the slow path.
%ArrayBufferDetach(typed_array.buffer);
result = detached_later_typed_array_test();
assertOptimized(detached_later_typed_array_test);
assertEquals(0, fast_c_api.fast_call_count());
assertEquals(1, fast_c_api.slow_call_count());
assertEquals(0, result);
})();
// Externalised backing store.
(function () { (function () {
function shared_array_buffer_ta_test(should_fallback = false) { function shared_array_buffer_ta_test() {
let sab = new SharedArrayBuffer(16); let sab = new SharedArrayBuffer(16);
let typed_array = new Int32Array(sab); let typed_array = new Int32Array(sab);
typed_array.set([-42, 1, 2, 3]); typed_array.set([-42, 1, 2, 3]);
@ -204,8 +240,9 @@ for (let i = 0; i < 100; i++) {
ExpectSlowCall(shared_array_buffer_ta_test, -36); ExpectSlowCall(shared_array_buffer_ta_test, -36);
})(); })();
// Externalised backing store.
(function () { (function () {
function shared_array_buffer_ext_ta_test(should_fallback = false) { function shared_array_buffer_ext_ta_test() {
let sab = new SharedArrayBuffer(400); let sab = new SharedArrayBuffer(400);
let typed_array = new Int32Array(sab); let typed_array = new Int32Array(sab);
typed_array.set(large_array); typed_array.set(large_array);
@ -217,7 +254,7 @@ for (let i = 0; i < 100; i++) {
// Empty TypedArray. // Empty TypedArray.
(function () { (function () {
function int32_test(should_fallback = false) { function int32_test() {
let typed_array = new Int32Array(0); let typed_array = new Int32Array(0);
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */, return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
typed_array); typed_array);
@ -240,3 +277,16 @@ for (let i = 0; i < 100; i++) {
assert_throws_and_optimized(invalid_test, 'string'); assert_throws_and_optimized(invalid_test, 'string');
assert_throws_and_optimized(invalid_test, Symbol()); assert_throws_and_optimized(invalid_test, Symbol());
})(); })();
// Passing `null` instead of a TypedArray in a non-overloaded function.
(function () {
function null_test(arg) {
return fast_c_api.add_all_int32_typed_array(false /* should_fallback */,
arg);
}
%PrepareFunctionForOptimization(null_test);
null_test(new Int32Array(0));
%OptimizeFunctionOnNextCall(null_test);
assert_throws_and_optimized(null_test, null);
})();