use anyhow::Result; use std::cell::Cell; use std::rc::Rc; use std::sync::atomic::{AtomicUsize, Ordering::SeqCst}; use wasmtime::*; #[test] fn func_constructors() { let store = Store::default(); Func::wrap(&store, || {}); Func::wrap(&store, |_: i32| {}); Func::wrap(&store, |_: i32, _: i64| {}); Func::wrap(&store, |_: f32, _: f64| {}); Func::wrap(&store, || -> i32 { 0 }); Func::wrap(&store, || -> i64 { 0 }); Func::wrap(&store, || -> f32 { 0.0 }); Func::wrap(&store, || -> f64 { 0.0 }); Func::wrap(&store, || -> Option { None }); Func::wrap(&store, || -> Option { None }); Func::wrap(&store, || -> Result<(), Trap> { loop {} }); Func::wrap(&store, || -> Result { loop {} }); Func::wrap(&store, || -> Result { loop {} }); Func::wrap(&store, || -> Result { loop {} }); Func::wrap(&store, || -> Result { loop {} }); Func::wrap(&store, || -> Result, Trap> { loop {} }); Func::wrap(&store, || -> Result, Trap> { loop {} }); } #[test] fn dtor_runs() { static HITS: AtomicUsize = AtomicUsize::new(0); struct A; impl Drop for A { fn drop(&mut self) { HITS.fetch_add(1, SeqCst); } } let store = Store::default(); let a = A; assert_eq!(HITS.load(SeqCst), 0); Func::wrap(&store, move || { drop(&a); }); drop(store); assert_eq!(HITS.load(SeqCst), 1); } #[test] fn dtor_delayed() -> Result<()> { static HITS: AtomicUsize = AtomicUsize::new(0); struct A; impl Drop for A { fn drop(&mut self) { HITS.fetch_add(1, SeqCst); } } let store = Store::default(); let a = A; let func = Func::wrap(&store, move || drop(&a)); assert_eq!(HITS.load(SeqCst), 0); let wasm = wat::parse_str(r#"(import "" "" (func))"#)?; let module = Module::new(store.engine(), &wasm)?; let instance = Instance::new(&store, &module, &[func.into()])?; assert_eq!(HITS.load(SeqCst), 0); drop((instance, module, store)); assert_eq!(HITS.load(SeqCst), 1); Ok(()) } #[test] fn signatures_match() { let store = Store::default(); let f = Func::wrap(&store, || {}); assert_eq!(f.ty().params().collect::>(), &[]); assert_eq!(f.param_arity(), 0); assert_eq!(f.ty().results().collect::>(), &[]); assert_eq!(f.result_arity(), 0); let f = Func::wrap(&store, || -> i32 { loop {} }); assert_eq!(f.ty().params().collect::>(), &[]); assert_eq!(f.ty().results().collect::>(), &[ValType::I32]); let f = Func::wrap(&store, || -> i64 { loop {} }); assert_eq!(f.ty().params().collect::>(), &[]); assert_eq!(f.ty().results().collect::>(), &[ValType::I64]); let f = Func::wrap(&store, || -> f32 { loop {} }); assert_eq!(f.ty().params().collect::>(), &[]); assert_eq!(f.ty().results().collect::>(), &[ValType::F32]); let f = Func::wrap(&store, || -> f64 { loop {} }); assert_eq!(f.ty().params().collect::>(), &[]); assert_eq!(f.ty().results().collect::>(), &[ValType::F64]); let f = Func::wrap( &store, |_: f32, _: f64, _: i32, _: i64, _: i32, _: Option, _: Option| -> f64 { loop {} }, ); assert_eq!( f.ty().params().collect::>(), &[ ValType::F32, ValType::F64, ValType::I32, ValType::I64, ValType::I32, ValType::ExternRef, ValType::FuncRef, ] ); assert_eq!(f.ty().results().collect::>(), &[ValType::F64]); } #[test] fn import_works() -> Result<()> { static HITS: AtomicUsize = AtomicUsize::new(0); let wasm = wat::parse_str( r#" (import "" "" (func)) (import "" "" (func (param i32) (result i32))) (import "" "" (func (param i32) (param i64))) (import "" "" (func (param i32 i64 i32 f32 f64 externref funcref))) (func (export "run") (param externref funcref) call 0 i32.const 0 call 1 i32.const 1 i32.add i64.const 3 call 2 i32.const 100 i64.const 200 i32.const 300 f32.const 400 f64.const 500 local.get 0 local.get 1 call 3 ) "#, )?; let mut config = Config::new(); config.wasm_reference_types(true); let engine = Engine::new(&config)?; let store = Store::new(&engine); let module = Module::new(&engine, &wasm)?; let instance = Instance::new( &store, &module, &[ Func::wrap(&store, || { assert_eq!(HITS.fetch_add(1, SeqCst), 0); }) .into(), Func::wrap(&store, |x: i32| -> i32 { assert_eq!(x, 0); assert_eq!(HITS.fetch_add(1, SeqCst), 1); 1 }) .into(), Func::wrap(&store, |x: i32, y: i64| { assert_eq!(x, 2); assert_eq!(y, 3); assert_eq!(HITS.fetch_add(1, SeqCst), 2); }) .into(), Func::wrap( &store, |a: i32, b: i64, c: i32, d: f32, e: f64, f: Option, g: Option| { assert_eq!(a, 100); assert_eq!(b, 200); assert_eq!(c, 300); assert_eq!(d, 400.0); assert_eq!(e, 500.0); assert_eq!( f.as_ref().unwrap().data().downcast_ref::().unwrap(), "hello" ); assert_eq!(g.as_ref().unwrap().call(&[]).unwrap()[0].unwrap_i32(), 42); assert_eq!(HITS.fetch_add(1, SeqCst), 3); }, ) .into(), ], )?; let run = instance.get_func("run").unwrap(); run.call(&[ Val::ExternRef(Some(ExternRef::new("hello".to_string()))), Val::FuncRef(Some(Func::wrap(&store, || -> i32 { 42 }))), ])?; assert_eq!(HITS.load(SeqCst), 4); Ok(()) } #[test] fn trap_smoke() -> Result<()> { let store = Store::default(); let f = Func::wrap(&store, || -> Result<(), Trap> { Err(Trap::new("test")) }); let err = f.call(&[]).unwrap_err().downcast::()?; assert!(err.to_string().contains("test")); assert!(err.i32_exit_status().is_none()); Ok(()) } #[test] fn trap_import() -> Result<()> { let wasm = wat::parse_str( r#" (import "" "" (func)) (start 0) "#, )?; let store = Store::default(); let module = Module::new(store.engine(), &wasm)?; let trap = Instance::new( &store, &module, &[Func::wrap(&store, || -> Result<(), Trap> { Err(Trap::new("foo")) }).into()], ) .err() .unwrap() .downcast::()?; assert!(trap.to_string().contains("foo")); Ok(()) } #[test] fn get_from_wrapper() { let store = Store::default(); let f = Func::wrap(&store, || {}); assert!(f.typed::<(), ()>().is_ok()); assert!(f.typed::<(), i32>().is_err()); assert!(f.typed::<(), ()>().is_ok()); assert!(f.typed::().is_err()); assert!(f.typed::().is_err()); assert!(f.typed::<(i32, i32), ()>().is_err()); assert!(f.typed::<(i32, i32), i32>().is_err()); let f = Func::wrap(&store, || -> i32 { loop {} }); assert!(f.typed::<(), i32>().is_ok()); let f = Func::wrap(&store, || -> f32 { loop {} }); assert!(f.typed::<(), f32>().is_ok()); let f = Func::wrap(&store, || -> f64 { loop {} }); assert!(f.typed::<(), f64>().is_ok()); let f = Func::wrap(&store, || -> Option { loop {} }); assert!(f.typed::<(), Option>().is_ok()); let f = Func::wrap(&store, || -> Option { loop {} }); assert!(f.typed::<(), Option>().is_ok()); let f = Func::wrap(&store, |_: i32| {}); assert!(f.typed::().is_ok()); assert!(f.typed::().is_err()); assert!(f.typed::().is_err()); assert!(f.typed::().is_err()); let f = Func::wrap(&store, |_: i64| {}); assert!(f.typed::().is_ok()); let f = Func::wrap(&store, |_: f32| {}); assert!(f.typed::().is_ok()); let f = Func::wrap(&store, |_: f64| {}); assert!(f.typed::().is_ok()); let f = Func::wrap(&store, |_: Option| {}); assert!(f.typed::, ()>().is_ok()); let f = Func::wrap(&store, |_: Option| {}); assert!(f.typed::, ()>().is_ok()); } #[test] fn get_from_signature() { let store = Store::default(); let ty = FuncType::new(None, None); let f = Func::new(&store, ty, |_, _, _| panic!()); assert!(f.typed::<(), ()>().is_ok()); assert!(f.typed::<(), i32>().is_err()); assert!(f.typed::().is_err()); let ty = FuncType::new(Some(ValType::I32), Some(ValType::F64)); let f = Func::new(&store, ty, |_, _, _| panic!()); assert!(f.typed::<(), ()>().is_err()); assert!(f.typed::<(), i32>().is_err()); assert!(f.typed::().is_err()); assert!(f.typed::().is_ok()); } #[test] fn get_from_module() -> anyhow::Result<()> { let store = Store::default(); let module = Module::new( store.engine(), r#" (module (func (export "f0")) (func (export "f1") (param i32)) (func (export "f2") (result i32) i32.const 0) ) "#, )?; let instance = Instance::new(&store, &module, &[])?; let f0 = instance.get_func("f0").unwrap(); assert!(f0.typed::<(), ()>().is_ok()); assert!(f0.typed::<(), i32>().is_err()); let f1 = instance.get_func("f1").unwrap(); assert!(f1.typed::<(), ()>().is_err()); assert!(f1.typed::().is_ok()); assert!(f1.typed::().is_err()); let f2 = instance.get_func("f2").unwrap(); assert!(f2.typed::<(), ()>().is_err()); assert!(f2.typed::<(), i32>().is_ok()); assert!(f2.typed::().is_err()); assert!(f2.typed::().is_err()); Ok(()) } #[test] fn call_wrapped_func() -> Result<()> { let store = Store::default(); let f = Func::wrap(&store, |a: i32, b: i64, c: f32, d: f64| { assert_eq!(a, 1); assert_eq!(b, 2); assert_eq!(c, 3.0); assert_eq!(d, 4.0); }); f.call(&[Val::I32(1), Val::I64(2), 3.0f32.into(), 4.0f64.into()])?; f.typed::<(i32, i64, f32, f64), ()>()? .call((1, 2, 3.0, 4.0))?; let f = Func::wrap(&store, || 1i32); let results = f.call(&[])?; assert_eq!(results.len(), 1); assert_eq!(results[0].unwrap_i32(), 1); assert_eq!(f.typed::<(), i32>()?.call(())?, 1); let f = Func::wrap(&store, || 2i64); let results = f.call(&[])?; assert_eq!(results.len(), 1); assert_eq!(results[0].unwrap_i64(), 2); assert_eq!(f.typed::<(), i64>()?.call(())?, 2); let f = Func::wrap(&store, || 3.0f32); let results = f.call(&[])?; assert_eq!(results.len(), 1); assert_eq!(results[0].unwrap_f32(), 3.0); assert_eq!(f.typed::<(), f32>()?.call(())?, 3.0); let f = Func::wrap(&store, || 4.0f64); let results = f.call(&[])?; assert_eq!(results.len(), 1); assert_eq!(results[0].unwrap_f64(), 4.0); assert_eq!(f.typed::<(), f64>()?.call(())?, 4.0); Ok(()) } #[test] fn caller_memory() -> anyhow::Result<()> { let store = Store::default(); let f = Func::wrap(&store, |c: Caller<'_>| { assert!(c.get_export("x").is_none()); assert!(c.get_export("y").is_none()); assert!(c.get_export("z").is_none()); }); f.call(&[])?; let f = Func::wrap(&store, |c: Caller<'_>| { assert!(c.get_export("x").is_none()); }); let module = Module::new( store.engine(), r#" (module (import "" "" (func $f)) (start $f) ) "#, )?; Instance::new(&store, &module, &[f.into()])?; let f = Func::wrap(&store, |c: Caller<'_>| { assert!(c.get_export("memory").is_some()); }); let module = Module::new( store.engine(), r#" (module (import "" "" (func $f)) (memory (export "memory") 1) (start $f) ) "#, )?; Instance::new(&store, &module, &[f.into()])?; let f = Func::wrap(&store, |c: Caller<'_>| { assert!(c.get_export("m").is_some()); assert!(c.get_export("f").is_some()); assert!(c.get_export("g").is_none()); assert!(c.get_export("t").is_none()); }); let module = Module::new( store.engine(), r#" (module (import "" "" (func $f)) (memory (export "m") 1) (func (export "f")) (global (export "g") i32 (i32.const 0)) (table (export "t") 1 funcref) (start $f) ) "#, )?; Instance::new(&store, &module, &[f.into()])?; Ok(()) } #[test] fn func_write_nothing() -> anyhow::Result<()> { let store = Store::default(); let ty = FuncType::new(None, Some(ValType::I32)); let f = Func::new(&store, ty, |_, _, _| Ok(())); let err = f.call(&[]).unwrap_err().downcast::()?; assert!(err .to_string() .contains("function attempted to return an incompatible value")); Ok(()) } #[test] // Note: Cranelift only supports refrerence types (used in the wasm in this // test) on x64. #[cfg(target_arch = "x86_64")] fn return_cross_store_value() -> anyhow::Result<()> { let wasm = wat::parse_str( r#" (import "" "" (func (result funcref))) (func (export "run") (result funcref) call 0 ) "#, )?; let mut config = Config::new(); config.wasm_reference_types(true); let engine = Engine::new(&config)?; let module = Module::new(&engine, &wasm)?; let store1 = Store::new(&engine); let store2 = Store::new(&engine); let store2_func = Func::wrap(&store2, || {}); let return_cross_store_func = Func::wrap(&store1, move || Some(store2_func.clone())); let instance = Instance::new(&store1, &module, &[return_cross_store_func.into()])?; let run = instance.get_func("run").unwrap(); let result = run.call(&[]); assert!(result.is_err()); assert!(result.unwrap_err().to_string().contains("cross-`Store`")); Ok(()) } #[test] // Note: Cranelift only supports refrerence types (used in the wasm in this // test) on x64. #[cfg(target_arch = "x86_64")] fn pass_cross_store_arg() -> anyhow::Result<()> { let mut config = Config::new(); config.wasm_reference_types(true); let engine = Engine::new(&config)?; let store1 = Store::new(&engine); let store2 = Store::new(&engine); let store1_func = Func::wrap(&store1, |_: Option| {}); let store2_func = Func::wrap(&store2, || {}); // Using regular `.call` fails with cross-Store arguments. assert!(store1_func .call(&[Val::FuncRef(Some(store2_func.clone()))]) .is_err()); // And using `.get` followed by a function call also fails with cross-Store // arguments. let f = store1_func.typed::, ()>()?; let result = f.call(Some(store2_func)); assert!(result.is_err()); assert!(result.unwrap_err().to_string().contains("cross-`Store`")); Ok(()) } #[test] fn externref_signature_no_reference_types() -> anyhow::Result<()> { let store = Store::default(); Func::wrap(&store, |_: Option| {}); Func::new( &store, FuncType::new( [ValType::FuncRef, ValType::ExternRef].iter().cloned(), [ValType::FuncRef, ValType::ExternRef].iter().cloned(), ), |_, _, _| Ok(()), ); Ok(()) } #[test] fn trampolines_always_valid() -> anyhow::Result<()> { let func = { // Compile two modules up front let store = Store::default(); let module1 = Module::new(store.engine(), "(module (import \"\" \"\" (func)))")?; let module2 = Module::new(store.engine(), "(module (func (export \"\")))")?; // Start instantiating the first module, but this will fail. // Historically this registered the module's trampolines with `Store` // before the failure, but then after the failure the `Store` didn't // hold onto the trampoline. drop(Instance::new(&store, &module1, &[])); drop(module1); // Then instantiate another module which has the same function type (no // parameters or results) which tries to use the trampoline defined in // the previous module. Then we extract the function and, in another // scope where everything is dropped, we call the func. let i = Instance::new(&store, &module2, &[])?; i.get_func("").unwrap() }; // ... and no segfaults! right? right? ... func.call(&[])?; Ok(()) } #[test] #[cfg(not(feature = "old-x86-backend"))] fn typed_multiple_results() -> anyhow::Result<()> { let store = Store::default(); let module = Module::new( store.engine(), r#" (module (func (export "f0") (result i32 i64) i32.const 0 i64.const 1) (func (export "f1") (param i32 i32 i32) (result f32 f64) f32.const 2 f64.const 3) ) "#, )?; let instance = Instance::new(&store, &module, &[])?; let f0 = instance.get_func("f0").unwrap(); assert!(f0.typed::<(), ()>().is_err()); assert!(f0.typed::<(), (i32, f32)>().is_err()); assert!(f0.typed::<(), i32>().is_err()); assert_eq!(f0.typed::<(), (i32, i64)>()?.call(())?, (0, 1)); let f1 = instance.get_func("f1").unwrap(); assert_eq!( f1.typed::<(i32, i32, i32), (f32, f64)>()?.call((1, 2, 3))?, (2., 3.) ); Ok(()) } #[test] fn trap_doesnt_leak() -> anyhow::Result<()> { struct Canary(Rc>); impl Drop for Canary { fn drop(&mut self) { self.0.set(true); } } let store = Store::default(); // test that `Func::wrap` is correct let canary1 = Canary(Rc::new(Cell::new(false))); let dtor1_run = canary1.0.clone(); let f1 = Func::wrap(&store, move || -> Result<(), Trap> { drop(&canary1); Err(Trap::new("")) }); assert!(f1.typed::<(), ()>()?.call(()).is_err()); assert!(f1.call(&[]).is_err()); // test that `Func::new` is correct let canary2 = Canary(Rc::new(Cell::new(false))); let dtor2_run = canary2.0.clone(); let f2 = Func::new(&store, FuncType::new(None, None), move |_, _, _| { drop(&canary2); Err(Trap::new("")) }); assert!(f2.typed::<(), ()>()?.call(()).is_err()); assert!(f2.call(&[]).is_err()); // drop everything and ensure dtors are run drop((store, f1, f2)); assert!(dtor1_run.get()); assert!(dtor2_run.get()); Ok(()) } #[test] #[cfg(not(feature = "old-x86-backend"))] fn wrap_multiple_results() -> anyhow::Result<()> { fn test(store: &Store, t: T) -> anyhow::Result<()> where T: WasmRet + WasmResults + PartialEq + Copy + std::fmt::Debug + EqualToValues + 'static, { let f = Func::wrap(store, move || t); assert_eq!(f.typed::<(), T>()?.call(())?, t); assert!(t.eq_values(&f.call(&[])?)); let module = Module::new(store.engine(), &T::gen_wasm())?; let instance = Instance::new(store, &module, &[f.into()])?; let f = instance.get_func("foo").unwrap(); assert_eq!(f.typed::<(), T>()?.call(())?, t); assert!(t.eq_values(&f.call(&[])?)); Ok(()) } let store = Store::default(); // 0 element test(&store, ())?; // 1 element test(&store, (1i32,))?; test(&store, (2u32,))?; test(&store, (3i64,))?; test(&store, (4u64,))?; test(&store, (5.0f32,))?; test(&store, (6.0f64,))?; // 2 element ... test(&store, (7i32, 8i32))?; test(&store, (7i32, 8i64))?; test(&store, (7i32, 8f32))?; test(&store, (7i32, 8f64))?; test(&store, (7i64, 8i32))?; test(&store, (7i64, 8i64))?; test(&store, (7i64, 8f32))?; test(&store, (7i64, 8f64))?; test(&store, (7f32, 8i32))?; test(&store, (7f32, 8i64))?; test(&store, (7f32, 8f32))?; test(&store, (7f32, 8f64))?; test(&store, (7f64, 8i32))?; test(&store, (7f64, 8i64))?; test(&store, (7f64, 8f32))?; test(&store, (7f64, 8f64))?; // and beyond... test(&store, (1i32, 2i32, 3i32))?; test(&store, (1i32, 2f32, 3i32))?; test(&store, (1f64, 2f32, 3i32))?; test(&store, (1f64, 2i64, 3i32))?; test(&store, (1f32, 2f32, 3i64, 4f64))?; test(&store, (1f64, 2i64, 3i32, 4i64, 5f32))?; test(&store, (1i32, 2f64, 3i64, 4f64, 5f64, 6f32))?; test(&store, (1i64, 2i32, 3i64, 4f32, 5f32, 6i32, 7u64))?; test(&store, (1u32, 2f32, 3u64, 4f64, 5i32, 6f32, 7u64, 8u32))?; test( &store, (1f32, 2f64, 3f32, 4i32, 5u32, 6i64, 7f32, 8i32, 9u64), )?; return Ok(()); trait EqualToValues { fn eq_values(&self, values: &[Val]) -> bool; fn gen_wasm() -> String; } macro_rules! equal_tuples { ($($cnt:tt ($($a:ident),*))*) => ($( #[allow(non_snake_case)] impl<$($a: EqualToValue,)*> EqualToValues for ($($a,)*) { fn eq_values(&self, values: &[Val]) -> bool { let ($($a,)*) = self; let mut _values = values.iter(); _values.len() == $cnt && $($a.eq_value(_values.next().unwrap()) &&)* true } fn gen_wasm() -> String { let mut wasm = String::new(); wasm.push_str("(module "); wasm.push_str("(type $t (func (result "); $( wasm.push_str($a::wasm_ty()); wasm.push_str(" "); )* wasm.push_str(")))"); wasm.push_str("(import \"\" \"\" (func $host (type $t)))"); wasm.push_str("(func (export \"foo\") (type $t)"); wasm.push_str("call $host"); wasm.push_str(")"); wasm.push_str(")"); wasm } } )*) } equal_tuples! { 0 () 1 (A1) 2 (A1, A2) 3 (A1, A2, A3) 4 (A1, A2, A3, A4) 5 (A1, A2, A3, A4, A5) 6 (A1, A2, A3, A4, A5, A6) 7 (A1, A2, A3, A4, A5, A6, A7) 8 (A1, A2, A3, A4, A5, A6, A7, A8) 9 (A1, A2, A3, A4, A5, A6, A7, A8, A9) } trait EqualToValue { fn eq_value(&self, value: &Val) -> bool; fn wasm_ty() -> &'static str; } macro_rules! equal_values { ($a:ident $($ty:ident $wasm:tt $variant:ident $e:expr,)*) => ($( impl EqualToValue for $ty { fn eq_value(&self, val: &Val) -> bool { if let Val::$variant($a) = *val { return *self == $e; } false } fn wasm_ty() -> &'static str { $wasm } } )*) } equal_values! { a i32 "i32" I32 a, u32 "i32" I32 a as u32, i64 "i64" I64 a, u64 "i64" I64 a as u64, f32 "f32" F32 f32::from_bits(a), f64 "f64" F64 f64::from_bits(a), } }