| // 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"); |