chromium/third_party/blink/web_tests/external/wpt/wasm/jsapi/gc/i31.tentative.any.js

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