wasmer_runtime_core_fl/
macros.rs

1/// Helper macro to create a new `Func` object using the provided function pointer.
2///
3/// # Usage
4///
5/// Function pointers or closures are supported. Closures can capture
6/// their environment (with `move`). The first parameter is likely to
7/// be of kind `vm::Ctx`, though it can be optional.
8///
9/// ```
10/// # use wasmer_runtime_core::{imports, func};
11/// # use wasmer_runtime_core::vm;
12///
13/// // A function that has access to `vm::Ctx`.
14/// fn func_with_vmctx(_: &mut vm::Ctx, n: i32) -> i32 {
15///     n
16/// }
17///
18/// // A function that cannot access `vm::Ctx`.
19/// fn func(n: i32) -> i32 {
20///     n
21/// }
22///
23/// let i = 7;
24///
25/// let import_object = imports! {
26///     "env" => {
27///         "foo" => func!(func_with_vmctx),
28///         "bar" => func!(func),
29///         // A closure with a captured environment, and an access to `vm::Ctx`.
30///         "baz" => func!(move |_: &mut vm::Ctx, n: i32| -> i32 {
31///             n + i
32///         }),
33///         // A closure without a captured environment, and no access to `vm::Ctx`.
34///         "qux" => func!(|n: i32| -> i32 {
35///             n
36///         }),
37///     },
38/// };
39/// ```
40#[macro_export]
41macro_rules! func {
42    ($func:expr) => {{
43        $crate::Func::new($func)
44    }};
45}
46
47/// Generate an [`ImportObject`] safely.
48///
49/// [`ImportObject`]: struct.ImportObject.html
50///
51/// # Note
52/// The `import` macro currently only supports
53/// importing functions.
54///
55///
56/// # Usage
57/// ```
58/// # use wasmer_runtime_core::{imports, func};
59/// # use wasmer_runtime_core::vm::Ctx;
60/// let import_object = imports! {
61///     "env" => {
62///         "foo" => func!(foo),
63///     },
64/// };
65///
66/// let imports_with_state = imports! {
67///     || (0 as _, |_a| {}),
68///     "env" => {
69///         "foo" => func!(foo),
70///     },
71/// };
72///
73/// fn foo(_: &mut Ctx, n: i32) -> i32 {
74///     n
75/// }
76/// ```
77#[macro_export]
78macro_rules! imports {
79    ( $( $ns_name:expr => $ns:tt ),* $(,)? ) => {{
80        use $crate::{
81            import::{ImportObject, Namespace},
82        };
83
84        let mut import_object = ImportObject::new();
85
86        $({
87            let ns = $crate::__imports_internal!($ns);
88
89            import_object.register($ns_name, ns);
90        })*
91
92        import_object
93    }};
94    ($state_gen:expr, $( $ns_name:expr => $ns:tt ),* $(,)? ) => {{
95        use $crate::{
96            import::{ImportObject, Namespace},
97        };
98
99        let mut import_object = ImportObject::new_with_data($state_gen);
100
101        $({
102            let ns = $crate::__imports_internal!($ns);
103
104            import_object.register($ns_name, ns);
105        })*
106
107        import_object
108    }};
109}
110
111#[macro_export]
112#[doc(hidden)]
113macro_rules! __imports_internal {
114    ( { $( $imp_name:expr => $import_item:expr ),* $(,)? } ) => {{
115        let mut ns = Namespace::new();
116        $(
117            ns.insert($imp_name, $import_item);
118        )*
119        ns
120    }};
121    ($ns:ident) => {
122        $ns
123    };
124}
125
126#[macro_export]
127#[doc(hidden)]
128macro_rules! namespace {
129    ( $( $imp_name:expr => $import_item:expr ),* $(,)? ) => {{
130        let mut ns = $crate::import::Namespace::new();
131        $(
132            ns.insert($imp_name, $import_item);
133        )*
134        ns
135    }};
136}
137
138#[cfg(test)]
139mod test {
140    fn func(arg: i32) -> i32 {
141        arg + 1
142    }
143
144    #[test]
145    fn imports_macro_allows_trailing_comma_and_none() {
146        let _ = imports! {
147            "env" => {
148                "func" => func!(func),
149            },
150        };
151        let _ = imports! {
152            "env" => {
153                "func" => func!(func),
154            }
155        };
156        let _ = imports! {
157            "env" => {
158                "func" => func!(func),
159            },
160            "abc" => {
161                "def" => func!(func),
162            }
163        };
164        let _ = imports! {
165            "env" => {
166                "func" => func!(func)
167            },
168        };
169        let _ = imports! {
170            "env" => {
171                "func" => func!(func)
172            }
173        };
174        let _ = imports! {
175            "env" => {
176                "func1" => func!(func),
177                "func2" => func!(func)
178            }
179        };
180        let _ = imports! {
181            "env" => {
182                "func1" => func!(func),
183                "func2" => func!(func),
184            }
185        };
186    }
187
188    #[test]
189    fn imports_macro_allows_trailing_comma_and_none_with_state() {
190        use std::{ffi, ptr};
191
192        fn dtor(_arg: *mut ffi::c_void) {}
193        fn state_creator() -> (*mut ffi::c_void, fn(*mut ffi::c_void)) {
194            (ptr::null_mut() as *mut ffi::c_void, dtor)
195        }
196        let _ = imports! {
197            state_creator,
198            "env" => {
199                "func1" => func!(func),
200                "func2" => func!(func),
201            }
202        };
203        let _ = imports! {
204            state_creator,
205            "env" => {
206                "func1" => func!(func),
207                "func2" => func!(func)
208            },
209        };
210        let _ = imports! {
211            state_creator,
212            "env" => {
213                "func1" => func!(func),
214                "func2" => func!(func),
215            },
216        };
217    }
218}