Skip to content

Commit e5126f5

Browse files
authored
Add support for user-registered host functions (#89)
* Add support for user-registered host functions * format the code
1 parent a1ae8e0 commit e5126f5

File tree

7 files changed

+54
-4
lines changed

7 files changed

+54
-4
lines changed

runtime/wasm-bpf-rs/src/runner.rs

+16-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use std::sync::mpsc;
22

33
use anyhow::{anyhow, Context};
4-
use wasmtime::{Engine, Linker, Module, Store, TypedFunc};
4+
use wasmtime::{Engine, IntoFunc, Linker, Module, Store, TypedFunc};
55
use wasmtime_wasi::WasiCtxBuilder;
66

77
use crate::add_bind_function_with_module;
@@ -39,6 +39,7 @@ pub struct WasmBpfModuleRunner {
3939
/// The linker which will be used
4040
pub linker: Linker<AppState>,
4141
operation_tx: mpsc::Sender<ProgramOperation>,
42+
main_module: Module,
4243
}
4344

4445
impl WasmBpfModuleRunner {
@@ -88,14 +89,12 @@ impl WasmBpfModuleRunner {
8889
wrapper_poll::bpf_buffer_poll_wrapper,
8990
POLL_WRAPPER_FUNCTION_NAME
9091
)?;
91-
linker
92-
.module(&mut store, MAIN_MODULE_NAME, &main_module)
93-
.with_context(|| anyhow!("Failed to link main module"))?;
9492
Ok(Self {
9593
engine,
9694
store,
9795
linker,
9896
operation_tx: tx,
97+
main_module,
9998
})
10099
}
101100
/// Consume this runner, return a handle to the wasm program, which can control the pause/resume/terminate of the program
@@ -104,6 +103,10 @@ impl WasmBpfModuleRunner {
104103
pub fn into_engine_and_entry_func(
105104
mut self,
106105
) -> anyhow::Result<(WasmProgramHandle, WasmBpfEntryFuncWrapper)> {
106+
self.linker
107+
.module(&mut self.store, MAIN_MODULE_NAME, &self.main_module)
108+
.with_context(|| anyhow!("Failed to link main module"))?;
109+
107110
let func = self
108111
.linker
109112
.get(&mut self.store, MAIN_MODULE_NAME, "_start")
@@ -119,4 +122,13 @@ impl WasmBpfModuleRunner {
119122
},
120123
))
121124
}
125+
/// Register a custom host function. It has the similar signature as `wasmtime::linker::Linker::func_wrap`
126+
pub fn register_host_function<Params, Args>(
127+
&mut self,
128+
module: &str,
129+
name: &str,
130+
func: impl IntoFunc<AppState, Params, Args>,
131+
) -> anyhow::Result<()> {
132+
self.linker.func_wrap(module, name, func).map(|_| ())
133+
}
122134
}

runtime/wasm-bpf-rs/src/tests/mod.rs

+18
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
use crate::handle::WasmProgramHandle;
22
use crate::pipe::ReadableWritePipe;
3+
use crate::state::CallerType;
34

45
use super::*;
56
use std::fs::File;
@@ -154,3 +155,20 @@ fn test_pause_and_resume_wasm_program() {
154155
assert!(tick_count_3 - tick_count_2 >= 2);
155156
handle.as_mut().unwrap().terminate().unwrap();
156157
}
158+
159+
#[test]
160+
fn test_custom_host_function() {
161+
let module_binary = std::fs::read(get_test_file_path("custom_host_func.wasm")).unwrap();
162+
let args = vec!["test".to_string()];
163+
let mut runner =
164+
WasmBpfModuleRunner::new(&module_binary[..], &args[..], Config::default()).unwrap();
165+
runner
166+
.register_host_function("host_func_test", "plus_i32", host_func_plus_i32)
167+
.unwrap();
168+
let (_, wrapper) = runner.into_engine_and_entry_func().unwrap();
169+
wrapper.run().unwrap();
170+
}
171+
172+
fn host_func_plus_i32(_caller: CallerType, a: i32, b: i32) -> i32 {
173+
a + b
174+
}
18.1 KB
Binary file not shown.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
custom_host_func.wasm
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
WASI_CLANG = /opt/wasi-sdk/bin/clang
2+
WASI_CFLAGS = -O2 --sysroot=/opt/wasi-sdk/share/wasi-sysroot -Wl,--allow-undefined,--export-table
3+
4+
custom_host_func.wasm: custom_host_func.c
5+
$(WASI_CLANG) $(WASI_CFLAGS) -o $@ $<
6+
cp custom_host_func.wasm ..
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# Test program for custom host func
2+
3+
It invokes a host function to calculate `a+b`, and verifies the result.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
#include <inttypes.h>
2+
#include <assert.h>
3+
__attribute__((import_module("host_func_test"), import_name("plus_i32")))
4+
int32_t
5+
plus_i32(int32_t a, int32_t b);
6+
int main() {
7+
int32_t c = plus_i32(0xABCD, 0x1234);
8+
assert(c == 0xABCD + 0x1234);
9+
return 0;
10+
}

0 commit comments

Comments
 (0)