nodejs/deps/v8/test/mjsunit/wasm/deopt/deopt-multi-instance-call-indirect.js
Michaël Zasso 918fe04351
deps: update V8 to 13.6.233.8
PR-URL: https://github.com/nodejs/node/pull/58070
Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
Reviewed-By: Darshan Sen <raisinten@gmail.com>
Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com>
Reviewed-By: Rafael Gonzaga <rafael.nunu@hotmail.com>
2025-05-02 15:06:53 +02:00

87 lines
2.8 KiB
JavaScript

// Copyright 2024 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: --allow-natives-syntax --wasm-deopt --liftoff
// Flags: --wasm-inlining-call-indirect --wasm-inlining-ignore-call-counts
// See also inlining-multi-instance.js, from which this is adapted.
// This additionally checks that we don't run into a deopt loop.
d8.file.execute("test/mjsunit/wasm/wasm-module-builder.js");
function setup() {
const kInitialTableSize = 1;
let sharedTable = new WebAssembly.Table({
element: "anyfunc",
initial: kInitialTableSize,
});
let builder = new WasmModuleBuilder();
let type_i_v = builder.addType(kSig_i_v);
builder.addImportedTable("env", "table", kInitialTableSize);
// We use this global's value to differentiate which instance we are executing.
let importedGlobal = builder.addImportedGlobal("env", "g", kWasmI32);
builder
.addFunction("callee", kSig_i_v)
.addBody([kExprGlobalGet, importedGlobal, ...wasmI32Const(5), kExprI32Add])
.exportFunc();
builder
.addFunction("call_indirect", type_i_v)
.addBody([
...wasmI32Const(/* index of funcref in table */ 0),
kExprCallIndirect, type_i_v, kTableZero,
])
.exportFunc();
let wasmModule = builder.toModule();
// Both instances share the same table, but have different global values to
// distinguish them.
let instance1 = new WebAssembly.Instance(wasmModule, {
env: {
g: 1000,
table: sharedTable,
},
});
let instance2 = new WebAssembly.Instance(wasmModule, {
env: {
g: 2000,
table: sharedTable,
},
});
// And the table contains a reference to the function of the first instance.
sharedTable.set(0, instance1.exports.callee);
return {instance1, instance2};
}
let {instance1, instance2} = setup();
const func1 = instance1.exports.call_indirect;
const func2 = instance2.exports.call_indirect;
// Same-instance call can be inlined.
assertEquals(1005, func1());
%WasmTierUpFunction(func1);
if (%IsWasmTieringPredictable()) {
assertTrue(%IsTurboFanFunction(func1));
// Since we share code between instances, this is optimized as well.
assertTrue(%IsTurboFanFunction(func2));
}
// Cross-instance call triggers a deopt (and returns the correct result).
assertEquals(1005, func2());
if (%IsWasmTieringPredictable()) {
assertFalse(%IsTurboFanFunction(func1));
assertFalse(%IsTurboFanFunction(func2));
}
// Re-optimizing after deopting due to non-inlineable target, we may not deopt
// again, otherwise we would have a deopt loop.
%WasmTierUpFunction(func2);
assertEquals(1005, func1());
assertEquals(1005, func2());
if (%IsWasmTieringPredictable()) {
assertTrue(%IsTurboFanFunction(func1));
assertTrue(%IsTurboFanFunction(func2));
}