Files
wazero/internal/engine/compiler/compiler_vec_test.go
Takeshi Yoneda 0c303258c7 SIMD: implements v128 load, store and lane manipulations. (#588)
This implements various SIMD instructions related to
load, store, and lane manipulations for all engines.

Notablely, now our engines pass the following specification tests:

* simd_address.wast
* simd_const.wast
* simd_align.wast
* simd_laod16_lane.wast
* simd_laod32_lane.wast
* simd_laod64_lane.wast
* simd_laod8_lane.wast
* simd_lane.wast
* simd_load_extend.wast
* simd_load_splat.wast
* simd_load_zero.wast
* simd_store.wast
* simd_store16_lane.wast
* simd_store32_lane.wast
* simd_store64_lane.wast
* simd_store8_lane.wast

part of #484


Signed-off-by: Takeshi Yoneda <takeshi@tetrate.io>
Co-authored-by: Adrian Cole <adrian@tetrate.io>
2022-06-01 09:30:05 +09:00

1759 lines
50 KiB
Go

package compiler
import (
"encoding/binary"
"math"
"testing"
"github.com/tetratelabs/wazero/internal/testing/require"
"github.com/tetratelabs/wazero/internal/wasm"
"github.com/tetratelabs/wazero/internal/wazeroir"
)
func TestCompiler_compileV128Add(t *testing.T) {
// TODO
}
func TestCompiler_compileV128Sub(t *testing.T) {
tests := []struct {
name string
shape wazeroir.Shape
x1, x2, exp [16]byte
}{
{
name: "i8x16",
shape: wazeroir.ShapeI8x16,
x1: [16]byte{0: 1, 2: 10, 10: 10},
x2: [16]byte{0: 10, 4: 5, 10: 5},
exp: [16]byte{0: i8ToU8(-9), 2: 10, 4: i8ToU8(-5), 10: 5},
},
// TODO: add more cases.
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.x1[:8]),
Hi: binary.LittleEndian.Uint64(tc.x1[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.x2[:8]),
Hi: binary.LittleEndian.Uint64(tc.x2[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Sub(&wazeroir.OperationV128Sub{Shape: tc.shape})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, nativeCallStatusCodeReturned, env.callEngine().statusCode)
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}
func TestCompiler_compileV128Load(t *testing.T) {
tests := []struct {
name string
memSetupFn func(buf []byte)
loadType wazeroir.LoadV128Type
offset uint32
exp [16]byte
}{
{
name: "v128 offset=0", loadType: wazeroir.LoadV128Type128, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20})
},
exp: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
},
{
name: "v128 offset=2", loadType: wazeroir.LoadV128Type128, offset: 2,
memSetupFn: func(buf []byte) {
copy(buf, []byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20})
},
exp: [16]byte{3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18},
},
{
name: "8x8s offset=0", loadType: wazeroir.LoadV128Type8x8s, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0, 0xff, 0xff, 3, 0, 0xff, 0xff, 5, 0, 0xff, 0xff, 7, 0, 0xff, 0xff,
},
},
{
name: "8x8s offset=3", loadType: wazeroir.LoadV128Type8x8s, offset: 3,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
0xff, 0xff, 5, 0, 0xff, 0xff, 7, 0, 0xff, 0xff, 9, 0, 10, 0, 11, 0,
},
},
{
name: "8x8u offset=0", loadType: wazeroir.LoadV128Type8x8u, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0, 0xff, 0, 3, 0, 0xff, 0, 5, 0, 0xff, 0, 7, 0, 0xff, 0,
},
},
{
name: "8x8i offset=3", loadType: wazeroir.LoadV128Type8x8u, offset: 3,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
0xff, 0, 5, 0, 0xff, 0, 7, 0, 0xff, 0, 9, 0, 10, 0, 11, 0,
},
},
{
name: "16x4s offset=0", loadType: wazeroir.LoadV128Type16x4s, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 0xff, 0xff,
3, 0xff, 0xff, 0xff,
5, 0xff, 0xff, 0xff,
7, 0xff, 0xff, 0xff,
},
},
{
name: "16x4s offset=3", loadType: wazeroir.LoadV128Type16x4s, offset: 3,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 0xff, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
0xff, 5, 0, 0,
6, 0xff, 0xff, 0xff,
0xff, 9, 0, 0,
10, 11, 0, 0,
},
},
{
name: "16x4u offset=0", loadType: wazeroir.LoadV128Type16x4u, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 0xff, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 0, 0,
3, 0xff, 0, 0,
5, 0xff, 0, 0,
7, 0xff, 0, 0,
},
},
{
name: "16x4u offset=3", loadType: wazeroir.LoadV128Type16x4u, offset: 3,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 0xff, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
0xff, 5, 0, 0,
6, 0xff, 0, 0,
0xff, 9, 0, 0,
10, 11, 0, 0,
},
},
{
name: "32x2s offset=0", loadType: wazeroir.LoadV128Type32x2s, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 3, 0xff, 0xff, 0xff, 0xff, 0xff,
5, 6, 7, 0xff, 0xff, 0xff, 0xff, 0xff,
},
},
{
name: "32x2s offset=2", loadType: wazeroir.LoadV128Type32x2s, offset: 2,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
3, 0xff, 5, 6, 0, 0, 0, 0,
7, 0xff, 9, 0xff, 0xff, 0xff, 0xff, 0xff,
},
},
{
name: "32x2u offset=0", loadType: wazeroir.LoadV128Type32x2u, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 10,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 3, 0xff, 0, 0, 0, 0,
5, 6, 7, 0xff, 0, 0, 0, 0,
},
},
{
name: "32x2u offset=2", loadType: wazeroir.LoadV128Type32x2u, offset: 2,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
3, 0xff, 5, 6, 0, 0, 0, 0,
7, 0xff, 9, 0xff, 0, 0, 0, 0,
},
},
{
name: "32zero offset=0", loadType: wazeroir.LoadV128Type32zero, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 3, 0xff, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
},
},
{
name: "32zero offset=3", loadType: wazeroir.LoadV128Type32zero, offset: 3,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 0xff, 8, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
0xff, 5, 6, 0xff, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
},
},
{
name: "64zero offset=0", loadType: wazeroir.LoadV128Type64zero, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff,
0, 0, 0, 0, 0, 0, 0, 0,
},
},
{
name: "64zero offset=2", loadType: wazeroir.LoadV128Type64zero, offset: 2,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
11, 12, 13, 14, 15, 16, 17, 18, 19, 20,
})
},
exp: [16]byte{
3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
0, 0, 0, 0, 0, 0, 0, 0,
},
},
{
name: "8splat offset=0", loadType: wazeroir.LoadV128Type8Splat, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1},
},
{
name: "8splat offset=1", loadType: wazeroir.LoadV128Type8Splat, offset: 1,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
},
{
name: "16splat offset=0", loadType: wazeroir.LoadV128Type16Splat, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{1, 0xff, 1, 0xff, 1, 0xff, 1, 0xff, 1, 0xff, 1, 0xff, 1, 0xff, 1, 0xff},
},
{
name: "16splat offset=5", loadType: wazeroir.LoadV128Type16Splat, offset: 5,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{6, 7, 6, 7, 6, 7, 6, 7, 6, 7, 6, 7, 6, 7, 6, 7},
},
{
name: "32splat offset=0", loadType: wazeroir.LoadV128Type32Splat, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{1, 0xff, 3, 0xff, 1, 0xff, 3, 0xff, 1, 0xff, 3, 0xff, 1, 0xff, 3, 0xff},
},
{
name: "32splat offset=1", loadType: wazeroir.LoadV128Type32Splat, offset: 1,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{0xff, 3, 0xff, 5, 0xff, 3, 0xff, 5, 0xff, 3, 0xff, 5, 0xff, 3, 0xff, 5},
},
{
name: "64splat offset=0", loadType: wazeroir.LoadV128Type64Splat, offset: 0,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 1, 0xff, 3, 0xff, 5, 6, 7, 0xff},
},
{
name: "64splat offset=1", loadType: wazeroir.LoadV128Type64Splat, offset: 1,
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff,
})
},
exp: [16]byte{0xff, 3, 0xff, 5, 6, 7, 0xff, 9, 0xff, 3, 0xff, 5, 6, 7, 0xff, 9},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
tc.memSetupFn(env.memory())
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileConstI32(&wazeroir.OperationConstI32{Value: tc.offset})
require.NoError(t, err)
err = compiler.compileV128Load(&wazeroir.OperationV128Load{
Type: tc.loadType, Arg: &wazeroir.MemoryArg{},
})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
loadedLocation := compiler.runtimeValueLocationStack().peek()
require.True(t, loadedLocation.onRegister())
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, uint64(2), env.stackPointer())
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}
func TestCompiler_compileV128LoadLane(t *testing.T) {
originalVecLo, originalVecHi := uint64(0), uint64(0)
tests := []struct {
name string
memSetupFn func(buf []byte)
laneIndex, laneSize byte
offset uint32
exp [16]byte
}{
{
name: "8_lane offset=0 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff,
})
},
laneSize: 8,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "8_lane offset=1 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff,
})
},
laneSize: 8,
laneIndex: 0,
offset: 1,
exp: [16]byte{0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "8_lane offset=1 laneIndex=5",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff,
})
},
laneSize: 8,
laneIndex: 5,
offset: 1,
exp: [16]byte{0, 0, 0, 0, 0, 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=0 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa,
})
},
laneSize: 16,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=1 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa,
})
},
laneSize: 16,
laneIndex: 0,
offset: 1,
exp: [16]byte{0xff, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=1 laneIndex=5",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa,
})
},
laneSize: 16,
laneIndex: 5,
offset: 1,
exp: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 1, 0, 0, 0, 0},
},
{
name: "32_lane offset=0 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8,
})
},
laneSize: 32,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 0xff, 1, 0xa, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "32_lane offset=1 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8,
})
},
laneSize: 32,
laneIndex: 0,
offset: 1,
exp: [16]byte{0xff, 1, 0xa, 0x9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "32_lane offset=1 laneIndex=3",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8,
})
},
laneSize: 32,
laneIndex: 3,
offset: 1,
exp: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 1, 0xa, 0x9},
},
{
name: "64_lane offset=0 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8, 0x1, 0x2, 0x3, 0x4,
})
},
laneSize: 64,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 0xff, 1, 0xa, 0x9, 0x8, 0x1, 0x2, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "64_lane offset=1 laneIndex=0",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8, 0x1, 0x2, 0x3, 0x4,
})
},
laneSize: 64,
laneIndex: 0,
offset: 1,
exp: [16]byte{0xff, 1, 0xa, 0x9, 0x8, 0x1, 0x2, 0x3, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "64_lane offset=3 laneIndex=1",
memSetupFn: func(buf []byte) {
copy(buf, []byte{
1, 0xff, 1, 0xa, 0x9, 0x8, 0x1, 0x2, 0x3, 0x4, 0xa,
})
},
laneSize: 64,
laneIndex: 1,
offset: 3,
exp: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0xa, 0x9, 0x8, 0x1, 0x2, 0x3, 0x4, 0xa},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
tc.memSetupFn(env.memory())
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileConstI32(&wazeroir.OperationConstI32{Value: tc.offset})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: originalVecLo,
Hi: originalVecHi,
})
require.NoError(t, err)
err = compiler.compileV128LoadLane(&wazeroir.OperationV128LoadLane{
LaneIndex: tc.laneIndex, LaneSize: tc.laneSize, Arg: &wazeroir.MemoryArg{},
})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
loadedLocation := compiler.runtimeValueLocationStack().peek()
require.True(t, loadedLocation.onRegister())
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, uint64(2), env.stackPointer())
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}
func TestCompiler_compileV128Store(t *testing.T) {
tests := []struct {
name string
offset uint32
}{
{name: "offset=1", offset: 1},
{name: "offset=5", offset: 5},
{name: "offset=10", offset: 10},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileConstI32(&wazeroir.OperationConstI32{Value: tc.offset})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{Lo: ^uint64(0), Hi: ^uint64(0)})
require.NoError(t, err)
err = compiler.compileV128Store(&wazeroir.OperationV128Store{Arg: &wazeroir.MemoryArg{}})
require.NoError(t, err)
require.Equal(t, uint64(0), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 0, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, uint64(0), env.stackPointer())
mem := env.memory()
require.Equal(t, []byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
mem[tc.offset:tc.offset+16])
})
}
}
func TestCompiler_compileV128StoreLane(t *testing.T) {
vecBytes := [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}
tests := []struct {
name string
laneIndex, laneSize byte
offset uint32
exp [16]byte
}{
{
name: "8_lane offset=0 laneIndex=0",
laneSize: 8,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "8_lane offset=1 laneIndex=0",
laneSize: 8,
laneIndex: 0,
offset: 1,
exp: [16]byte{0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "8_lane offset=3 laneIndex=5",
laneSize: 8,
laneIndex: 5,
offset: 3,
exp: [16]byte{0, 0, 0, 6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=0 laneIndex=0",
laneSize: 16,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=1 laneIndex=0",
laneSize: 16,
laneIndex: 0,
offset: 1,
exp: [16]byte{0, 1, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "16_lane offset=5 laneIndex=7",
laneSize: 16,
laneIndex: 7,
offset: 5,
exp: [16]byte{0, 0, 0, 0, 0, 15, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "32_lane offset=0 laneIndex=0",
laneSize: 32,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 2, 3, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "32_lane offset=1 laneIndex=0",
laneSize: 32,
laneIndex: 0,
offset: 1,
exp: [16]byte{0, 1, 2, 3, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "32_lane offset=5 laneIndex=3",
laneSize: 32,
laneIndex: 3,
offset: 5,
exp: [16]byte{0, 0, 0, 0, 0, 13, 14, 15, 16, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "64_lane offset=0 laneIndex=0",
laneSize: 64,
laneIndex: 0,
offset: 0,
exp: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "64_lane offset=1 laneIndex=0",
laneSize: 64,
laneIndex: 0,
offset: 1,
exp: [16]byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "64_lane offset=5 laneIndex=3",
laneSize: 64,
laneIndex: 1,
offset: 6,
exp: [16]byte{0, 0, 0, 0, 0, 0, 9, 10, 11, 12, 13, 14, 15, 16, 0, 0},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileConstI32(&wazeroir.OperationConstI32{Value: tc.offset})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(vecBytes[:8]),
Hi: binary.LittleEndian.Uint64(vecBytes[8:]),
})
require.NoError(t, err)
err = compiler.compileV128StoreLane(&wazeroir.OperationV128StoreLane{
LaneIndex: tc.laneIndex, LaneSize: tc.laneSize, Arg: &wazeroir.MemoryArg{},
})
require.NoError(t, err)
require.Equal(t, uint64(0), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 0, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, tc.exp[:], env.memory()[:16])
})
}
}
func TestCompiler_compileV128ExtractLane(t *testing.T) {
tests := []struct {
name string
vecBytes [16]byte
shape wazeroir.Shape
signed bool
laneIndex byte
exp uint64
}{
{
name: "i8x16 unsigned index=0",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
shape: wazeroir.ShapeI8x16,
signed: false,
laneIndex: 0,
exp: uint64(byte(1)),
},
{
name: "i8x16 unsigned index=15",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 0xff},
shape: wazeroir.ShapeI8x16,
signed: false,
laneIndex: 15,
exp: uint64(byte(0xff)),
},
{
name: "i8x16 signed index=0",
vecBytes: [16]byte{0xf1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
shape: wazeroir.ShapeI8x16,
signed: true,
laneIndex: 0,
exp: uint64(0xff_ff_ff_f1),
},
{
name: "i8x16 signed index=1",
vecBytes: [16]byte{0xf0, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
shape: wazeroir.ShapeI8x16,
signed: true,
laneIndex: 1,
exp: uint64(2),
},
{
name: "i16x8 unsigned index=0",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
shape: wazeroir.ShapeI16x8,
signed: false,
laneIndex: 0,
exp: uint64(uint16(0x2<<8 | 0x1)),
},
{
name: "i16x8 unsigned index=7",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 0xff},
shape: wazeroir.ShapeI16x8,
signed: false,
laneIndex: 7,
exp: uint64(uint16(0xff<<8 | 15)),
},
{
name: "i16x8 signed index=0",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16},
shape: wazeroir.ShapeI16x8,
signed: true,
laneIndex: 0,
exp: uint64(uint16(0x2<<8 | 0x1)),
},
{
name: "i16x8 signed index=7",
vecBytes: [16]byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 0xf1},
shape: wazeroir.ShapeI16x8,
signed: true,
laneIndex: 7,
exp: uint64(uint32(0xffff<<16) | uint32(uint16(0xf1<<8|15))),
},
{
name: "i32x4 index=0",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeI32x4,
laneIndex: 0,
exp: uint64(uint32(0x04_03_02_01)),
},
{
name: "i32x4 index=3",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeI32x4,
laneIndex: 3,
exp: uint64(uint32(0x16_15_14_13)),
},
{
name: "i64x4 index=0",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeI64x2,
laneIndex: 0,
exp: uint64(0x08_07_06_05_04_03_02_01),
},
{
name: "i64x4 index=1",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeI64x2,
laneIndex: 1,
exp: uint64(0x16_15_14_13_12_11_10_09),
},
{
name: "f32x4 index=0",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeF32x4,
laneIndex: 0,
exp: uint64(uint32(0x04_03_02_01)),
},
{
name: "f32x4 index=3",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeF32x4,
laneIndex: 3,
exp: uint64(uint32(0x16_15_14_13)),
},
{
name: "f64x4 index=0",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeF64x2,
laneIndex: 0,
exp: uint64(0x08_07_06_05_04_03_02_01),
},
{
name: "f64x4 index=1",
vecBytes: [16]byte{0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16},
shape: wazeroir.ShapeF64x2,
laneIndex: 1,
exp: uint64(0x16_15_14_13_12_11_10_09),
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.vecBytes[:8]),
Hi: binary.LittleEndian.Uint64(tc.vecBytes[8:]),
})
require.NoError(t, err)
err = compiler.compileV128ExtractLane(&wazeroir.OperationV128ExtractLane{
LaneIndex: tc.laneIndex,
Signed: tc.signed,
Shape: tc.shape,
})
require.NoError(t, err)
require.Equal(t, uint64(1), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
vt := compiler.runtimeValueLocationStack().peek().valueType
switch tc.shape {
case wazeroir.ShapeI8x16, wazeroir.ShapeI16x8, wazeroir.ShapeI32x4:
require.Equal(t, runtimeValueTypeI32, vt)
case wazeroir.ShapeI64x2:
require.Equal(t, runtimeValueTypeI64, vt)
case wazeroir.ShapeF32x4:
require.Equal(t, runtimeValueTypeF32, vt)
case wazeroir.ShapeF64x2:
require.Equal(t, runtimeValueTypeF64, vt)
}
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
switch tc.shape {
case wazeroir.ShapeI8x16, wazeroir.ShapeI16x8, wazeroir.ShapeI32x4, wazeroir.ShapeF32x4:
require.Equal(t, uint32(tc.exp), env.stackTopAsUint32())
case wazeroir.ShapeI64x2, wazeroir.ShapeF64x2:
require.Equal(t, tc.exp, env.stackTopAsUint64())
}
})
}
}
func TestCompiler_compileV128ReplaceLane(t *testing.T) {
tests := []struct {
name string
originValueSetupFn func(*testing.T, compilerImpl)
shape wazeroir.Shape
laneIndex byte
exp [16]byte
lo, hi uint64
}{
{
name: "i8x16 index=0",
shape: wazeroir.ShapeI8x16,
laneIndex: 5,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xff})
require.NoError(t, err)
},
exp: [16]byte{5: 0xff},
},
{
name: "i8x16 index=3",
shape: wazeroir.ShapeI8x16,
laneIndex: 5,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xff << 8})
require.NoError(t, err)
},
exp: [16]byte{},
},
{
name: "i8x16 index=5",
shape: wazeroir.ShapeI8x16,
laneIndex: 5,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xff})
require.NoError(t, err)
},
exp: [16]byte{5: 0xff},
},
{
name: "i16x8 index=0",
shape: wazeroir.ShapeI16x8,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xee_ff})
require.NoError(t, err)
},
exp: [16]byte{0: 0xff, 1: 0xee},
},
{
name: "i16x8 index=3",
shape: wazeroir.ShapeI16x8,
laneIndex: 3,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xaa_00})
require.NoError(t, err)
},
exp: [16]byte{7: 0xaa},
},
{
name: "i16x8 index=7",
shape: wazeroir.ShapeI16x8,
laneIndex: 3,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xaa_bb << 16})
require.NoError(t, err)
},
exp: [16]byte{},
},
{
name: "i32x4 index=0",
shape: wazeroir.ShapeI32x4,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xaa_bb_cc_dd})
require.NoError(t, err)
},
exp: [16]byte{0: 0xdd, 1: 0xcc, 2: 0xbb, 3: 0xaa},
},
{
name: "i32x4 index=3",
shape: wazeroir.ShapeI32x4,
laneIndex: 3,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xaa_bb_cc_dd})
require.NoError(t, err)
},
exp: [16]byte{12: 0xdd, 13: 0xcc, 14: 0xbb, 15: 0xaa},
},
{
name: "i64x2 index=0",
shape: wazeroir.ShapeI64x2,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI64(&wazeroir.OperationConstI64{Value: 0xaa_bb_cc_dd_01_02_03_04})
require.NoError(t, err)
},
exp: [16]byte{0: 0x04, 1: 0x03, 2: 0x02, 3: 0x01, 4: 0xdd, 5: 0xcc, 6: 0xbb, 7: 0xaa},
},
{
name: "i64x2 index=1",
shape: wazeroir.ShapeI64x2,
laneIndex: 1,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI64(&wazeroir.OperationConstI64{Value: 0xaa_bb_cc_dd_01_02_03_04})
require.NoError(t, err)
},
exp: [16]byte{8: 0x04, 9: 0x03, 10: 0x02, 11: 0x01, 12: 0xdd, 13: 0xcc, 14: 0xbb, 15: 0xaa},
},
{
name: "f32x4 index=0",
shape: wazeroir.ShapeF32x4,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF32(&wazeroir.OperationConstF32{Value: math.Float32frombits(0xaa_bb_cc_dd)})
require.NoError(t, err)
},
exp: [16]byte{0: 0xdd, 1: 0xcc, 2: 0xbb, 3: 0xaa},
},
{
name: "f32x4 index=1",
shape: wazeroir.ShapeF32x4,
laneIndex: 1,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF32(&wazeroir.OperationConstF32{Value: math.Float32frombits(0xaa_bb_cc_dd)})
require.NoError(t, err)
},
exp: [16]byte{4: 0xdd, 5: 0xcc, 6: 0xbb, 7: 0xaa},
},
{
name: "f32x4 index=2",
shape: wazeroir.ShapeF32x4,
laneIndex: 2,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF32(&wazeroir.OperationConstF32{Value: math.Float32frombits(0xaa_bb_cc_dd)})
require.NoError(t, err)
},
exp: [16]byte{8: 0xdd, 9: 0xcc, 10: 0xbb, 11: 0xaa},
},
{
name: "f32x4 index=3",
shape: wazeroir.ShapeF32x4,
laneIndex: 3,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF32(&wazeroir.OperationConstF32{Value: math.Float32frombits(0xaa_bb_cc_dd)})
require.NoError(t, err)
},
exp: [16]byte{12: 0xdd, 13: 0xcc, 14: 0xbb, 15: 0xaa},
},
{
name: "f64x2 index=0",
shape: wazeroir.ShapeF64x2,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF64(&wazeroir.OperationConstF64{Value: math.Float64frombits(0xaa_bb_cc_dd_01_02_03_04)})
require.NoError(t, err)
},
exp: [16]byte{0: 0x04, 1: 0x03, 2: 0x02, 3: 0x01, 4: 0xdd, 5: 0xcc, 6: 0xbb, 7: 0xaa},
},
{
name: "f64x2 index=1",
shape: wazeroir.ShapeF64x2,
laneIndex: 1,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF64(&wazeroir.OperationConstF64{Value: math.Float64frombits(0xaa_bb_cc_dd_01_02_03_04)})
require.NoError(t, err)
},
exp: [16]byte{8: 0x04, 9: 0x03, 10: 0x02, 11: 0x01, 12: 0xdd, 13: 0xcc, 14: 0xbb, 15: 0xaa},
},
{
name: "f64x2 index=0 / lo,hi = 1.0",
shape: wazeroir.ShapeF64x2,
laneIndex: 0,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF64(&wazeroir.OperationConstF64{Value: math.Float64frombits(0.0)})
require.NoError(t, err)
},
lo: math.Float64bits(1.0),
hi: math.Float64bits(1.0),
exp: [16]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xf0, 0x3f},
},
{
name: "f64x2 index=1 / lo,hi = 1.0",
shape: wazeroir.ShapeF64x2,
laneIndex: 1,
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF64(&wazeroir.OperationConstF64{Value: math.Float64frombits(0.0)})
require.NoError(t, err)
},
lo: math.Float64bits(1.0),
hi: math.Float64bits(1.0),
exp: [16]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xf0, 0x3f, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{Lo: tc.lo, Hi: tc.hi})
require.NoError(t, err)
tc.originValueSetupFn(t, compiler)
err = compiler.compileV128ReplaceLane(&wazeroir.OperationV128ReplaceLane{
LaneIndex: tc.laneIndex,
Shape: tc.shape,
})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}
func TestCompiler_compileV128Splat(t *testing.T) {
tests := []struct {
name string
originValueSetupFn func(*testing.T, compilerImpl)
shape wazeroir.Shape
exp [16]byte
}{
{
name: "i8x16",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0x1})
require.NoError(t, err)
},
shape: wazeroir.ShapeI8x16,
exp: [16]byte{0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, 0x1},
},
{
name: "i16x8",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xff_11})
require.NoError(t, err)
},
shape: wazeroir.ShapeI16x8,
exp: [16]byte{0x11, 0xff, 0x11, 0xff, 0x11, 0xff, 0x11, 0xff, 0x11, 0xff, 0x11, 0xff, 0x11, 0xff, 0x11, 0xff},
},
{
name: "i32x4",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI32(&wazeroir.OperationConstI32{Value: 0xff_11_ee_22})
require.NoError(t, err)
},
shape: wazeroir.ShapeI32x4,
exp: [16]byte{0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff},
},
{
name: "i64x2",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstI64(&wazeroir.OperationConstI64{Value: 0xff_00_ee_00_11_00_22_00})
require.NoError(t, err)
},
shape: wazeroir.ShapeI64x2,
exp: [16]byte{0x00, 0x22, 0x00, 0x11, 0x00, 0xee, 0x00, 0xff, 0x00, 0x22, 0x00, 0x11, 0x00, 0xee, 0x00, 0xff},
},
{
name: "f32x4",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF32(&wazeroir.OperationConstF32{Value: math.Float32frombits(0xff_11_ee_22)})
require.NoError(t, err)
},
shape: wazeroir.ShapeF32x4,
exp: [16]byte{0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff, 0x22, 0xee, 0x11, 0xff},
},
{
name: "f64x2",
originValueSetupFn: func(t *testing.T, c compilerImpl) {
err := c.compileConstF64(&wazeroir.OperationConstF64{Value: math.Float64frombits(0xff_00_ee_00_11_00_22_00)})
require.NoError(t, err)
},
shape: wazeroir.ShapeF64x2,
exp: [16]byte{0x00, 0x22, 0x00, 0x11, 0x00, 0xee, 0x00, 0xff, 0x00, 0x22, 0x00, 0x11, 0x00, 0xee, 0x00, 0xff},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
tc.originValueSetupFn(t, compiler)
err = compiler.compileV128Splat(&wazeroir.OperationV128Splat{Shape: tc.shape})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}
func TestCompiler_compileV128AnyTrue(t *testing.T) {
tests := []struct {
name string
lo, hi uint64
exp uint32
}{
{name: "lo == 0 && hi == 0", lo: 0, hi: 0, exp: 0},
{name: "lo != 0", lo: 1, exp: 1},
{name: "hi != 0", hi: 1, exp: 1},
{name: "lo != 0 && hi != 0", lo: 1, hi: 1, exp: 1},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{Lo: tc.lo, Hi: tc.hi})
require.NoError(t, err)
err = compiler.compileV128AnyTrue(&wazeroir.OperationV128AnyTrue{})
require.NoError(t, err)
require.Equal(t, uint64(1), compiler.runtimeValueLocationStack().sp)
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, nativeCallStatusCodeReturned, env.callEngine().statusCode)
require.Equal(t, uint64(1), env.stackPointer())
require.Equal(t, tc.exp, env.stackTopAsUint32())
})
}
}
func TestCompiler_compileV128AllTrue(t *testing.T) {
tests := []struct {
name string
shape wazeroir.Shape
lo, hi uint64
exp uint32
}{
{
name: "i8x16 - true",
shape: wazeroir.ShapeI8x16,
lo: 0xffff_ffff_ffff_ffff,
hi: 0x0101_0101_0101_0101,
exp: 1,
},
{
name: "i8x16 - false on lo",
shape: wazeroir.ShapeI8x16,
lo: 0xffff_ffff_ffff_ffff,
hi: 0x1111_1111_0011_1111,
exp: 0,
},
{
name: "i8x16 - false on hi",
shape: wazeroir.ShapeI8x16,
lo: 0xffff_00ff_ffff_ffff,
hi: 0x1111_1111_1111_1111,
exp: 0,
},
{
name: "i16x8 - true",
shape: wazeroir.ShapeI16x8,
lo: 0x1000_0100_0010_0001,
hi: 0x0101_0101_0101_0101,
exp: 1,
},
{
name: "i16x8 - false on hi",
shape: wazeroir.ShapeI16x8,
lo: 0x1000_0100_0010_0001,
hi: 0x1111_1111_0000_1111,
exp: 0,
},
{
name: "i16x8 - false on lo",
shape: wazeroir.ShapeI16x8,
lo: 0xffff_0000_ffff_ffff,
hi: 0x1111_1111_1111_1111,
exp: 0,
},
{
name: "i32x4 - true",
shape: wazeroir.ShapeI32x4,
lo: 0x1000_0000_0010_0000,
hi: 0x0000_0001_0000_1000,
exp: 1,
},
{
name: "i32x4 - true",
shape: wazeroir.ShapeI32x4,
lo: 0x0000_1111_1111_0000,
hi: 0x0000_0001_1000_0000,
exp: 1,
},
{
name: "i32x4 - false on lo",
shape: wazeroir.ShapeI32x4,
lo: 0x1111_1111_0000_0000,
hi: 0x1111_1111_1111_1111,
exp: 0,
},
{
name: "i32x4 - false on lo",
shape: wazeroir.ShapeI32x4,
lo: 0x0000_0000_1111_1111,
hi: 0x1111_1111_1111_1111,
exp: 0,
},
{
name: "i32x4 - false on hi",
shape: wazeroir.ShapeI32x4,
lo: 0x1111_1111_1111_1111,
hi: 0x1111_1111_0000_0000,
exp: 0,
},
{
name: "i32x4 - false on hi",
shape: wazeroir.ShapeI32x4,
lo: 0x1111_1111_1111_1111,
hi: 0x0000_0000_1111_1111,
exp: 0,
},
{
name: "i64x2 - true",
shape: wazeroir.ShapeI64x2,
lo: 0x1000_0000_0000_0000,
hi: 0x0000_0001_0000_0000,
exp: 1,
},
{
name: "i64x2 - true",
shape: wazeroir.ShapeI64x2,
lo: 0x0000_0000_0010_0000,
hi: 0x0000_0000_0000_0100,
exp: 1,
},
{
name: "i64x2 - true",
shape: wazeroir.ShapeI64x2,
lo: 0x0000_0000_0000_1000,
hi: 0x1000_0000_0000_0000,
exp: 1,
},
{
name: "i64x2 - false on lo",
shape: wazeroir.ShapeI64x2,
lo: 0,
hi: 0x1111_1111_1111_1111,
exp: 0,
},
{
name: "i64x2 - false on hi",
shape: wazeroir.ShapeI64x2,
lo: 0x1111_1111_1111_1111,
hi: 0,
exp: 0,
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{Lo: tc.lo, Hi: tc.hi})
require.NoError(t, err)
err = compiler.compileV128AllTrue(&wazeroir.OperationV128AllTrue{Shape: tc.shape})
require.NoError(t, err)
require.Equal(t, 0, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, nativeCallStatusCodeReturned, env.callEngine().statusCode)
require.Equal(t, uint64(1), env.stackPointer())
require.Equal(t, tc.exp, env.stackTopAsUint32())
})
}
}
func i8ToU8(v int8) byte {
return byte(v)
}
func TestCompiler_compileV128Swizzle(t *testing.T) {
tests := []struct {
name string
indexVec, baseVec [16]byte
expVec [16]byte
}{
{
name: "1",
baseVec: [16]byte{16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31},
indexVec: [16]byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15},
expVec: [16]byte{16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31},
},
{
name: "2",
baseVec: [16]byte{i8ToU8(-16), i8ToU8(-15), i8ToU8(-14), i8ToU8(-13), i8ToU8(-12),
i8ToU8(-11), i8ToU8(-10), i8ToU8(-9), i8ToU8(-8), i8ToU8(-7), i8ToU8(-6), i8ToU8(-5),
i8ToU8(-4), i8ToU8(-3), i8ToU8(-2), i8ToU8(-1)},
indexVec: [16]byte{i8ToU8(-8), i8ToU8(-7), i8ToU8(-6), i8ToU8(-5), i8ToU8(-4),
i8ToU8(-3), i8ToU8(-2), i8ToU8(-1), 16, 17, 18, 19, 20, 21, 22, 23},
expVec: [16]byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
},
{
name: "3",
baseVec: [16]byte{100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115},
indexVec: [16]byte{15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0},
expVec: [16]byte{115, 114, 113, 112, 111, 110, 109, 108, 107, 106, 105, 104, 103, 102, 101, 100},
},
{
name: "4",
baseVec: [16]byte{100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115},
indexVec: [16]byte{
9, 16, 10, 17, 11, 18, 12, 19, 13, 20, 14, 21, 15, 22, 16, 23,
},
expVec: [16]byte{109, 0, 110, 0, 111, 0, 112, 0, 113, 0, 114, 0, 115, 0, 0, 0},
},
{
name: "5",
baseVec: [16]byte{0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x73},
indexVec: [16]byte{9, 16, 10, 17, 11, 18, 12, 19, 13, 20, 14, 21, 15, 22, 16, 23},
expVec: [16]byte{0x6d, 0, 0x6e, 0, 0x6f, 0, 0x70, 0, 0x71, 0, 0x72, 0, 0x73, 0, 0, 0},
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.baseVec[:8]),
Hi: binary.LittleEndian.Uint64(tc.baseVec[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.indexVec[:8]),
Hi: binary.LittleEndian.Uint64(tc.indexVec[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Swizzle(&wazeroir.OperationV128Swizzle{})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
require.Equal(t, nativeCallStatusCodeReturned, env.callEngine().statusCode)
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.expVec, actual)
})
}
}
func TestCompiler_compileV128Shuffle(t *testing.T) {
tests := []struct {
name string
lanes, w, v, exp [16]byte
}{
{
name: "v only",
lanes: [16]byte{1, 1, 1, 1, 0, 0, 0, 0, 10, 10, 10, 10, 0, 0, 0, 0},
v: [16]byte{0: 0xa, 1: 0xb, 10: 0xc},
w: [16]byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
exp: [16]byte{
0xb, 0xb, 0xb, 0xb,
0xa, 0xa, 0xa, 0xa,
0xc, 0xc, 0xc, 0xc,
0xa, 0xa, 0xa, 0xa,
},
},
{
name: "w only",
lanes: [16]byte{17, 17, 17, 17, 16, 16, 16, 16, 26, 26, 26, 26, 16, 16, 16, 16},
v: [16]byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
w: [16]byte{0: 0xa, 1: 0xb, 10: 0xc},
exp: [16]byte{
0xb, 0xb, 0xb, 0xb,
0xa, 0xa, 0xa, 0xa,
0xc, 0xc, 0xc, 0xc,
0xa, 0xa, 0xa, 0xa,
},
},
{
name: "mix",
lanes: [16]byte{0, 17, 2, 19, 4, 21, 6, 23, 8, 25, 10, 27, 12, 29, 14, 31},
v: [16]byte{
0x1, 0xff, 0x2, 0xff, 0x3, 0xff, 0x4, 0xff,
0x5, 0xff, 0x6, 0xff, 0x7, 0xff, 0x8, 0xff,
},
w: [16]byte{
0xff, 0x11, 0xff, 0x12, 0xff, 0x13, 0xff, 0x14,
0xff, 0x15, 0xff, 0x16, 0xff, 0x17, 0xff, 0x18,
},
exp: [16]byte{
0x1, 0x11, 0x2, 0x12, 0x3, 0x13, 0x4, 0x14,
0x5, 0x15, 0x6, 0x16, 0x7, 0x17, 0x8, 0x18,
},
},
{
name: "mix",
lanes: [16]byte{0, 17, 2, 19, 4, 21, 6, 23, 8, 25, 10, 27, 12, 29, 14, 31},
v: [16]byte{
0x1, 0xff, 0x2, 0xff, 0x3, 0xff, 0x4, 0xff,
0x5, 0xff, 0x6, 0xff, 0x7, 0xff, 0x8, 0xff,
},
w: [16]byte{
0xff, 0x11, 0xff, 0x12, 0xff, 0x13, 0xff, 0x14,
0xff, 0x15, 0xff, 0x16, 0xff, 0x17, 0xff, 0x18,
},
exp: [16]byte{
0x1, 0x11, 0x2, 0x12, 0x3, 0x13, 0x4, 0x14,
0x5, 0x15, 0x6, 0x16, 0x7, 0x17, 0x8, 0x18,
},
},
}
/*
*/
for _, tc := range tests {
tc := tc
t.Run(tc.name, func(t *testing.T) {
env := newCompilerEnvironment()
compiler := env.requireNewCompiler(t, newCompiler,
&wazeroir.CompilationResult{HasMemory: true, Signature: &wasm.FunctionType{}})
err := compiler.compilePreamble()
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.v[:8]),
Hi: binary.LittleEndian.Uint64(tc.v[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Const(&wazeroir.OperationV128Const{
Lo: binary.LittleEndian.Uint64(tc.w[:8]),
Hi: binary.LittleEndian.Uint64(tc.w[8:]),
})
require.NoError(t, err)
err = compiler.compileV128Shuffle(&wazeroir.OperationV128Shuffle{Lanes: tc.lanes})
require.NoError(t, err)
require.Equal(t, uint64(2), compiler.runtimeValueLocationStack().sp)
require.Equal(t, 1, len(compiler.runtimeValueLocationStack().usedRegisters))
err = compiler.compileReturnFunction()
require.NoError(t, err)
// Generate and run the code under test.
code, _, _, err := compiler.compile()
require.NoError(t, err)
env.exec(code)
lo, hi := env.stackTopAsV128()
var actual [16]byte
binary.LittleEndian.PutUint64(actual[:8], lo)
binary.LittleEndian.PutUint64(actual[8:], hi)
require.Equal(t, tc.exp, actual)
})
}
}