↩️ Exposing host (imported) functions
A Wasm module can import entities, like functions, memories, globals and tables. This example illustrates how to expose functions from the host.
Up until now, our WebAssembly program has only been able to do pure computation, that is, take arguments and return values. Most interesting use cases require more than just computation though. In this section we'll go over how to give the Wasm modules we run extra abilities in the form of host functions.
In this example, we'll create a system for getting and adjusting a counter value. However, host functions are not limited to storing data outside of Wasm, they're normal host functions and can do anything that the host can do.
There will be a
get_counter
function that will return ani32
ofthe current global counter.
There will be an
add_to_counter
function will add the passedi32
value to the counter, and return ani32
of the currentglobal counter.
First we are going to want to initialize a new project. To do this we can navigate to our project folder, or create one. In this example, we will create a new project. Lets create it and navigate to it:
The final code for this example can be found on GitHub.
Please take a look at the setup steps for Rust.
We have to modify Cargo.toml
to add the Wasmer dependencies as shown below:
Now that we have everything set up, let's go ahead and try it out!
Declaring the data
Because we want to store data outside of the Wasm module and have host functions use this data, we need to do some preparation. We'll need to declare the data we want to use and the container to hold it.
Here we use a combination of Arc
and RefCell
to guarantee thread safety while allowing mutability.
Declaring functions and imports
Now that our data is available we'll declare the functions.
As you can see here, both functions take an extra parameter in the form of a mutable reference to an Env
which is the container we created to hold our data.
The last thing we need to do now is to imports the function in the Wasm module.
We use Function::new_native_with_env
here to tell Wasmer our host functions need our Env
to be passed in addition to other arguments.
If the host function does not need external data (it is pure) we can use Function::new_native
instead of Function::new_native_with_env
.
Now each time the add_to_counter
will be run from the Wasm module it will alter the data on the host side.
Running
We now have everything we need to run the Wasm module, let's do it!
You should be able to run it using the cargo run
command. The output should look like this:
If you want to run the examples from the Wasmer repository codebase directly, you can also do:
Last updated