We at one point considered making `ModuleBuilder` create complete WebAssembly binaries. However, we recently spun out [wabin](https://github.com/tetratelabs/wabin), which allows this. Meanwhile, the features in `ModuleBuilder` were confusing and misused. For example, the only two cases memory was exported on GitHub were done by accident. This is because host functions act on the guest's memory, not their own. Hence, this removes memory and globals from host side definitions, and renames the type to HostModuleBuilder to clarify this is not ever going to be used to construct normal Wasm binaries. Most importantly, this simplifies the API and reduces a lot of code. It is important to make changes like this, particularly deleting any experimental things that didn't end up useful. Signed-off-by: Adrian Cole <adrian@tetrate.io> Co-authored-by: Anuraag Agrawal <anuraaga@gmail.com>
217 lines
6.1 KiB
Go
217 lines
6.1 KiB
Go
package vs
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/tetratelabs/wazero"
|
|
"github.com/tetratelabs/wazero/api"
|
|
"github.com/tetratelabs/wazero/imports/wasi_snapshot_preview1"
|
|
"github.com/tetratelabs/wazero/internal/wasm"
|
|
)
|
|
|
|
type RuntimeConfig struct {
|
|
Name string
|
|
ModuleName string
|
|
ModuleWasm []byte
|
|
FuncNames []string
|
|
NeedsWASI bool
|
|
NeedsMemoryExport 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
|
|
// EnvFReturnValue is set to non-zero if we want the runtime to instantiate "env" module with the function "f"
|
|
// which accepts one i64 value and returns the EnvFReturnValue as i64. This is mutually exclusive to LogFn.
|
|
EnvFReturnValue uint64
|
|
}
|
|
|
|
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
|
|
CallV_V(ctx context.Context, funcName string) error
|
|
CallI64_I64(ctx context.Context, funcName string, param uint64) (uint64, error)
|
|
WriteMemory(ctx context.Context, offset uint32, bytes []byte) error
|
|
Memory() []byte
|
|
Close(context.Context) error
|
|
}
|
|
|
|
func NewWazeroInterpreterRuntime() Runtime {
|
|
return newWazeroRuntime("wazero-interpreter", wazero.NewRuntimeConfigInterpreter())
|
|
}
|
|
|
|
func NewWazeroCompilerRuntime() Runtime {
|
|
return newWazeroRuntime(compilerRuntime, wazero.NewRuntimeConfigCompiler())
|
|
}
|
|
|
|
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.CompiledModule
|
|
}
|
|
|
|
type wazeroModule struct {
|
|
wasi api.Closer
|
|
env, mod api.Module
|
|
funcs map[string]api.Function
|
|
}
|
|
|
|
func (r *wazeroRuntime) Name() string {
|
|
return r.name
|
|
}
|
|
|
|
func (m *wazeroModule) Memory() []byte {
|
|
return m.mod.Memory().(*wasm.MemoryInstance).Buffer
|
|
}
|
|
|
|
func (r *wazeroRuntime) log(ctx context.Context, m api.Module, offset, byteCount uint32) {
|
|
buf, ok := m.Memory().Read(ctx, offset, byteCount)
|
|
if !ok {
|
|
panic("out of memory reading log buffer")
|
|
}
|
|
if err := r.logFn(buf); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
func (r *wazeroRuntime) Compile(ctx context.Context, cfg *RuntimeConfig) (err error) {
|
|
r.runtime = wazero.NewRuntimeWithConfig(ctx, r.config)
|
|
if cfg.LogFn != nil {
|
|
r.logFn = cfg.LogFn
|
|
if r.env, err = r.runtime.NewHostModuleBuilder("env").
|
|
ExportFunction("log", r.log).Compile(ctx); err != nil {
|
|
return err
|
|
}
|
|
} else if cfg.EnvFReturnValue != 0 {
|
|
if r.env, err = r.runtime.NewHostModuleBuilder("env").
|
|
ExportFunction("f",
|
|
// Note: accepting (context.Context, api.Module) is the slowest type of host function with wazero.
|
|
func(context.Context, api.Module, uint64) uint64 {
|
|
return cfg.EnvFReturnValue
|
|
},
|
|
).Compile(ctx); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
r.compiled, err = r.runtime.CompileModule(ctx, cfg.ModuleWasm, wazero.NewCompileConfig())
|
|
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_snapshot_preview1.Instantiate(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, wazero.NewModuleConfig()); err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
// Instantiate the module.
|
|
if m.mod, err = r.runtime.InstantiateModule(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) CallV_V(ctx context.Context, funcName string) (err error) {
|
|
_, err = m.funcs[funcName].Call(ctx)
|
|
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 errors.New("out of memory writing name")
|
|
}
|
|
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
|
|
}
|