Files
wazero/internal/integration_test/bench/bench_test.go
Crypt Keeper a91140f7f7 Changes RuntimeConfig to an interface and exposes WithWasmCore2 (#518)
WebAssembly Core Working Draft 1 recently came out. Before that, we had
a toe-hold feature bucked called FinishedFeatures. This replaces
`RuntimeConfig.WithFinishedFeatures` with `RuntimeConfig.WithWasmCore2`.
This also adds `WithWasmCore1` for those who want to lock into 1.0
features as opposed to relying on defaults.

This also fixes some design debt where we hadn't finished migrating
public types that require constructor functions (NewXxx) to interfaces.
By using interfaces, we prevent people from accidentally initializing
key configuration directly (via &Xxx), causing nil field refs. This also
helps prevent confusion about how to use the type (ex pointer or not) as
there's only one way (as an interface).

See https://github.com/tetratelabs/wazero/issues/516

Signed-off-by: Adrian Cole <adrian@tetrate.io>
2022-05-02 10:29:38 +08:00

199 lines
5.1 KiB
Go

package bench
import (
"context"
_ "embed"
"fmt"
"math/rand"
"runtime"
"testing"
"github.com/tetratelabs/wazero"
"github.com/tetratelabs/wazero/api"
"github.com/tetratelabs/wazero/wasi"
)
// testCtx is an arbitrary, non-default context. Non-nil also prevents linter errors.
var testCtx = context.WithValue(context.Background(), struct{}{}, "arbitrary")
// caseWasm was compiled from TinyGo testdata/case.go
//go:embed testdata/case.wasm
var caseWasm []byte
func BenchmarkInvocation(b *testing.B) {
b.Run("interpreter", func(b *testing.B) {
m := instantiateHostFunctionModuleWithEngine(b, wazero.NewRuntimeConfigInterpreter())
defer m.Close(testCtx)
runAllInvocationBenches(b, m)
})
if runtime.GOARCH == "amd64" || runtime.GOARCH == "arm64" {
b.Run("jit", func(b *testing.B) {
m := instantiateHostFunctionModuleWithEngine(b, wazero.NewRuntimeConfigJIT())
defer m.Close(testCtx)
runAllInvocationBenches(b, m)
})
}
}
func BenchmarkInitialization(b *testing.B) {
b.Run("interpreter", func(b *testing.B) {
r := createRuntime(b, wazero.NewRuntimeConfigInterpreter())
runInitializationBench(b, r)
})
if runtime.GOARCH == "amd64" || runtime.GOARCH == "arm64" {
b.Run("jit", func(b *testing.B) {
r := createRuntime(b, wazero.NewRuntimeConfigJIT())
runInitializationBench(b, r)
})
}
}
func runInitializationBench(b *testing.B, r wazero.Runtime) {
compiled, err := r.CompileModule(testCtx, caseWasm)
if err != nil {
b.Fatal(err)
}
defer compiled.Close(testCtx)
b.ResetTimer()
for i := 0; i < b.N; i++ {
mod, err := r.InstantiateModule(testCtx, compiled)
if err != nil {
b.Fatal(err)
}
mod.Close(testCtx)
}
}
func runAllInvocationBenches(b *testing.B, m api.Module) {
runBase64Benches(b, m)
runFibBenches(b, m)
runStringManipulationBenches(b, m)
runReverseArrayBenches(b, m)
runRandomMatMul(b, m)
}
func runBase64Benches(b *testing.B, m api.Module) {
base64 := m.ExportedFunction("base64")
for _, numPerExec := range []int{5, 100, 10000} {
numPerExec := uint64(numPerExec)
b.ResetTimer()
b.Run(fmt.Sprintf("base64_%d_per_exec", numPerExec), func(b *testing.B) {
for i := 0; i < b.N; i++ {
if _, err := base64.Call(testCtx, numPerExec); err != nil {
b.Fatal(err)
}
}
})
}
}
func runFibBenches(b *testing.B, m api.Module) {
fibonacci := m.ExportedFunction("fibonacci")
for _, num := range []int{5, 10, 20, 30} {
num := uint64(num)
b.ResetTimer()
b.Run(fmt.Sprintf("fib_for_%d", num), func(b *testing.B) {
for i := 0; i < b.N; i++ {
if _, err := fibonacci.Call(testCtx, num); err != nil {
b.Fatal(err)
}
}
})
}
}
func runStringManipulationBenches(b *testing.B, m api.Module) {
stringManipulation := m.ExportedFunction("string_manipulation")
for _, initialSize := range []int{50, 100, 1000} {
initialSize := uint64(initialSize)
b.ResetTimer()
b.Run(fmt.Sprintf("string_manipulation_size_%d", initialSize), func(b *testing.B) {
for i := 0; i < b.N; i++ {
if _, err := stringManipulation.Call(testCtx, initialSize); err != nil {
b.Fatal(err)
}
}
})
}
}
func runReverseArrayBenches(b *testing.B, m api.Module) {
reverseArray := m.ExportedFunction("reverse_array")
for _, arraySize := range []int{500, 1000, 10000} {
arraySize := uint64(arraySize)
b.ResetTimer()
b.Run(fmt.Sprintf("reverse_array_size_%d", arraySize), func(b *testing.B) {
for i := 0; i < b.N; i++ {
if _, err := reverseArray.Call(testCtx, arraySize); err != nil {
b.Fatal(err)
}
}
})
}
}
func runRandomMatMul(b *testing.B, m api.Module) {
randomMatMul := m.ExportedFunction("random_mat_mul")
for _, matrixSize := range []int{5, 10, 20} {
matrixSize := uint64(matrixSize)
b.ResetTimer()
b.Run(fmt.Sprintf("random_mat_mul_size_%d", matrixSize), func(b *testing.B) {
for i := 0; i < b.N; i++ {
if _, err := randomMatMul.Call(testCtx, matrixSize); err != nil {
b.Fatal(err)
}
}
})
}
}
func instantiateHostFunctionModuleWithEngine(b *testing.B, config wazero.RuntimeConfig) api.Module {
r := createRuntime(b, config)
// InstantiateModuleFromCode runs the "_start" function which is what TinyGo compiles "main" to.
m, err := r.InstantiateModuleFromCode(testCtx, caseWasm)
if err != nil {
b.Fatal(err)
}
return m
}
func createRuntime(b *testing.B, config wazero.RuntimeConfig) wazero.Runtime {
getRandomString := func(ctx context.Context, m api.Module, retBufPtr uint32, retBufSize uint32) {
results, err := m.ExportedFunction("allocate_buffer").Call(ctx, 10)
if err != nil {
b.Fatal(err)
}
offset := uint32(results[0])
m.Memory().WriteUint32Le(ctx, retBufPtr, offset)
m.Memory().WriteUint32Le(ctx, retBufSize, 10)
b := make([]byte, 10)
_, _ = rand.Read(b)
m.Memory().Write(ctx, offset, b)
}
r := wazero.NewRuntimeWithConfig(config)
_, err := r.NewModuleBuilder("env").
ExportFunction("get_random_string", getRandomString).
Instantiate(testCtx)
if err != nil {
b.Fatal(err)
}
// Note: host_func.go doesn't directly use WASI, but TinyGo needs to be initialized as a WASI Command.
// Add WASI to satisfy import tests
_, err = wasi.InstantiateSnapshotPreview1(testCtx, r)
if err != nil {
b.Fatal(err)
}
return r
}