// |jit-test| skip-if: !wasmSimdEnabled() || wasmCompileMode() != "ion" || !this.wasmSimdAnalysis // White-box tests for SIMD optimizations. These are sensitive to internal // details of the front-end and lowering logic, which is partly platform-dependent. // // In DEBUG builds, the testing function wasmSimdAnalysis() returns a string // describing the last decision made by the SIMD lowering code: to perform an // optimized lowering or the default byte shuffle+blend for i8x16.shuffle; to // shift by a constant or a variable for the various shifts; and so on. // // We test that the expected transformation applies, and that the machine code // generates the expected result. var isArm64 = getBuildConfiguration("arm64"); // 32-bit permutation that is not a rotation. let perm32x4_pattern = [4, 5, 6, 7, 12, 13, 14, 15, 8, 9, 10, 11, 0, 1, 2, 3]; // Operands the same, dword permutation { let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm32x4_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm32x4_pattern); } // Right operand ignored, dword permutation { let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${perm32x4_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); set(mem, 32, iota(16).map(x => x+16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm32x4_pattern); } // Left operand ignored, dword permutation { let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${perm32x4_pattern.map(x => x+16).join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16).map(x => x+16)); set(mem, 32, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm32x4_pattern); } // Operands the same, word permutation on both sides of the qword divide, with a qword swap { let perm16x8_pattern = [12, 13, 14, 15, 10, 11, 8, 9, 6, 7, 4, 5, 2, 3, 0, 1]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm16x8_pattern); } // Operands the same, word permutation on both sides of the qword divide, no qword swap { let perm16x8_pattern = [ 6, 7, 4, 5, 2, 3, 0, 1, 12, 13, 14, 15, 10, 11, 8, 9]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm16x8_pattern); } // Operands the same, word permutation on low side of the qword divide, no qword swap { let perm16x8_pattern = [ 6, 7, 4, 5, 2, 3, 0, 1, 8, 9, 10, 11, 12, 13, 14, 15]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm16x8_pattern); } // Operands the same, word permutation on high side of the qword divide, no qword swap { let perm16x8_pattern = [ 0, 1, 2, 3, 4, 5, 6, 7, 12, 13, 14, 15, 10, 11, 8, 9]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm16x8_pattern); } // Same operands, byte rotate { // 8-bit permutation that is a rotation let rot8x16_pattern = [5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 0, 1, 2, 3, 4]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rot8x16_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> rotate-right 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rot8x16_pattern); } // Operands the same, random jumble => byte permutation { // 8-bit permutation that is not a rotation let perm8x16_pattern = [5, 7, 6, 8, 9, 10, 11, 4, 13, 14, 15, 0, 1, 2, 3, 12]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${perm8x16_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), perm8x16_pattern); } // Operands differ, both accessed, rhs is constant zero, left-shift pattern { // 8-bit shift with zeroes shifted in at the right end let shift8x16_pattern = [16, 16, 16, 16, 16, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${shift8x16_pattern.join(' ')} (local.get 0) (v128.const i32x4 0 0 0 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shift-left 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), shift8x16_pattern.map(x => x >= 16 ? 0 : x)); } // The same as above but the constant is lhs. { // 8-bit shift with zeroes shifted in at the right end let shift8x16_pattern = [16, 16, 16, 16, 16, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map(x => x ^ 16); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${shift8x16_pattern.join(' ')} (v128.const i32x4 0 0 0 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shift-left 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), shift8x16_pattern.map(x => x < 16 ? 0 : x - 16)); } // Operands differ, both accessed, rhs is constant zero, left-shift pattern that // does not start properly. { // 8-bit shift with zeroes shifted in at the right end let shift8x16_pattern = [16, 16, 16, 16, 16, 16, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${shift8x16_pattern.join(' ')} (local.get 0) (v128.const i32x4 0 0 0 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shuffle+blend 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), shift8x16_pattern.map(x => x >= 16 ? 0 : x)); } // Operands differ, both accessed, rhs is constant zero, right-shift pattern { // 8-bit shift with zeroes shifted in at the right end let shift8x16_pattern = [6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 20, 20, 20, 20, 20, 20]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${shift8x16_pattern.join(' ')} (local.get 0) (v128.const i32x4 0 0 0 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shift-right 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), shift8x16_pattern.map(x => x >= 16 ? 0 : x)); } // Operands differ, both accessed, rhs is constant zero, right-shift pattern // that does not end properly. { // 8-bit shift with zeroes shifted in at the right end let shift8x16_pattern = [6, 7, 8, 9, 10, 11, 12, 13, 14, 20, 20, 20, 20, 20, 20, 20]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${shift8x16_pattern.join(' ')} (local.get 0) (v128.const i32x4 0 0 0 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shuffle+blend 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), shift8x16_pattern.map(x => x >= 16 ? 0 : x)); } // Operands differ and are variable, both accessed, (lhs ++ rhs) >> k { let concat8x16_pattern = [27, 28, 29, 30, 31, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${concat8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> concat+shift-right 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); set(mem, 32, iota(16).map(k => k+16)); ins.exports.run(); assertSame(get(mem, 0, 16), concat8x16_pattern); } // Operands differ and are variable, both accessed, (rhs ++ lhs) >> k { let concat8x16_pattern = [11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${concat8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> concat+shift-right 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); set(mem, 32, iota(16).map(k => k+16)); ins.exports.run(); assertSame(get(mem, 0, 16), concat8x16_pattern); } // Operands differ, both accessed, but inputs stay in their lanes => byte blend { let blend8x16_pattern = iota(16).map(x => (x % 3 == 0) ? x + 16 : x); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${blend8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> blend 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); let lhs = iota(16); let rhs = iota(16).map(x => x+16); set(mem, 16, lhs); set(mem, 32, rhs); ins.exports.run(); assertSame(get(mem, 0, 16), blend8x16_pattern); } // Operands differ, both accessed, but inputs stay in their lanes => word blend { let blend16x8_pattern = iota(16).map(x => (x & 2) ? x + 16 : x); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${blend16x8_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> blend 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); let lhs = iota(16); let rhs = iota(16).map(x => x+16); set(mem, 16, lhs); set(mem, 32, rhs); ins.exports.run(); assertSame(get(mem, 0, 16), blend16x8_pattern); } // Interleave i32x4s for ( let [lhs, rhs, expected] of [[[0, 1], [4, 5], "shuffle -> interleave-low 32x4"], [[2, 3], [6, 7], "shuffle -> interleave-high 32x4"]] ) { for (let swap of [false, true]) { if (swap) [lhs, rhs] = [rhs, lhs]; let interleave_pattern = i32ToI8(interleave(lhs, rhs)); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${interleave_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), expected); let mem = new Int8Array(ins.exports.mem.buffer); let lhsval = iota(16); let rhsval = iota(16).map(x => x+16); set(mem, 16, lhsval); set(mem, 32, rhsval); ins.exports.run(); assertSame(get(mem, 0, 16), interleave_pattern); } } // Interleave i64x2s for ( let [lhs, rhs, expected] of [[[0], [2], "shuffle -> interleave-low 64x2"], [[1], [3], "shuffle -> interleave-high 64x2"]] ) { for (let swap of [false, true]) { if (swap) [lhs, rhs] = [rhs, lhs]; let interleave_pattern = i64ToI2(interleave(lhs, rhs)); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${interleave_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), expected); let mem = new Int8Array(ins.exports.mem.buffer); let lhsval = iota(16); let rhsval = iota(16).map(x => x+16); set(mem, 16, lhsval); set(mem, 32, rhsval); ins.exports.run(); assertSame(get(mem, 0, 16), interleave_pattern); } } // Interleave i16x8s for ( let [lhs, rhs, expected] of [[[0, 1, 2, 3], [8, 9, 10, 11], "shuffle -> interleave-low 16x8"], [[4, 5, 6, 7], [12, 13, 14, 15], "shuffle -> interleave-high 16x8"]] ) { for (let swap of [false, true]) { if (swap) [lhs, rhs] = [rhs, lhs]; let interleave_pattern = i16ToI8(interleave(lhs, rhs)); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${interleave_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), expected); let mem = new Int8Array(ins.exports.mem.buffer); let lhsval = iota(16); let rhsval = iota(16).map(x => x+16); set(mem, 16, lhsval); set(mem, 32, rhsval); ins.exports.run(); assertSame(get(mem, 0, 16), interleave_pattern); } } // Interleave i8x16s for ( let [lhs, rhs, expected] of [[[0, 1, 2, 3, 4, 5, 6, 7], [16, 17, 18, 19, 20, 21, 22, 23], "shuffle -> interleave-low 8x16"], [[8, 9, 10, 11, 12, 13, 14, 15],[24, 25, 26, 27, 28, 29, 30, 31], "shuffle -> interleave-high 8x16"]] ) { for (let swap of [false, true]) { if (swap) [lhs, rhs] = [rhs, lhs]; let interleave_pattern = interleave(lhs, rhs); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${interleave_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), expected); let mem = new Int8Array(ins.exports.mem.buffer); let lhsval = iota(16); let rhsval = iota(16).map(x => x+16); set(mem, 16, lhsval); set(mem, 32, rhsval); ins.exports.run(); assertSame(get(mem, 0, 16), interleave_pattern); } } // Operands differ, both accessed, random jumble => byte shuffle+blend { let blend_perm8x16_pattern = [5, 23, 6, 24, 9, 10, 11, 7, 7, 14, 15, 19, 1, 2, 3, 12]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${blend_perm8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> shuffle+blend 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); let lhs = iota(16).map(x => x+16); let rhs = iota(16); set(mem, 16, lhs); set(mem, 32, rhs); ins.exports.run(); assertSame(get(mem, 0, 16), blend_perm8x16_pattern.map(x => x < 16 ? lhs[x] : rhs[x-16])); } // No-op, ignoring right operand, should turn into a move. { let nop8x16_pattern = iota(16); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${nop8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> move"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); set(mem, 32, iota(16).map(x => x+16)); ins.exports.run(); assertSame(get(mem, 0, 16), nop8x16_pattern); } // No-op, ignoring left operand, should turn into a move. { let nop8x16_pattern = iota(16).map(x => x+16); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16)) (v128.load (i32.const 32))))) (func $f (param v128) (param v128) (result v128) (i8x16.shuffle ${nop8x16_pattern.join(' ')} (local.get 0) (local.get 1))))`); assertEq(wasmSimdAnalysis(), "shuffle -> move"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); set(mem, 32, iota(16).map(x => x+16)); ins.exports.run(); assertSame(get(mem, 0, 16), nop8x16_pattern); } // Broadcast byte for ( let byte of [3, 11, 8, 2] ) { let broadcast8x16_pattern = iota(16).map(_ => byte); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${broadcast8x16_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> broadcast 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), broadcast8x16_pattern); } // Broadcast word from high quadword { let broadcast16x8_pattern = [10, 11, 10, 11, 10, 11, 10, 11, 10, 11, 10, 11, 10, 11, 10, 11]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${broadcast16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> broadcast 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), broadcast16x8_pattern); } // Broadcast word from low quadword { let broadcast16x8_pattern = [4, 5, 4, 5, 4, 5, 4, 5, 4, 5, 4, 5, 4, 5, 4, 5]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${broadcast16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> broadcast 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), broadcast16x8_pattern); } // Broadcast dword from low quadword should turn into a dword permute { let broadcast32x4_pattern = [4, 5, 6, 7, 4, 5, 6, 7, 4, 5, 6, 7, 4, 5, 6, 7]; let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${broadcast32x4_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), broadcast32x4_pattern); } // Broadcast high qword should turn into a dword permute { let broadcast64x2_pattern = [8, 9, 10, 11, 12, 13, 14, 15, 8, 9, 10, 11, 12, 13, 14, 15] let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${broadcast64x2_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), broadcast64x2_pattern); } // Byte reversal should be a byte permute { let rev8x16_pattern = iota(16).reverse(); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rev8x16_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 8x16"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rev8x16_pattern); } // Byteswap of half-word, word and quad-word groups should be // reverse bytes analysis for (let k of [2, 4, 8]) { let rev8_pattern = iota(16).map(i => i ^ (k - 1)); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rev8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), `shuffle -> reverse bytes in ${8 * k}-bit lanes`); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rev8_pattern); } // Word reversal should be a word permute { let rev16x8_pattern = i16ToI8(iota(8).reverse()); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rev16x8_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 16x8"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rev16x8_pattern); } // Dword reversal should be a dword permute { let rev32x4_pattern = i32ToI8([3, 2, 1, 0]); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rev32x4_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rev32x4_pattern); } // Qword reversal should be a dword permute { let rev64x2_pattern = i32ToI8([2, 3, 0, 1]); let ins = wasmCompile(` (module (memory (export "mem") 1 1) (func (export "run") (v128.store (i32.const 0) (call $f (v128.load (i32.const 16))))) (func $f (param v128) (result v128) (i8x16.shuffle ${rev64x2_pattern.join(' ')} (local.get 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); let mem = new Int8Array(ins.exports.mem.buffer); set(mem, 16, iota(16)); ins.exports.run(); assertSame(get(mem, 0, 16), rev64x2_pattern); } // In the case of shifts, we have separate tests that constant shifts work // correctly, so no such testing is done here. for ( let lanes of ['i8x16', 'i16x8', 'i32x4', 'i64x2'] ) { for ( let shift of ['shl', 'shr_s', 'shr_u'] ) { for ( let [count, result] of [['(i32.const 5)', /shift -> constant shift/], ['(local.get 1)', /shift -> variable(?: scalarized)? shift/]] ) { wasmCompile(`(module (func (param v128) (param i32) (result v128) (${lanes}.${shift} (local.get 0) ${count})))`); assertEq(wasmSimdAnalysis().match(result).length, 1); } } } // Zero extending int values. { const zeroExtTypes = [ {ty: '8x16', size: 1, ch: 'b'}, {ty: '16x8', size: 2, ch: 'w'}, {ty: '32x4', size: 4, ch: 'd'}, {ty: '64x2', size: 8, ch: 'q'}]; function generateZeroExtend (src, dest, inv) { const ar = new Array(16); for (let i = 0, j = 0; i < ar.length; i++) { if ((i % dest) >= src) { ar[i] = (inv ? 0 : 16) + (i % 16); continue; } ar[i] = j++ + (inv ? 16 : 0); } return ar.join(' '); } for (let i = 0; i < 3; i++) { for (let j = i + 1; j < 4; j++) { const result = `shuffle -> zero-extend ${zeroExtTypes[i].ty} to ${zeroExtTypes[j].ty}`; const pat = generateZeroExtend(zeroExtTypes[i].size, zeroExtTypes[j].size, false); wasmCompile(`(module (func (param v128) (result v128) (i8x16.shuffle ${pat} (local.get 0) (v128.const i32x4 0 0 0 0))))`); assertEq(wasmSimdAnalysis(), result); const patInv = generateZeroExtend(zeroExtTypes[i].size, zeroExtTypes[j].size, true); wasmCompile(`(module (func (param v128) (result v128) (i8x16.shuffle ${patInv} (v128.const i32x4 0 0 0 0) (local.get 0))))`); assertEq(wasmSimdAnalysis(), result); // Test in wasm by "hidding" zero constant as an argument. const ins = wasmEvalText(`(module (func $t (param v128) (result v128) (i8x16.shuffle ${pat} (local.get 0) (v128.const i32x4 0 0 0 0))) (func $check (param v128) (param v128) (result v128) (i8x16.shuffle ${pat} (local.get 0) (local.get 1))) (func (export "test") (result i32) v128.const i32x4 0xff01ee02 0xdd03cc04 0xaa059906 0x88776655 call $t v128.const i32x4 0xff01ee02 0xdd03cc04 0xaa059906 0x88776655 v128.const i32x4 0 0 0 0 call $check i8x16.eq i8x16.bitmask ))`); assertEq(ins.exports.test(), 0xffff); } } // Some patterns that look like zero extend. for (let pat of ["0 2 4 6 8 10 12 14 16 18 20 22 24 26 28 30"]) { wasmCompile(`(module (func (param v128) (result v128) (i8x16.shuffle ${pat} (local.get 0) (v128.const i32x4 0 0 0 0))))`); const res = wasmSimdAnalysis(); assertEq(!res.includes("shuffle -> zero-extend"), true); } } // Constant folding scalar->simd. There are functional tests for all these in // ad-hack.js so here we only check that the transformation is triggered. for ( let [ty128, ty] of [['i8x16', 'i32'], ['i16x8', 'i32'], ['i32x4', 'i32'], ['i64x2', 'i64'], ['f32x4', 'f32'], ['f64x2', 'f64']] ) { wasmCompile(`(module (func (result v128) (${ty128}.splat (${ty}.const 37))))`); assertEq(wasmSimdAnalysis(), "scalar-to-simd128 -> constant folded"); } // Ditto simd->scalar. for ( let [ty128, suffix] of [['i8x16', '_s'], ['i8x16', '_u'], ['i16x8','_s'], ['i16x8','_u'], ['i32x4', '']] ) { for ( let op of ['any_true', 'all_true', 'bitmask', `extract_lane${suffix} 0`] ) { let operation = op == 'any_true' ? 'v128.any_true' : `${ty128}.${op}`; wasmCompile(`(module (func (result i32) (${operation} (v128.const i64x2 0 0))))`); assertEq(wasmSimdAnalysis(), "simd128-to-scalar -> constant folded"); } } for ( let ty128 of ['f32x4','f64x2','i64x2'] ) { wasmCompile(`(module (func (result ${ty128.match(/(...)x.*/)[1]}) (${ty128}.extract_lane 0 (v128.const i64x2 0 0))))`); assertEq(wasmSimdAnalysis(), "simd128-to-scalar -> constant folded"); } // Optimizing all_true, any_true, and bitmask that are used for control flow, also when negated. for ( let [ty128,size] of [['i8x16',1], ['i16x8',2], ['i32x4',4]] ) { let all = iota(16/size).map(n => n*n); let some = iota(16/size).map(n => n*(n % 3)); let none = iota(16/size).map(n => 0); let inputs = [all, some, none]; let ops = { all_true: allTrue, any_true: anyTrue, bitmask }; for ( let op of ['any_true', 'all_true', 'bitmask'] ) { let folded = op != 'bitmask' || (size == 2 && !isArm64); let operation = op == 'any_true' ? 'v128.any_true' : `${ty128}.${op}`; let positive = wasmCompile( `(module (memory (export "mem") 1 1) (func $f (param v128) (result i32) (if (result i32) (${operation} (local.get 0)) (then (i32.const 42)) (else (i32.const 37)))) (func (export "run") (result i32) (call $f (v128.load (i32.const 16)))))`); assertEq(wasmSimdAnalysis(), folded ? "simd128-to-scalar-and-branch -> folded" : "none"); let negative = wasmCompile( `(module (memory (export "mem") 1 1) (func $f (param v128) (result i32) (if (result i32) (i32.eqz (${operation} (local.get 0))) (then (i32.const 42)) (else (i32.const 37)))) (func (export "run") (result i32) (call $f (v128.load (i32.const 16)))))`); assertEq(wasmSimdAnalysis(), folded ? "simd128-to-scalar-and-branch -> folded" : "none"); for ( let inp of inputs ) { let mem = new this[`Int${8*size}Array`](positive.exports.mem.buffer); set(mem, 16/size, inp); assertEq(positive.exports.run(), ops[op](inp) ? 42 : 37); mem = new this[`Int${8*size}Array`](negative.exports.mem.buffer); set(mem, 16/size, inp); assertEq(negative.exports.run(), ops[op](inp) ? 37 : 42); } } } // Constant folding { // Swizzle-with-constant rewritten as shuffle, and then further optimized // into a dword permute. Correctness is tested in ad-hack.js. wasmCompile(` (module (func (param v128) (result v128) (i8x16.swizzle (local.get 0) (v128.const i8x16 4 5 6 7 0 1 2 3 12 13 14 15 8 9 10 11)))) `); assertEq(wasmSimdAnalysis(), "shuffle -> permute 32x4"); } // Bitselect with constant mask folded into shuffle operation if (!isArm64) { wasmCompile(` (module (func (param v128) (param v128) (result v128) (v128.bitselect (local.get 0) (local.get 1) (v128.const i8x16 0 -1 -1 0 0 0 0 0 -1 -1 -1 -1 -1 -1 0 0)))) `); assertEq(wasmSimdAnalysis(), "shuffle -> blend 8x16"); } // Library function wasmCompile(text) { return new WebAssembly.Instance(new WebAssembly.Module(wasmTextToBinary(text))) } function get(arr, loc, len) { let res = []; for ( let i=0; i < len; i++ ) { res.push(arr[loc+i]); } return res; } function set(arr, loc, vals) { for ( let i=0; i < vals.length; i++ ) { arr[loc+i] = vals[i]; } } function i32ToI8(xs) { return xs.map(x => [x*4, x*4+1, x*4+2, x*4+3]).flat(); } function i64ToI2(xs) { return xs.map(x => [x*8, x*8+1, x*8+2, x*8+3, x*8+4, x*8+5, x*8+6, x*8+7]).flat(); } function i16ToI8(xs) { return xs.map(x => [x*2, x*2+1]).flat(); } function allTrue(xs) { return xs.every(v => v != 0); } function anyTrue(xs) { return xs.some(v => v != 0); } function bitmask(xs) { let shift = 128/xs.length - 1; let res = 0; let k = 0; xs.forEach(v => { res |= ((v >>> shift) & 1) << k; k++; }); return res; }