v8/test/mjsunit/wasm/reference-globals.js
Matthias Liedtke 63261c2608 [wasm-gc] Disallow anyref in JS interop
Also fix eqref/i31ref fromJS() handling to accept unwrapped Smis.
This does not convert HeapNumbers to Smis if they fit.

Bug: v8:7748
Change-Id: Ida70a826f9541b7f3fbe9eecbb2b4fe362b5ef70
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3829477
Commit-Queue: Matthias Liedtke <mliedtke@chromium.org>
Reviewed-by: Jakob Kummerow <jkummerow@chromium.org>
Cr-Commit-Position: refs/heads/main@{#82558}
2022-08-18 10:47:58 +00:00

383 lines
13 KiB
JavaScript

// Copyright 2020 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: --experimental-wasm-gc
d8.file.execute("test/mjsunit/wasm/wasm-module-builder.js");
(function TestReferenceGlobals() {
print(arguments.callee.name);
var exporting_instance = (function() {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_ii);
var wrong_sig_index = builder.addType(kSig_i_i);
var addition_index = builder.addFunction("addition", sig_index)
.addBody([kExprLocalGet, 0, kExprLocalGet, 1, kExprI32Add])
.exportFunc();
builder.addGlobal(wasmRefType(sig_index), false,
[kExprRefFunc, addition_index.index])
.exportAs("global");
builder.addGlobal(wasmRefNullType(wrong_sig_index), false)
.exportAs("mistyped_global");
return builder.instantiate({});
})();
// Mistyped imported global.
assertThrows(
() => {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_ii);
builder.addImportedGlobal("imports", "global", wasmRefNullType(sig_index),
false);
builder.instantiate(
{imports: { global: exporting_instance.exports.mistyped_global }})},
WebAssembly.LinkError,
/imported global does not match the expected type/
);
// Mistyped imported global due to cross-module typechecking.
assertThrows(
() => {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_i);
builder.addImportedGlobal("imports", "global", wasmRefNullType(sig_index),
false);
builder.instantiate(
{imports: { global: exporting_instance.exports.global }})},
WebAssembly.LinkError,
/imported global does not match the expected type/
);
// Non-function imported into function-typed global.
assertThrows(
() => {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_ii);
builder.addImportedGlobal("imports", "global", wasmRefNullType(sig_index),
false);
builder.instantiate({imports: { global: 42 }})},
WebAssembly.LinkError,
/function-typed object must be null \(if nullable\) or a Wasm function object/
);
// Mistyped function import.
assertThrows(
() => {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_i);
builder.addImportedGlobal("imports", "global", wasmRefType(sig_index),
false);
builder.instantiate(
{imports: { global: exporting_instance.exports.addition }})},
WebAssembly.LinkError,
/assigned exported function has to be a subtype of the expected type/
);
var instance = (function () {
var builder = new WasmModuleBuilder();
var sig_index = builder.addType(kSig_i_ii);
builder.addImportedGlobal("imports", "global", wasmRefNullType(sig_index),
false);
builder.addFunction("test_import", kSig_i_ii)
.addBody([kExprLocalGet, 0, kExprLocalGet, 1, kExprGlobalGet, 0,
kExprCallRef])
.exportFunc();
return builder.instantiate({imports: {
global: exporting_instance.exports.global
}});
})();
// This module is valid.
assertFalse(instance === undefined);
assertFalse(instance === null);
assertFalse(instance === 0);
// The correct function reference has been passed.
assertEquals(66, instance.exports.test_import(42, 24));
})();
(function TestStructInitExpr() {
print(arguments.callee.name);
var builder = new WasmModuleBuilder();
var struct_index = builder.addStruct([{type: kWasmI32, mutability: true}]);
var composite_struct_index = builder.addStruct(
[{type: kWasmI32, mutability: true},
{type: wasmRefNullType(struct_index), mutability: true},
{type: kWasmI8, mutability: true}]);
let field1_value = 432;
let field2_value = -123;
let field3_value = -555;
var global0 = builder.addGlobal(
wasmRefType(struct_index), false,
[...wasmI32Const(field2_value),
kGCPrefix, kExprStructNew, struct_index]);
var global = builder.addGlobal(
wasmRefType(composite_struct_index), false,
[...wasmI32Const(field1_value), kExprGlobalGet, global0.index,
...wasmI32Const(field3_value),
kGCPrefix, kExprStructNew, composite_struct_index]);
var global_default = builder.addGlobal(
wasmRefType(composite_struct_index), false,
[kGCPrefix, kExprStructNewDefault, composite_struct_index]);
builder.addFunction("field_1", kSig_i_v)
.addBody([
kExprGlobalGet, global.index,
kGCPrefix, kExprStructGet, composite_struct_index, 0
])
.exportFunc();
builder.addFunction("field_2", kSig_i_v)
.addBody([
kExprGlobalGet, global.index,
kGCPrefix, kExprStructGet, composite_struct_index, 1,
kGCPrefix, kExprStructGet, struct_index, 0
])
.exportFunc();
builder.addFunction("field_3", kSig_i_v)
.addBody([
kExprGlobalGet, global.index,
kGCPrefix, kExprStructGetS, composite_struct_index, 2])
.exportFunc();
builder.addFunction("field_1_default", kSig_i_v)
.addBody([
kExprGlobalGet, global_default.index,
kGCPrefix, kExprStructGet, composite_struct_index, 0])
.exportFunc();
builder.addFunction("field_2_default", makeSig([], [kWasmDataRef]))
.addBody([
kExprGlobalGet, global_default.index,
kGCPrefix, kExprStructGet, composite_struct_index, 1])
.exportFunc();
builder.addFunction("field_3_default", kSig_i_v)
.addBody([
kExprGlobalGet, global_default.index,
kGCPrefix, kExprStructGetS, composite_struct_index, 2])
.exportFunc();
var instance = builder.instantiate({});
assertEquals(field1_value, instance.exports.field_1());
assertEquals(field2_value, instance.exports.field_2());
assertEquals((field3_value << 24) >> 24, instance.exports.field_3());
assertEquals(0, instance.exports.field_1_default());
assertEquals(null, instance.exports.field_2_default());
assertEquals(0, instance.exports.field_3_default());
})();
(function TestArrayNewFixedExprNumeric() {
print(arguments.callee.name);
var builder = new WasmModuleBuilder();
var array_index = builder.addArray(kWasmI16, true);
let element0_value = -44;
let element1_value = 55;
var global0 = builder.addGlobal(
kWasmI32, false,
wasmI32Const(element0_value));
var global = builder.addGlobal(
wasmRefType(array_index), false,
[kExprGlobalGet, global0.index, ...wasmI32Const(element1_value),
kGCPrefix, kExprArrayNewFixedStatic, array_index, 2]);
builder.addFunction("get_element", kSig_i_i)
.addBody([
kExprGlobalGet, global.index,
kExprLocalGet, 0,
kGCPrefix, kExprArrayGetS, array_index])
.exportFunc();
var instance = builder.instantiate({});
assertEquals(element0_value, instance.exports.get_element(0));
assertEquals(element1_value, instance.exports.get_element(1));
})();
(function TestArrayNewFixedExprRef() {
print(arguments.callee.name);
var builder = new WasmModuleBuilder();
var struct_index = builder.addStruct([{type: kWasmI32, mutability: false}]);
var array_index = builder.addArray(wasmRefNullType(struct_index), true);
let element0_value = 44;
let element2_value = 55;
var global0 = builder.addGlobal(
wasmRefType(struct_index), false,
[...wasmI32Const(element0_value),
kGCPrefix, kExprStructNew, struct_index]);
var global = builder.addGlobal(
wasmRefType(array_index), false,
[kExprGlobalGet, global0.index, kExprRefNull, struct_index,
...wasmI32Const(element2_value),
kGCPrefix, kExprStructNew, struct_index,
kGCPrefix, kExprArrayNewFixedStatic, array_index, 3]);
builder.addFunction("element0", kSig_i_v)
.addBody([
kExprGlobalGet, global.index,
kExprI32Const, 0,
kGCPrefix, kExprArrayGet, array_index,
kGCPrefix, kExprStructGet, struct_index, 0])
.exportFunc();
builder.addFunction("element1", makeSig([], [kWasmDataRef]))
.addBody([
kExprGlobalGet, global.index,
kExprI32Const, 1,
kGCPrefix, kExprArrayGet, array_index])
.exportFunc();
builder.addFunction("element2", kSig_i_v)
.addBody([
kExprGlobalGet, global.index,
kExprI32Const, 2,
kGCPrefix, kExprArrayGet, array_index,
kGCPrefix, kExprStructGet, struct_index, 0])
.exportFunc();
var instance = builder.instantiate({});
assertEquals(element0_value, instance.exports.element0());
assertEquals(null, instance.exports.element1());
assertEquals(element2_value, instance.exports.element2());
})();
(function TestArrayNew() {
print(arguments.callee.name);
var builder = new WasmModuleBuilder();
var struct_index = builder.addStruct([makeField(kWasmI64, true)]);
var array_num_index = builder.addArray(kWasmI64, true);
var array_ref_index = builder.addArray(wasmRefNullType(struct_index), true);
let elem1 = -44;
let elem2 = 15;
let length = 20;
let global_elem_1 = builder.addGlobal(kWasmI64, false, wasmI64Const(elem1));
let global_elem_2 = builder.addGlobal(kWasmI64, false, wasmI64Const(elem2));
let global_length = builder.addGlobal(kWasmI32, false, wasmI32Const(length));
var global_array_1 = builder.addGlobal(
wasmRefType(array_num_index), false,
[kExprGlobalGet, global_elem_1.index,
kExprGlobalGet, global_length.index,
kGCPrefix, kExprArrayNew, array_num_index]);
var global_array_2 = builder.addGlobal(
wasmRefType(array_ref_index), false,
[kExprGlobalGet, global_elem_2.index,
kGCPrefix, kExprStructNew, struct_index,
kExprGlobalGet, global_length.index,
kGCPrefix, kExprArrayNew, array_ref_index]);
builder.addFunction("get_elements", kSig_l_i)
.addBody([
kExprGlobalGet, global_array_1.index,
kExprLocalGet, 0,
kGCPrefix, kExprArrayGet, array_num_index,
kExprGlobalGet, global_array_2.index,
kExprLocalGet, 0,
kGCPrefix, kExprArrayGet, array_ref_index,
kGCPrefix, kExprStructGet, struct_index, 0,
kExprI64Add])
.exportFunc();
var instance = builder.instantiate({});
let result = BigInt(elem1 + elem2);
assertEquals(result, instance.exports.get_elements(0));
assertEquals(result, instance.exports.get_elements(length / 2));
assertEquals(result, instance.exports.get_elements(length - 1));
assertTraps(kTrapArrayOutOfBounds,
() => instance.exports.get_elements(length));
})();
(function TestArrayNewArrayTooLarge() {
print(arguments.callee.name);
var builder = new WasmModuleBuilder();
var array_num_index = builder.addArray(kWasmI64, true);
builder.addGlobal(
wasmRefType(array_num_index), false,
[...wasmI32Const(0x8ffffff),
kGCPrefix, kExprArrayNewDefault, array_num_index]);
assertTraps(kTrapArrayTooLarge, () => builder.instantiate({}));
})();
(function TestI31RefConstantExpr() {
print(arguments.callee.name);
let builder = new WasmModuleBuilder();
let array_index = builder.addArray(kWasmI31Ref, false);
let values = [0, 10, -22, 0x7fffffff, -1];
let global = builder.addGlobal(
wasmRefType(array_index), true,
[...values.flatMap(
value => [...wasmI32Const(value), kGCPrefix, kExprI31New]),
kGCPrefix, kExprArrayNewFixedStatic, array_index, 5]);
for (signed of [true, false]) {
builder.addFunction(`get_${signed ? "s" : "u"}`, kSig_i_i)
.addBody([kExprGlobalGet, global.index,
kExprLocalGet, 0, kGCPrefix, kExprArrayGet, array_index,
kGCPrefix, signed ? kExprI31GetS : kExprI31GetU])
.exportFunc();
}
let instance = builder.instantiate();
assertEquals(values[0], instance.exports.get_s(0));
assertEquals(values[1], instance.exports.get_s(1));
assertEquals(values[2], instance.exports.get_s(2));
assertEquals(values[3] | 0x80000000, instance.exports.get_s(3));
assertEquals(values[4], instance.exports.get_s(4));
assertEquals(values[0], instance.exports.get_u(0));
assertEquals(values[1], instance.exports.get_u(1));
assertEquals(values[2] & 0x7fffffff, instance.exports.get_u(2));
assertEquals(values[3], instance.exports.get_u(3));
assertEquals(values[4] & 0x7fffffff, instance.exports.get_u(4));
})();
(function TestI31RefConstantExprTypeError() {
print(arguments.callee.name);
let builder = new WasmModuleBuilder();
builder.addGlobal(kWasmI31Ref, false,
[...wasmI64Const(0), kGCPrefix, kExprI31New]);
assertThrows(() => builder.instantiate(), WebAssembly.CompileError,
/i31.new\[0\] expected type i32, found i64.const of type i64/);
})();