2021-06-23 13:08:06 +00:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
// This file excercises sequences support for fast API calls.
|
|
|
|
|
|
|
|
// Flags: --turbo-fast-api-calls --allow-natives-syntax --opt
|
|
|
|
// --always-opt is disabled because we rely on particular feedback for
|
|
|
|
// optimizing to the fastest path.
|
|
|
|
// Flags: --no-always-opt
|
|
|
|
// The test relies on optimizing/deoptimizing at predictable moments, so
|
|
|
|
// it's not suitable for deoptimization fuzzing.
|
|
|
|
// Flags: --deopt-every-n-times=0
|
|
|
|
|
|
|
|
const fast_c_api = new d8.test.FastCAPI();
|
|
|
|
|
|
|
|
// ----------- add_all_sequence -----------
|
|
|
|
// `add_all_sequence` has the following signature:
|
|
|
|
// double add_all_sequence(bool /*should_fallback*/, Local<Array>)
|
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
const max_safe_float = 2**24 - 1;
|
|
|
|
const add_all_result_full = -42 + 45 +
|
|
|
|
Number.MIN_SAFE_INTEGER + Number.MAX_SAFE_INTEGER +
|
|
|
|
max_safe_float * 0.5 + Math.PI;
|
|
|
|
const full_array = [-42, 45,
|
|
|
|
Number.MIN_SAFE_INTEGER, Number.MAX_SAFE_INTEGER,
|
|
|
|
max_safe_float * 0.5, Math.PI];
|
2021-06-23 13:08:06 +00:00
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
function add_all_sequence_smi(arg) {
|
|
|
|
return fast_c_api.add_all_sequence(false /* should_fallback */, arg);
|
|
|
|
}
|
2021-06-23 13:08:06 +00:00
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
%PrepareFunctionForOptimization(add_all_sequence_smi);
|
|
|
|
assertEquals(3, add_all_sequence_smi([-42, 45]));
|
|
|
|
%OptimizeFunctionOnNextCall(add_all_sequence_smi);
|
|
|
|
|
|
|
|
function add_all_sequence_full(arg) {
|
|
|
|
return fast_c_api.add_all_sequence(false /* should_fallback */, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(add_all_sequence_full);
|
|
|
|
if (fast_c_api.supports_fp_params) {
|
|
|
|
assertEquals(add_all_result_full, add_all_sequence_full(full_array));
|
|
|
|
} else {
|
|
|
|
assertEquals(3, add_all_sequence_smi([-42, 45]));
|
|
|
|
}
|
|
|
|
%OptimizeFunctionOnNextCall(add_all_sequence_full);
|
|
|
|
|
|
|
|
if (fast_c_api.supports_fp_params) {
|
|
|
|
// Test that regular call hits the fast path.
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertEquals(add_all_result_full, add_all_sequence_full(full_array));
|
|
|
|
assertOptimized(add_all_sequence_full);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(0, fast_c_api.slow_call_count());
|
|
|
|
} else {
|
|
|
|
// Smi only test - regular call hits the fast path.
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertEquals(3, add_all_sequence_smi([-42, 45]));
|
|
|
|
assertOptimized(add_all_sequence_smi);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(0, fast_c_api.slow_call_count());
|
|
|
|
}
|
|
|
|
|
2021-07-20 14:29:16 +00:00
|
|
|
// Test holey arrays
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
if (fast_c_api.supports_fp_params) {
|
|
|
|
// Test that regular call hits the fast path.
|
|
|
|
assertEquals(add_all_result_full, add_all_sequence_full([, ...full_array]));
|
|
|
|
assertOptimized(add_all_sequence_full);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
|
|
|
} else {
|
|
|
|
// Smi only test - regular call hits the fast path.
|
|
|
|
assertEquals(3, add_all_sequence_smi([-42, , 45]));
|
|
|
|
assertOptimized(add_all_sequence_smi);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
function add_all_sequence_mismatch(arg) {
|
|
|
|
return fast_c_api.add_all_sequence(false /*should_fallback*/, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(add_all_sequence_mismatch);
|
|
|
|
assertThrows(() => add_all_sequence_mismatch());
|
|
|
|
%OptimizeFunctionOnNextCall(add_all_sequence_mismatch);
|
|
|
|
|
|
|
|
// Test that passing non-array arguments falls down the slow path.
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertThrows(() => add_all_sequence_mismatch(42));
|
|
|
|
assertOptimized(add_all_sequence_mismatch);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertThrows(() => add_all_sequence_mismatch({}));
|
|
|
|
assertOptimized(add_all_sequence_mismatch);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertThrows(() => add_all_sequence_mismatch('string'));
|
|
|
|
assertOptimized(add_all_sequence_mismatch);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
assertThrows(() => add_all_sequence_mismatch(Symbol()));
|
|
|
|
assertOptimized(add_all_sequence_mismatch);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
|
|
|
assertEquals(1, fast_c_api.slow_call_count());
|
2021-07-08 00:48:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
//----------- Test function overloads with same arity. -----------
|
|
|
|
//Only overloads between JSArray and TypedArray are supported
|
|
|
|
|
|
|
|
// Test with TypedArray.
|
|
|
|
(function () {
|
|
|
|
function overloaded_test(should_fallback = false) {
|
2021-07-16 13:09:57 +00:00
|
|
|
let typed_array = new Uint32Array([1, 2, 3]);
|
2021-07-08 00:48:40 +00:00
|
|
|
return fast_c_api.add_all_overload(false /* should_fallback */,
|
|
|
|
typed_array);
|
|
|
|
}
|
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
%PrepareFunctionForOptimization(overloaded_test);
|
|
|
|
let result = overloaded_test();
|
|
|
|
assertEquals(0, result);
|
2021-07-08 00:48:40 +00:00
|
|
|
|
2021-07-16 13:09:57 +00:00
|
|
|
fast_c_api.reset_counts();
|
|
|
|
%OptimizeFunctionOnNextCall(overloaded_test);
|
|
|
|
result = overloaded_test();
|
|
|
|
assertEquals(0, result);
|
|
|
|
assertOptimized(overloaded_test);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
2021-07-08 00:48:40 +00:00
|
|
|
})();
|
|
|
|
|
|
|
|
// Mismatched TypedArray.
|
|
|
|
(function () {
|
|
|
|
function overloaded_test(should_fallback = false) {
|
|
|
|
let typed_array = new Float32Array([1.1, 2.2, 3.3]);
|
|
|
|
return fast_c_api.add_all_overload(false /* should_fallback */,
|
|
|
|
typed_array);
|
|
|
|
}
|
2021-07-16 13:09:57 +00:00
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(overloaded_test);
|
|
|
|
let result = overloaded_test();
|
|
|
|
assertEquals(0, result);
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
%OptimizeFunctionOnNextCall(overloaded_test);
|
|
|
|
result = overloaded_test();
|
|
|
|
assertEquals(0, result);
|
|
|
|
assertOptimized(overloaded_test);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
2021-07-08 00:48:40 +00:00
|
|
|
})();
|
|
|
|
|
|
|
|
// Test with JSArray.
|
|
|
|
(function () {
|
|
|
|
function overloaded_test(should_fallback = false) {
|
|
|
|
let js_array = [26, -6, 42];
|
|
|
|
return fast_c_api.add_all_overload(false /* should_fallback */, js_array);
|
|
|
|
}
|
2021-07-16 13:09:57 +00:00
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(overloaded_test);
|
|
|
|
let result = overloaded_test();
|
|
|
|
assertEquals(62, result);
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
%OptimizeFunctionOnNextCall(overloaded_test);
|
|
|
|
result = overloaded_test();
|
|
|
|
assertEquals(62, result);
|
|
|
|
assertOptimized(overloaded_test);
|
|
|
|
assertEquals(1, fast_c_api.fast_call_count());
|
2021-07-08 00:48:40 +00:00
|
|
|
})();
|
|
|
|
|
|
|
|
// Test function overloads with undefined.
|
|
|
|
(function () {
|
|
|
|
function overloaded_test(should_fallback = false) {
|
|
|
|
return fast_c_api.add_all_overload(false /* should_fallback */, undefined);
|
|
|
|
}
|
2021-07-16 13:09:57 +00:00
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(overloaded_test);
|
|
|
|
assertThrows(() => overloaded_test());
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
%OptimizeFunctionOnNextCall(overloaded_test);
|
|
|
|
assertThrows(() => overloaded_test());
|
|
|
|
assertOptimized(overloaded_test);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
2021-07-08 00:48:40 +00:00
|
|
|
})();
|
|
|
|
|
|
|
|
// Test function with invalid overloads.
|
|
|
|
(function () {
|
|
|
|
function overloaded_test(should_fallback = false) {
|
|
|
|
return fast_c_api.add_all_invalid_overload(false /* should_fallback */,
|
|
|
|
[26, -6, 42]);
|
|
|
|
}
|
|
|
|
|
|
|
|
%PrepareFunctionForOptimization(overloaded_test);
|
|
|
|
result = overloaded_test();
|
|
|
|
assertEquals(62, result);
|
|
|
|
|
|
|
|
fast_c_api.reset_counts();
|
|
|
|
%OptimizeFunctionOnNextCall(overloaded_test);
|
|
|
|
result = overloaded_test();
|
|
|
|
assertEquals(62, result);
|
|
|
|
// Here we deopt because with this invalid overload:
|
|
|
|
// - add_all_int_invalid_func(Receiver, Bool, Int32, Options)
|
|
|
|
// - add_all_seq_c_func(Receiver, Bool, JSArray, Options)
|
|
|
|
// we expect that a number will be passed as 3rd argument
|
|
|
|
// (SimplifiedLowering takes the type from the first overloaded function).
|
|
|
|
assertUnoptimized(overloaded_test);
|
|
|
|
assertEquals(0, fast_c_api.fast_call_count());
|
|
|
|
})();
|