From 36d22d82aa202bb199967e9512281e9a53db42c9 Mon Sep 17 00:00:00 2001 From: Daniel Baumann Date: Sun, 7 Apr 2024 21:33:14 +0200 Subject: Adding upstream version 115.7.0esr. Signed-off-by: Daniel Baumann --- .../tests/wasm/gc/externref-boxing-struct.js | 74 ++++++++++++++++++++++ 1 file changed, 74 insertions(+) create mode 100644 js/src/jit-test/tests/wasm/gc/externref-boxing-struct.js (limited to 'js/src/jit-test/tests/wasm/gc/externref-boxing-struct.js') diff --git a/js/src/jit-test/tests/wasm/gc/externref-boxing-struct.js b/js/src/jit-test/tests/wasm/gc/externref-boxing-struct.js new file mode 100644 index 0000000000..b9cfed080f --- /dev/null +++ b/js/src/jit-test/tests/wasm/gc/externref-boxing-struct.js @@ -0,0 +1,74 @@ +// |jit-test| skip-if: !wasmGcEnabled() + +// Moving a JS value through a wasm externref is a pair of boxing/unboxing +// conversions that leaves the value unchanged. There are many cases, +// along these axes: +// +// - global variables, see externref-boxing.js +// - tables, see externref-boxing.js +// - function parameters and returns, see externref-boxing.js +// - struct fields [for the gc feature], this file +// - TypedObject fields when we construct with the externref type; this file + +// Struct fields of externref type can receive their value in these ways: +// +// - the struct.new and struct.set instructions +// - storing into mutable fields from JS +// - from a constructor called from JS +// +// Struct fields can be read in these ways: +// +// - the struct.get instruction +// - reading the field from JS +// +// We're especially interested in two cases: where JS stores a non-object value +// into a field, in this case there should be boxing; and where JS reads a +// non-pointer value from a field, in this case there should be unboxing. + +// Write with struct.new, read with the JS getter + +for (let v of WasmExternrefValues) +{ + let ins = wasmEvalText( + `(module + (type $S (struct (field $S.x (mut externref)))) + (func (export "make") (param $v externref) (result eqref) + (struct.new $S (local.get $v))))`); + let x = ins.exports.make(v); + assertEq(x[0], v); +} + +// Try to make sure externrefs are properly traced + +{ + let fields = iota(10).map(() => `(field externref)`).join(' '); + let params = iota(10).map((i) => `(param $${i} externref)`).join(' '); + let args = iota(10).map((i) => `(local.get $${i})`).join(' '); + let txt = `(module + (type $S (struct ${fields})) + (func (export "make") ${params} (result eqref) + (struct.new $S ${args})))`; + let ins = wasmEvalText(txt); + let x = ins.exports.make({x:0}, {x:1}, {x:2}, {x:3}, {x:4}, {x:5}, {x:6}, {x:7}, {x:8}, {x:9}) + gc('shrinking'); + assertEq(typeof x[0], "object"); + assertEq(x[0].x, 0); + assertEq(typeof x[1], "object"); + assertEq(x[1].x, 1); + assertEq(typeof x[2], "object"); + assertEq(x[2].x, 2); + assertEq(typeof x[3], "object"); + assertEq(x[3].x, 3); + assertEq(typeof x[4], "object"); + assertEq(x[4].x, 4); + assertEq(typeof x[5], "object"); + assertEq(x[5].x, 5); + assertEq(typeof x[6], "object"); + assertEq(x[6].x, 6); + assertEq(typeof x[7], "object"); + assertEq(x[7].x, 7); + assertEq(typeof x[8], "object"); + assertEq(x[8].x, 8); + assertEq(typeof x[9], "object"); + assertEq(x[9].x, 9); +} -- cgit v1.2.3