↩️ 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_counterfunction that will return ani32ofthe current global counter.
There will be an
add_to_counterfunction will add the passedi32value to the counter, and return ani32of 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:
cargo new imported-function-env
cd imported-function-envWe have to modify Cargo.toml to add the Wasmer dependencies as shown below:
[dependencies]
# The Wasmer API
wasmer = "1.0"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.
let shared_counter: Arc<RefCell<i32>> = Arc::new(RefCell::new(0));
#[derive(WasmerEnv, Clone)]
struct Env {
counter: Arc<RefCell<i32>>,
}Declaring functions and imports
Now that our data is available we'll declare the functions.
fn get_counter(env: &Env) -> i32 {
*env.counter.borrow()
}
fn add_to_counter(env: &Env, add: i32) -> i32 {
let mut counter_ref = env.counter.borrow_mut();
*counter_ref += add;
*counter_ref
}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.
let get_counter_func = Function::new_native_with_env(
&store,
Env { counter: shared_counter.clone() },
get_counter
);
let add_to_counter_func = Function::new_native_with_env(
&store,
Env { counter: shared_counter.clone() },
add_to_counter
);
let import_object = imports! {
"env" => {
"get_counter" => get_counter_func,
"add_to_counter" => add_to_counter_func,
}
};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:
Compiling module...
Instantiating module...
Initial ounter value: 0
Calling `increment_counter_loop` function...
New counter value (host): 5
New counter value (guest): 5Last updated
Was this helpful?