Source code

Revision control

Copy as Markdown

Other Tools

// |jit-test| skip-if: true
// Common code to test simple binary operators. See runSimpleBinopTest below.
function expandConstantBinopInputs(op, memtype, inputs) {
let s = '';
let ident = 0;
for ( let [a, b] of inputs ) {
let constlhs = `${memtype.layoutName} ${a.map(jsValueToWasmName).join(' ')}`;
let constrhs = `${memtype.layoutName} ${b.map(jsValueToWasmName).join(' ')}`;
s += `
;; lhs is constant, rhs is variable
(func (export "run_constlhs${ident}")
(v128.store (i32.const 0)
(call $doit_constlhs${ident} (v128.const ${constrhs}))))
(func $doit_constlhs${ident} (param $b v128) (result v128)
(${op} (v128.const ${constlhs}) (local.get $b)))
;; rhs is constant, lhs is variable
(func (export "run_constrhs${ident}")
(v128.store (i32.const 0)
(call $doit_constrhs${ident} (v128.const ${constlhs}))))
(func $doit_constrhs${ident} (param $a v128) (result v128)
(${op} (local.get $a) (v128.const ${constrhs})))
;; both operands are constant
(func (export "run_constboth${ident}")
(v128.store (i32.const 0)
(call $doit_constboth${ident})))
(func $doit_constboth${ident} (result v128)
(${op} (v128.const ${constlhs}) (v128.const ${constrhs})))`
ident++;
}
return s;
}
function insAndMemBinop(op, memtype, resultmemtype, inputs) {
var ins = wasmEvalText(`
(module
(memory (export "mem") 1 1)
;; both arguments are variable
(func (export "run")
(v128.store (i32.const 0)
(call $doit (v128.load (i32.const 16)) (v128.load (i32.const 32)))))
(func $doit (param $a v128) (param $b v128) (result v128)
(${op} (local.get $a) (local.get $b)))
${expandConstantBinopInputs(op, memtype, inputs)})`);
var mem = new memtype(ins.exports.mem.buffer);
var resultmem = !resultmemtype || memtype == resultmemtype ? mem : new resultmemtype(ins.exports.mem.buffer);
return [ins, mem, resultmem];
}
function add(bits) { return (x, y) => sign_extend(x+y, bits) }
function add64(x, y) { return sign_extend(BigInt(x)+BigInt(y), 64) }
function sub(bits) { return (x, y) => sign_extend(x-y, bits) }
function sub64(x, y) { return sign_extend(BigInt(x)-BigInt(y), 64) }
// Even 32-bit multiply can overflow a Number, so always use BigInt
function mul(bits) { return (x, y) => sign_extend(BigInt(x)*BigInt(y), bits) }
function div(x, y) { return x/y }
function min(x, y) { return x < y ? x : y }
function max(x, y) { return x > y ? x : y }
function and(x, y) { return zero_extend(x&y, 8) }
function or(x, y) { return zero_extend(x|y, 8) }
function xor(x, y) { return zero_extend(x^y, 8) }
function andnot(x, y) { return zero_extend(x&~y, 8) }
function avgr(x, y) { return (x + y + 1) >> 1; }
function eq(truth) { return (x,y) => x==y ? truth : 0 }
function ne(truth) { return (x,y) => x!=y ? truth : 0 }
function lt(truth) { return (x, y) => x < y ? truth : 0 }
function gt(truth) { return (x, y) => x > y ? truth : 0 }
function le(truth) { return (x, y) => x <= y ? truth : 0 }
function ge(truth) { return (x, y) => x >= y ? truth : 0 }
function fadd(x, y) { return Math.fround(x+y) }
function fsub(x, y) { return Math.fround(x-y) }
function fmul(x, y) { return Math.fround(x*y) }
function fdiv(x, y) { return Math.fround(x/y) }
function fmin(x, y) {
if (x == y) return x;
if (x < y) return x;
if (y < x) return y;
if (isNaN(x)) return x;
return y;
}
function fmax(x, y) {
if (x == y) return x;
if (x > y) return x;
if (y > x) return y;
if (isNaN(x)) return x;
return y;
}
function dadd(x, y) { return x+y }
function dsub(x, y) { return x-y }
function dmul(x, y) { return x*y }
function ddiv(x, y) { return x/y }
var dmax = fmax;
var dmin = fmin;
function op_sat_s(bits, op) {
return (x, y) => {
return signed_saturate(op(sign_extend(x, bits),
sign_extend(y, bits)),
bits);
}
}
function op_sat_u(bits, op) {
return (x, y) => {
return unsigned_saturate(op(zero_extend(x, bits),
zero_extend(y, bits)),
bits);
}
}
function add_sat_s(bits) {
return op_sat_s(bits, (x,y) => x+y);
}
function sub_sat_s(bits) {
return op_sat_s(bits, (x,y) => x-y);
}
function add_sat_u(bits) {
return op_sat_u(bits, (x,y) => x+y);
}
function sub_sat_u(bits) {
return op_sat_u(bits, (x,y) => x-y);
}
function max_s(bits) {
return (x, y) => {
return sign_extend(max(sign_extend(x, bits),
sign_extend(y, bits)),
bits);
}
}
function min_s(bits) {
return (x, y) => {
return sign_extend(min(sign_extend(x, bits),
sign_extend(y, bits)),
bits);
}
}
function max_u(bits) {
return (x, y) => {
return max(zero_extend(x, bits),
zero_extend(y, bits));
}
}
function min_u(bits) {
return (x, y) => {
return min(zero_extend(x, bits),
zero_extend(y, bits));
}
}
function pmin(x, y) { return y < x ? y : x }
function pmax(x, y) { return x < y ? y : x }
assertEq(max_s(8)(1, 2), 2);
assertEq(max_s(8)(1, 128), 1);
assertEq(min_s(8)(1, 2), 1);
assertEq(min_s(8)(1, 128), -128);
assertEq(max_u(8)(1, 2), 2);
assertEq(max_u(8)(1, 128), 128);
assertEq(min_u(8)(1, 2), 1);
assertEq(min_u(8)(1, 128), 1);
var binopTests =
[['i8x16.add', Int8Array, add(8)],
['i16x8.add', Int16Array, add(16)],
['i32x4.add', Int32Array, add(32)],
['i64x2.add', BigInt64Array, add64],
['i8x16.sub', Int8Array, sub(8)],
['i16x8.sub', Int16Array, sub(16)],
['i32x4.sub', Int32Array, sub(32)],
['i64x2.sub', BigInt64Array, sub64],
['i8x16.add_sat_s', Int8Array, add_sat_s(8)],
['i8x16.add_sat_u', Uint8Array, add_sat_u(8)],
['i16x8.add_sat_s', Int16Array, add_sat_s(16)],
['i16x8.add_sat_u', Uint16Array, add_sat_u(16)],
['i8x16.sub_sat_s', Int8Array, sub_sat_s(8)],
['i8x16.sub_sat_u', Uint8Array, sub_sat_u(8)],
['i16x8.sub_sat_s', Int16Array, sub_sat_s(16)],
['i16x8.sub_sat_u', Uint16Array, sub_sat_u(16)],
['i16x8.mul', Int16Array, mul(16)],
['i32x4.mul', Int32Array, mul(32)],
['i64x2.mul', BigInt64Array, mul(64)],
['i8x16.avgr_u', Uint8Array, avgr],
['i16x8.avgr_u', Uint16Array, avgr],
['i8x16.max_s', Int8Array, max_s(8)],
['i8x16.max_u', Uint8Array, max_u(8)],
['i8x16.min_s', Int8Array, min_s(8)],
['i8x16.min_u', Uint8Array, min_u(8)],
['i16x8.max_s', Int16Array, max_s(16)],
['i16x8.max_u', Uint16Array, max_u(16)],
['i16x8.min_s', Int16Array, min_s(16)],
['i16x8.min_u', Uint16Array, min_u(16)],
['i32x4.max_s', Int32Array, max_s(32)],
['i32x4.max_u', Uint32Array, max_u(32)],
['i32x4.min_s', Int32Array, min_s(32)],
['i32x4.min_u', Uint32Array, min_u(32)],
['v128.and', Uint8Array, and],
['v128.or', Uint8Array, or],
['v128.xor', Uint8Array, xor],
['v128.andnot', Uint8Array, andnot],
['f32x4.add', Float32Array, fadd],
['f32x4.sub', Float32Array, fsub],
['f32x4.mul', Float32Array, fmul],
['f32x4.div', Float32Array, fdiv],
['f32x4.min', Float32Array, fmin],
['f32x4.max', Float32Array, fmax],
['f64x2.add', Float64Array, dadd],
['f64x2.sub', Float64Array, dsub],
['f64x2.mul', Float64Array, dmul],
['f64x2.div', Float64Array, ddiv],
['f64x2.min', Float64Array, dmin],
['f64x2.max', Float64Array, dmax],
['i8x16.eq', Int8Array, eq(-1)],
['i8x16.ne', Int8Array, ne(-1)],
['i8x16.lt_s', Int8Array, lt(-1)],
['i8x16.gt_s', Int8Array, gt(-1)],
['i8x16.le_s', Int8Array, le(-1)],
['i8x16.ge_s', Int8Array, ge(-1)],
['i8x16.gt_u', Uint8Array, gt(0xFF)],
['i8x16.ge_u', Uint8Array, ge(0xFF)],
['i8x16.lt_u', Uint8Array, lt(0xFF)],
['i8x16.le_u', Uint8Array, le(0xFF)],
['i16x8.eq', Int16Array, eq(-1)],
['i16x8.ne', Int16Array, ne(-1)],
['i16x8.lt_s', Int16Array, lt(-1)],
['i16x8.gt_s', Int16Array, gt(-1)],
['i16x8.le_s', Int16Array, le(-1)],
['i16x8.ge_s', Int16Array, ge(-1)],
['i16x8.gt_u', Uint16Array, gt(0xFFFF)],
['i16x8.ge_u', Uint16Array, ge(0xFFFF)],
['i16x8.lt_u', Uint16Array, lt(0xFFFF)],
['i16x8.le_u', Uint16Array, le(0xFFFF)],
['i32x4.eq', Int32Array, eq(-1)],
['i32x4.ne', Int32Array, ne(-1)],
['i32x4.lt_s', Int32Array, lt(-1)],
['i32x4.gt_s', Int32Array, gt(-1)],
['i32x4.le_s', Int32Array, le(-1)],
['i32x4.ge_s', Int32Array, ge(-1)],
['i32x4.gt_u', Uint32Array, gt(0xFFFFFFFF)],
['i32x4.ge_u', Uint32Array, ge(0xFFFFFFFF)],
['i32x4.lt_u', Uint32Array, lt(0xFFFFFFFF)],
['i32x4.le_u', Uint32Array, le(0xFFFFFFFF)],
['f32x4.eq', Float32Array, eq(-1), Int32Array],
['f32x4.ne', Float32Array, ne(-1), Int32Array],
['f32x4.lt', Float32Array, lt(-1), Int32Array],
['f32x4.gt', Float32Array, gt(-1), Int32Array],
['f32x4.le', Float32Array, le(-1), Int32Array],
['f32x4.ge', Float32Array, ge(-1), Int32Array],
['f64x2.eq', Float64Array, eq(-1), BigInt64Array],
['f64x2.ne', Float64Array, ne(-1), BigInt64Array],
['f64x2.lt', Float64Array, lt(-1), BigInt64Array],
['f64x2.gt', Float64Array, gt(-1), BigInt64Array],
['f64x2.le', Float64Array, le(-1), BigInt64Array],
['f64x2.ge', Float64Array, ge(-1), BigInt64Array],
['f32x4.pmin', Float32Array, pmin],
['f32x4.pmax', Float32Array, pmax],
['f64x2.pmin', Float64Array, pmin],
['f64x2.pmax', Float64Array, pmax]]
// Run v128 x v128 -> v128 tests. Inputs are taken from the common input sets,
// placed in memory, the test is run, and the result is extracted and checked.
//
// Runs tests with both operands as variables, either as constant, or both as
// constant. Also checks NaN behavior when appropriate.
//
// All runners that call this should use the same value for `ofParts` and should
// pass different values for `part`, up to `ofParts` - 1.
function runSimpleBinopTest(part, ofParts) {
let partSize = Math.ceil(binopTests.length / ofParts);
let start = part * partSize;
let end = Math.min((part + 1) * partSize, binopTests.length);
for ( let [op, memtype, rop, resultmemtype] of binopTests.slice(start, end) ) {
let inputs = cross(memtype.inputs);
let len = 16/memtype.BYTES_PER_ELEMENT;
let xs = iota(len);
let zero = xs.map(_ => 0);
let [ins, mem, resultmem] = insAndMemBinop(op, memtype, resultmemtype, inputs);
let bitsForF32 = memtype == Float32Array ? new Uint32Array(mem.buffer) : null;
let bitsForF64 = memtype == Float64Array ? new BigInt64Array(mem.buffer) : null;
function testIt(a,b,r) {
set(mem, len, a);
set(mem, len*2, b);
ins.exports.run();
assertSame(get(resultmem, 0, len), r);
// Test signalling NaN superficially by replacing QNaN inputs with SNaN
if (bitsForF32 != null && (a.some(isNaN) || b.some(isNaN))) {
a.forEach((x, i) => { if (isNaN(x)) { bitsForF32[len+i] = 0x7FA0_0000; } });
b.forEach((x, i) => { if (isNaN(x)) { bitsForF32[(len*2)+i] = 0x7FA0_0000; } });
ins.exports.run();
assertSame(get(resultmem, 0, len), r);
}
if (bitsForF64 != null && (a.some(isNaN) || b.some(isNaN))) {
a.forEach((x, i) => { if (isNaN(x)) { bitsForF64[len+i] = 0x7FF4_0000_0000_0000n; } });
b.forEach((x, i) => { if (isNaN(x)) { bitsForF64[(len*2)+i] = 0x7FF4_0000_0000_0000n; } });
ins.exports.run();
assertSame(get(resultmem, 0, len), r);
}
}
function testConstIt(i,r) {
set(resultmem, 0, zero);
ins.exports["run_constlhs" + i]();
assertSame(get(resultmem, 0, len), r);
set(resultmem, 0, zero);
ins.exports["run_constrhs" + i]();
assertSame(get(resultmem, 0, len), r);
set(resultmem, 0, zero);
ins.exports["run_constboth" + i]();
assertSame(get(resultmem, 0, len), r);
}
let i = 0;
for (let [a,b] of inputs) {
let r = xs.map((i) => rop(a[i], b[i]));
testIt(a,b,r);
testConstIt(i,r);
i++;
}
}
}