#![allow(clippy::let_unit_value)]
use std::error::Error;
use wasmer::{FromToNativeWasmType, Function, FunctionEnvMut, WasmTypeList};
use super::{Environment, InstanceBuilder};
use crate::{primitive_types::MaybeFlatType, ExportFunction, RuntimeError};
macro_rules! export_function {
($( $names:ident: $types:ident ),*) => {
impl<Handler, HandlerError, $( $types, )* FlatResult, UserData>
ExportFunction<Handler, ($( $types, )*), FlatResult> for InstanceBuilder<UserData>
where
$( $types: FromToNativeWasmType, )*
FlatResult: MaybeFlatType + WasmTypeList,
UserData: 'static,
HandlerError: Error + Send + Sync + 'static,
Handler:
Fn(
FunctionEnvMut<'_, Environment<UserData>>,
($( $types, )*),
) -> Result<FlatResult, HandlerError>
+ Send
+ Sync
+ 'static,
{
fn export(
&mut self,
module_name: &str,
function_name: &str,
handler: Handler,
) -> Result<(), RuntimeError> {
let environment = self.environment();
let function = Function::new_typed_with_env(
self,
&environment,
move |
environment: FunctionEnvMut<'_, Environment<UserData>>,
$( $names: $types ),*
| -> Result<FlatResult, wasmer::RuntimeError> {
handler(environment, ($( $names, )*))
.map_err(|error| -> Box<dyn std::error::Error + Send + Sync> {
Box::new(error)
})
.map_err(wasmer::RuntimeError::user)
},
);
self.define(
module_name,
function_name,
function,
);
Ok(())
}
}
};
}
repeat_macro!(export_function =>
a: A,
b: B,
c: C,
d: D,
e: E,
f: F,
g: G,
h: H,
i: I,
j: J,
k: K,
l: L,
m: M,
n: N,
o: O,
p: P,
q: Q,
);