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>
107 lines
3.7 KiB
JavaScript
107 lines
3.7 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: --wasm-deopt --allow-natives-syntax --no-jit-fuzzing --liftoff
|
|
|
|
d8.file.execute('test/mjsunit/wasm/wasm-module-builder.js');
|
|
|
|
const builder = new WasmModuleBuilder();
|
|
let calleeSig = builder.addType(makeSig([], [kWasmI32]));
|
|
let mainSig = builder.addType(makeSig([wasmRefType(calleeSig)], [kWasmI32]));
|
|
let global = builder.addGlobal(kWasmI32, true).exportAs("global");
|
|
|
|
builder.addFunction("a", calleeSig)
|
|
.exportFunc()
|
|
.addBody([
|
|
kExprGlobalGet, global.index,
|
|
kExprI32Const, 1,
|
|
kExprI32Add,
|
|
]);
|
|
|
|
builder.addFunction("b", calleeSig)
|
|
.exportFunc()
|
|
.addBody([
|
|
kExprGlobalGet, global.index,
|
|
kExprI32Const, 2,
|
|
kExprI32Add,
|
|
]);
|
|
|
|
builder.addFunction("main", mainSig).exportFunc()
|
|
.addBody([
|
|
kExprGlobalGet, global.index,
|
|
kExprI32Const, 10,
|
|
kExprI32Mul,
|
|
kExprLocalGet, 0,
|
|
kExprCallRef, calleeSig,
|
|
kExprI32Add,
|
|
]);
|
|
|
|
const instanceA = builder.instantiate({});
|
|
const instanceB = builder.instantiate({});
|
|
instanceA.exports.global.value = 10;
|
|
instanceB.exports.global.value = 20;
|
|
|
|
// Expected output:
|
|
// main: global * 10 + callee
|
|
// callee:
|
|
// - a: global + 1
|
|
// - b: global + 2
|
|
// Therefore we expect a three digit number:
|
|
// - first digit: 1 for instanceA.main, 2 for instanceB.main
|
|
// - second digit: 1 for instanceA.(a|b), 2 for instanceB.(a|b)
|
|
// - third digit: 1 for a, 2 for b
|
|
|
|
assertEquals(111, instanceA.exports.main(instanceA.exports.a));
|
|
%WasmTierUpFunction(instanceA.exports.main);
|
|
assertEquals(111, instanceA.exports.main(instanceA.exports.a));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertTrue(%IsTurboFanFunction(instanceA.exports.main));
|
|
}
|
|
|
|
// This triggers a deopt as the callee is not function a and the optimized code
|
|
// is shared with instanceA.
|
|
assertEquals(222, instanceB.exports.main(instanceB.exports.b));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertFalse(%IsTurboFanFunction(instanceB.exports.main));
|
|
}
|
|
|
|
// This re-optimizes the function with the feedback from instanceB.
|
|
// To prevent deopt loops, it must also keep the feedback from instanceA.
|
|
%WasmTierUpFunction(instanceB.exports.main);
|
|
assertEquals(222, instanceB.exports.main(instanceB.exports.b));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertTrue(%IsTurboFanFunction(instanceA.exports.main));
|
|
}
|
|
|
|
// Calling a or b does not trigger a deopt.
|
|
assertEquals(111, instanceA.exports.main(instanceA.exports.a));
|
|
assertEquals(112, instanceA.exports.main(instanceA.exports.b));
|
|
assertEquals(221, instanceB.exports.main(instanceB.exports.a));
|
|
assertEquals(222, instanceB.exports.main(instanceB.exports.b));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertTrue(%IsTurboFanFunction(instanceA.exports.main));
|
|
assertTrue(%IsTurboFanFunction(instanceB.exports.main));
|
|
}
|
|
|
|
// Note that so far no cross-intance calls have been performed.
|
|
// Doing so will trigger a deoptimization as cross-instance calls are not
|
|
// inlineable.
|
|
assertEquals(122, instanceA.exports.main(instanceB.exports.b));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertFalse(%IsTurboFanFunction(instanceA.exports.main));
|
|
assertFalse(%IsTurboFanFunction(instanceB.exports.main));
|
|
}
|
|
|
|
// Tiering up again will not emit a deopt any more as the feedback contains a
|
|
// non-inlineable target.
|
|
%WasmTierUpFunction(instanceA.exports.main);
|
|
assertEquals(121, instanceA.exports.main(instanceB.exports.a));
|
|
assertEquals(122, instanceA.exports.main(instanceB.exports.b));
|
|
assertEquals(211, instanceB.exports.main(instanceA.exports.a));
|
|
assertEquals(212, instanceB.exports.main(instanceA.exports.b));
|
|
if (%IsWasmTieringPredictable()) {
|
|
assertTrue(%IsTurboFanFunction(instanceA.exports.main));
|
|
assertTrue(%IsTurboFanFunction(instanceB.exports.main));
|
|
}
|