This makes wazero.CompiledCode an interface instead of a struct to prevent it from being used incorrectly. For example, even though the fields are not exported, someone can mistakenly instantiate this when it is a struct, and in doing so violate internal assumptions. Signed-off-by: Adrian Cole <adrian@tetrate.io>
189 lines
5.1 KiB
Go
189 lines
5.1 KiB
Go
package vs
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/tetratelabs/wazero"
|
|
"github.com/tetratelabs/wazero/api"
|
|
"github.com/tetratelabs/wazero/wasi"
|
|
)
|
|
|
|
type RuntimeConfig struct {
|
|
Name string
|
|
ModuleName string
|
|
ModuleWasm []byte
|
|
FuncNames []string
|
|
NeedsWASI bool
|
|
// LogFn requires the implementation to export a function "env.log" which accepts i32i32_v.
|
|
// The implementation invoke this with a byte slice allocated from the offset, length pair.
|
|
// This function simulates a host function that logs a message.
|
|
LogFn func([]byte) error
|
|
}
|
|
|
|
type Runtime interface {
|
|
Name() string
|
|
Compile(context.Context, *RuntimeConfig) error
|
|
Instantiate(context.Context, *RuntimeConfig) (Module, error)
|
|
Close(context.Context) error
|
|
}
|
|
|
|
type Module interface {
|
|
CallI32_I32(ctx context.Context, funcName string, param uint32) (uint32, error)
|
|
CallI32I32_V(ctx context.Context, funcName string, x, y uint32) error
|
|
CallI32_V(ctx context.Context, funcName string, param uint32) error
|
|
CallI64_I64(ctx context.Context, funcName string, param uint64) (uint64, error)
|
|
WriteMemory(ctx context.Context, offset uint32, bytes []byte) error
|
|
Close(context.Context) error
|
|
}
|
|
|
|
func NewWazeroInterpreterRuntime() Runtime {
|
|
return newWazeroRuntime("wazero-interpreter", wazero.NewRuntimeConfigInterpreter().WithWasmCore2())
|
|
}
|
|
|
|
func NewWazeroJITRuntime() Runtime {
|
|
return newWazeroRuntime(jitRuntime, wazero.NewRuntimeConfigJIT().WithWasmCore2())
|
|
}
|
|
|
|
func newWazeroRuntime(name string, config wazero.RuntimeConfig) *wazeroRuntime {
|
|
return &wazeroRuntime{name: name, config: config}
|
|
}
|
|
|
|
type wazeroRuntime struct {
|
|
name string
|
|
config wazero.RuntimeConfig
|
|
runtime wazero.Runtime
|
|
logFn func([]byte) error
|
|
env, compiled wazero.CompiledCode
|
|
}
|
|
|
|
type wazeroModule struct {
|
|
wasi, env, mod api.Module
|
|
funcs map[string]api.Function
|
|
}
|
|
|
|
func (r *wazeroRuntime) Name() string {
|
|
return r.name
|
|
}
|
|
|
|
func (r *wazeroRuntime) log(ctx context.Context, m api.Module, offset, byteCount uint32) {
|
|
buf, ok := m.Memory().Read(ctx, offset, byteCount)
|
|
if !ok {
|
|
panic(fmt.Errorf("Memory.Read(%d, %d) out of range", offset, byteCount))
|
|
}
|
|
if err := r.logFn(buf); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
func (r *wazeroRuntime) Compile(ctx context.Context, cfg *RuntimeConfig) (err error) {
|
|
r.runtime = wazero.NewRuntimeWithConfig(r.config)
|
|
if cfg.LogFn != nil {
|
|
r.logFn = cfg.LogFn
|
|
if r.env, err = r.runtime.NewModuleBuilder("env").
|
|
ExportFunction("log", r.log).Build(ctx); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
r.compiled, err = r.runtime.CompileModule(ctx, cfg.ModuleWasm)
|
|
return
|
|
}
|
|
|
|
func (r *wazeroRuntime) Instantiate(ctx context.Context, cfg *RuntimeConfig) (mod Module, err error) {
|
|
wazeroCfg := wazero.NewModuleConfig().WithName(cfg.ModuleName)
|
|
m := &wazeroModule{funcs: map[string]api.Function{}}
|
|
|
|
// Instantiate WASI, if configured.
|
|
if cfg.NeedsWASI {
|
|
if m.wasi, err = wasi.InstantiateSnapshotPreview1(ctx, r.runtime); err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
// Instantiate the host module, "env", if configured.
|
|
if env := r.env; env != nil {
|
|
if m.env, err = r.runtime.InstantiateModule(ctx, env); err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
// Instantiate the module.
|
|
if m.mod, err = r.runtime.InstantiateModuleWithConfig(ctx, r.compiled, wazeroCfg); err != nil {
|
|
return
|
|
}
|
|
|
|
// Ensure function exports exist.
|
|
for _, funcName := range cfg.FuncNames {
|
|
if fn := m.mod.ExportedFunction(funcName); fn == nil {
|
|
return nil, fmt.Errorf("%s is not an exported function", funcName)
|
|
} else {
|
|
m.funcs[funcName] = fn
|
|
}
|
|
}
|
|
mod = m
|
|
return
|
|
}
|
|
|
|
func (r *wazeroRuntime) Close(ctx context.Context) (err error) {
|
|
if compiled := r.compiled; compiled != nil {
|
|
err = compiled.Close(ctx)
|
|
}
|
|
r.compiled = nil
|
|
if env := r.env; env != nil {
|
|
err = env.Close(ctx)
|
|
}
|
|
r.env = nil
|
|
return
|
|
}
|
|
|
|
func (m *wazeroModule) CallI32_I32(ctx context.Context, funcName string, param uint32) (uint32, error) {
|
|
if results, err := m.funcs[funcName].Call(ctx, uint64(param)); err != nil {
|
|
return 0, err
|
|
} else if len(results) > 0 {
|
|
return uint32(results[0]), nil
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func (m *wazeroModule) CallI32I32_V(ctx context.Context, funcName string, x, y uint32) (err error) {
|
|
_, err = m.funcs[funcName].Call(ctx, uint64(x), uint64(y))
|
|
return
|
|
}
|
|
|
|
func (m *wazeroModule) CallI32_V(ctx context.Context, funcName string, param uint32) (err error) {
|
|
_, err = m.funcs[funcName].Call(ctx, uint64(param))
|
|
return
|
|
}
|
|
|
|
func (m *wazeroModule) CallI64_I64(ctx context.Context, funcName string, param uint64) (uint64, error) {
|
|
if results, err := m.funcs[funcName].Call(ctx, param); err != nil {
|
|
return 0, err
|
|
} else if len(results) > 0 {
|
|
return results[0], nil
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func (m *wazeroModule) WriteMemory(ctx context.Context, offset uint32, bytes []byte) error {
|
|
if !m.mod.Memory().Write(ctx, offset, bytes) {
|
|
return fmt.Errorf("Memory.Write(%d, %d) out of range of memory size %d",
|
|
offset, len(bytes), m.mod.Memory().Size(ctx))
|
|
}
|
|
return nil
|
|
}
|
|
func (m *wazeroModule) Close(ctx context.Context) (err error) {
|
|
if mod := m.mod; mod != nil {
|
|
err = mod.Close(ctx)
|
|
}
|
|
m.mod = nil
|
|
if env := m.env; env != nil {
|
|
err = env.Close(ctx)
|
|
}
|
|
m.env = nil
|
|
if wasi := m.wasi; wasi != nil {
|
|
err = wasi.Close(ctx)
|
|
}
|
|
m.wasi = nil
|
|
return
|
|
}
|