autodidaddict opened Issue #1567:
Feature
As of 0.13.0, host functions are no longer supported via the
Callable
trait, and instead with functions that accept aCaller
as a parameter. Because of this, we _lost_ a lot of functionality that we can't easily replicate, such as the ability to create an instance of a struct that would be the target of the invocation. These instances were capable of storing per-module unique metadata, which we can no longer do.Benefit
Something like this is _crucial_ for being able to support running multiple modules within the same Rust process that do not interfere with each other's execution. Uniquely identifying them gives our host functions the ability to store per-module call state, metadata, and properly segment/isolate host resources.
Implementation
The simplest approach would be to just assign a monotonically increasing number to each module created within a given runtime, and then expose that same identity as a field/function on the
Caller
struct.Alternatives
For more flexibility, you could alternatively expose a hashmap that stores arbitrary per-module metadata. This would let the consumers of wasmtime decide what they want to do with the metadata (e.g. manage an atomic counter for module identity) rather than wasmtime itself.
alexcrichton commented on Issue #1567:
Thanks for the report! I'm not sure I fully understand how functionality was lost, though, can you elaborate a bit more on what worked previously but doesn't work any more?
autodidaddict commented on Issue #1567:
Use case is for wapc and, by extension, wascc. In both of these projects, we load multiple webassembly modules at once. These modules are able to make host calls wherein the "call state" has fields like the current error, if there is one, the current result, etc. tl;dr this "RPC" contract requires the host call to access data that only belongs to a specific module.
In 0.12.0, with the
Callable
trait, I was able to drop anRc<RefCell<State>>
into each of my callables, which allowed me to do things like grab the unique ID that I'd assigned to the module and the fields like current error, etc.In 0.13.0+, the host imports are no longer
Callable
, they're just bare functions. This means that I cannot bind any state or metadata to that function, so that function has no way of knowing _which fields_ it needs to use in order to satisfy requests from the guest.A concrete example is we provide an import called
host_response
. This function lets the guest wasm module obtain a binary blob containing the host's reply to the last function call. We do this to avoid exposing memory allocation patterns in the RPC contract. So, when the guest module callshost_response
, we need a way to determine _which response_ to give back to the wasm module, because the assumption is that the host is holding multiple modules, all of which can be executing in parallel via threads.To recap - in the current version of waPC, we can maintain per-module call state because we have structs that implement
Callable
. In 0.15.0, we don't have a way to maintain per-module call state because we're running bare functions now, and these bare functions have no "key" with which to look up per-module call state in a static.
alexcrichton commented on Issue #1567:
Hm sorry I'm still not sure I fully understand. Each
Func
is created with anFn
which can capture any state that it wants, which means you can still create per-Func
state which can be passed on to host implementations. Do you have some code I could perhaps peek at which might illustrate the problem?
autodidaddict commented on Issue #1567:
https://github.com/wapc/wapc-rust/blob/master/src/callbacks.rs#L169
here's a function that uses the old
Callable
trait. It requires access to aModuleState
in order to satisfy the request.The parameter you pass to
Func::new()
needs to beimpl Fn(Caller, &[Val], &mut [Val])
.. so this is where I can't figure out how to give that function access to module-specific state.
autodidaddict commented on Issue #1567:
I think I might have figured it out. From the documentation on Func::new(), it didn't look like it would take the kind of closure I needed. I'm going to try and create a function that captures state in the closure...
autodidaddict commented on Issue #1567:
Confirmed... I was able to convert everything to closures and it all works now. Sorry for wasting your time with an issue - I just didn't realize I could satisfy that function requirement with a closure. This has given me some ideas on how to optimize my own library :)
autodidaddict closed Issue #1567:
Feature
As of 0.13.0, host functions are no longer supported via the
Callable
trait, and instead with functions that accept aCaller
as a parameter. Because of this, we _lost_ a lot of functionality that we can't easily replicate, such as the ability to create an instance of a struct that would be the target of the invocation. These instances were capable of storing per-module unique metadata, which we can no longer do.Benefit
Something like this is _crucial_ for being able to support running multiple modules within the same Rust process that do not interfere with each other's execution. Uniquely identifying them gives our host functions the ability to store per-module call state, metadata, and properly segment/isolate host resources.
Implementation
The simplest approach would be to just assign a monotonically increasing number to each module created within a given runtime, and then expose that same identity as a field/function on the
Caller
struct.Alternatives
For more flexibility, you could alternatively expose a hashmap that stores arbitrary per-module metadata. This would let the consumers of wasmtime decide what they want to do with the metadata (e.g. manage an atomic counter for module identity) rather than wasmtime itself.
Last updated: Jan 24 2025 at 00:11 UTC