Files
wazero/internal/integration_test/vs/wasmedge_test.go
Crypt Keeper 45ff2fe12f Propagates context to all api interface methods that aren't constant (#502)
This prepares for exposing operations like Memory.Grow while keeping the
ability to trace what did that, by adding a `context.Context` initial
parameter. This adds this to all API methods that mutate or return
mutated data.

Before, we made a change to trace functions and general lifecycle
commands, but we missed this part. Ex. We track functions, but can't
track what closed the module, changed memory or a mutable constant.
Changing to do this now is not only more consistent, but helps us
optimize at least the interpreter to help users identify otherwise
opaque code that can cause harm. This is critical before we add more
functions that can cause harm, such as Memory.Grow.

Signed-off-by: Adrian Cole <adrian@tetrate.io>
2022-04-25 08:13:18 +08:00

89 lines
1.9 KiB
Go

//go:build amd64 && cgo && wasmedge
// wasmedge depends on manual installation of a shared library, so is guarded by a flag by default.
package vs
import (
"context"
"fmt"
"github.com/second-state/WasmEdge-go/wasmedge"
)
func init() {
runtimes["WasmEdge-go"] = newWasmedgeRuntime
}
func newWasmedgeRuntime() runtime {
return &wasmedgeRuntime{}
}
type wasmedgeRuntime struct {
conf *wasmedge.Configure
}
type wasmedgeModule struct {
store *wasmedge.Store
vm *wasmedge.VM
}
func (r *wasmedgeRuntime) Compile(_ context.Context, _ *runtimeConfig) (err error) {
r.conf = wasmedge.NewConfigure()
// We can't re-use a store because "module name conflict" occurs even after releasing a VM
return
}
func (r *wasmedgeRuntime) Instantiate(_ context.Context, cfg *runtimeConfig) (mod module, err error) {
m := &wasmedgeModule{store: wasmedge.NewStore()}
m.vm = wasmedge.NewVMWithConfigAndStore(r.conf, m.store)
if err = m.vm.RegisterWasmBuffer(cfg.moduleName, cfg.moduleWasm); err != nil {
return
}
if err = m.vm.LoadWasmBuffer(cfg.moduleWasm); err != nil {
return
}
if err = m.vm.Validate(); err != nil {
return
}
if err = m.vm.Instantiate(); err != nil {
return
}
for _, funcName := range cfg.funcNames {
if fn := m.vm.GetFunctionType(funcName); fn == nil {
err = fmt.Errorf("%s is not an exported function", funcName)
return
}
}
mod = m
return
}
func (r *wasmedgeRuntime) Close(_ context.Context) error {
if conf := r.conf; conf != nil {
conf.Release()
}
r.conf = nil
return nil
}
func (m *wasmedgeModule) CallI64_I64(_ context.Context, funcName string, param uint64) (uint64, error) {
if result, err := m.vm.Execute(funcName, int64(param)); err != nil {
return 0, err
} else {
return uint64(result[0].(int64)), nil
}
}
func (m *wasmedgeModule) Close(_ context.Context) error {
if vm := m.vm; vm != nil {
vm.Release()
}
m.vm = nil
if store := m.store; store != nil {
store.Release()
}
m.store = nil
return nil
}