// META: global=window,dedicatedworker,jsshell // META: script=/wasm/jsapi/wasm-module-builder.js let exports = {}; setup(() => { const builder = new WasmModuleBuilder(); const i31Ref = wasmRefType(kWasmI31Ref); const i31NullableRef = wasmRefNullType(kWasmI31Ref); const anyRef = wasmRefType(kWasmAnyRef); builder .addFunction("makeI31", makeSig_r_x(i31Ref, kWasmI32)) .addBody([kExprLocalGet, 0, ...GCInstr(kExprI31New)]) .exportFunc(); builder .addFunction("castI31", makeSig_r_x(kWasmI32, anyRef)) .addBody([kExprLocalGet, 0, ...GCInstr(kExprRefCast), kI31RefCode, ...GCInstr(kExprI31GetU)]) .exportFunc(); builder .addFunction("getI31", makeSig_r_x(kWasmI32, i31Ref)) .addBody([kExprLocalGet, 0, ...GCInstr(kExprI31GetS)]) .exportFunc(); builder .addFunction("argI31", makeSig_v_x(i31NullableRef)) .addBody([]) .exportFunc(); builder .addGlobal(i31NullableRef, true, [...wasmI32Const(0), ...GCInstr(kExprI31New)]) builder .addExportOfKind("i31Global", kExternalGlobal, 0); builder .addTable(i31NullableRef, 10) builder .addExportOfKind("i31Table", kExternalTable, 0); const buffer = builder.toBuffer(); const module = new WebAssembly.Module(buffer); const instance = new WebAssembly.Instance(module, {}); exports = instance.exports; }); test(() => { assert_equals(exports.makeI31(42), 42); assert_equals(exports.makeI31(2 ** 30 - 1), 2 ** 30 - 1); assert_equals(exports.makeI31(2 ** 30), -(2 ** 30)); assert_equals(exports.makeI31(-(2 ** 30)), -(2 ** 30)); assert_equals(exports.makeI31(2 ** 31 - 1), -1); assert_equals(exports.makeI31(2 ** 31), 0); }, "i31ref conversion to Number"); test(() => { assert_equals(exports.getI31(exports.makeI31(42)), 42); assert_equals(exports.getI31(42), 42); assert_equals(exports.getI31(2.0 ** 30 - 1), 2 ** 30 - 1); assert_equals(exports.getI31(-(2 ** 30)), -(2 ** 30)); }, "Number conversion to i31ref"); test(() => { exports.argI31(null); assert_throws_js(TypeError, () => exports.argI31(2 ** 30)); assert_throws_js(TypeError, () => exports.argI31(-(2 ** 30) - 1)); assert_throws_js(TypeError, () => exports.argI31(2n)); assert_throws_js(TypeError, () => exports.argI31(() => 3)); assert_throws_js(TypeError, () => exports.argI31(exports.getI31)); }, "Check i31ref argument type"); test(() => { assert_equals(exports.castI31(42), 42); assert_equals(exports.castI31(2 ** 30 - 1), 2 ** 30 - 1); assert_throws_js(WebAssembly.RuntimeError, () => { exports.castI31(2 ** 30); }); assert_throws_js(WebAssembly.RuntimeError, () => { exports.castI31(-(2 ** 30) - 1); }); assert_throws_js(WebAssembly.RuntimeError, () => { exports.castI31(2 ** 32); }); }, "Numbers in i31 range are i31ref, not hostref"); test(() => { assert_equals(exports.i31Global.value, 0); exports.i31Global.value = 42; assert_throws_js(TypeError, () => exports.i31Global.value = 2 ** 30); assert_throws_js(TypeError, () => exports.i31Global.value = -(2 ** 30) - 1); assert_equals(exports.i31Global.value, 42); }, "i31ref global"); test(() => { assert_equals(exports.i31Table.get(0), null); exports.i31Table.set(0, 42); assert_throws_js(TypeError, () => exports.i31Table.set(0, 2 ** 30)); assert_throws_js(TypeError, () => exports.i31Table.set(0, -(2 ** 30) - 1)); assert_equals(exports.i31Table.get(0), 42); }, "i31ref table");