Files
wazero/experimental/listener.go
2023-04-18 08:22:58 +09:00

87 lines
3.9 KiB
Go

package experimental
import (
"context"
"github.com/tetratelabs/wazero/api"
)
// StackIterator allows iterating on each function of the call stack, starting
// from the top. At least one call to Next() is required to start the iteration.
//
// Note: The iterator provides a view of the call stack at the time of
// iteration. As a result, parameter values may be different than the ones their
// function was called with.
type StackIterator interface {
// Next moves the iterator to the next function in the stack. Returns false
// if it reached the bottom of the stack.
Next() bool
// FunctionDefinition returns the function type of the current function.
FunctionDefinition() api.FunctionDefinition
// Parameters returns api.ValueType-encoded parameters of the current
// function. Do not modify the content of the slice, and copy out any value
// you need.
Parameters() []uint64
}
// FunctionListenerFactoryKey is a context.Context Value key. Its associated value should be a FunctionListenerFactory.
//
// See https://github.com/tetratelabs/wazero/issues/451
type FunctionListenerFactoryKey struct{}
// FunctionListenerFactory returns FunctionListeners to be notified when a
// function is called.
type FunctionListenerFactory interface {
// NewListener returns a FunctionListener for a defined function. If nil is
// returned, no listener will be notified.
NewListener(api.FunctionDefinition) FunctionListener
// ^^ A single instance can be returned to avoid instantiating a listener
// per function, especially as they may be thousands of functions. Shared
// listeners use their FunctionDefinition parameter to clarify.
}
// FunctionListener can be registered for any function via
// FunctionListenerFactory to be notified when the function is called.
type FunctionListener interface {
// Before is invoked before a function is called. The returned context will
// be used as the context of this function call.
//
// # Params
//
// - ctx: the context of the caller function which must be the same
// instance or parent of the result.
// - mod: the calling module.
// - def: the function definition.
// - paramValues: api.ValueType encoded parameters.
// - stackIterator: iterator on the call stack. At least one entry is
// guaranteed (the called function), whose Args() will be equal to
// paramValues. The iterator will be reused between calls to Before.
//
// Note: api.Memory is meant for inspection, not modification.
Before(ctx context.Context, mod api.Module, def api.FunctionDefinition, paramValues []uint64, stackIterator StackIterator) context.Context
// After is invoked after a function is called.
//
// # Params
//
// - ctx: the context returned by Before.
// - mod: the calling module.
// - def: the function definition.
// - err: nil if the function didn't err
// - resultValues: api.ValueType encoded results.
//
// Note: api.Memory is meant for inspection, not modification.
After(ctx context.Context, mod api.Module, def api.FunctionDefinition, err error, resultValues []uint64)
}
// TODO: We need to add tests to enginetest to ensure contexts nest. A good test can use a combination of call and call
// indirect in terms of depth and breadth. The test could show a tree 3 calls deep where the there are a couple calls at
// each depth under the root. The main thing this can help prevent is accidentally swapping the context internally.
// TODO: Errors aren't handled, and the After hook should accept one along with the result values.
// TODO: The context parameter of the After hook is not the same as the Before hook. This means interceptor patterns
// are awkward. e.g. something like timing is difficult as it requires propagating a stack. Otherwise, nested calls will
// overwrite each other's "since" time. Propagating a stack is further awkward as the After hook needs to know the
// position to read from which might be subtle.