From 352908e960332ebd2148ab2b7da82a08f2a595f8 Mon Sep 17 00:00:00 2001 From: Alex Crichton Date: Fri, 4 Mar 2022 12:29:44 -0600 Subject: [PATCH] Fix calling call hooks with `unchecked` func variants (#3881) This commit fixes calling the call hooks configured in a store for host functions defined with `Func::new_unchecked` or similar. I believe that this was just an accidental oversight and there's no fundamental reason to not support this. --- crates/wasmtime/src/func.rs | 10 ++-- tests/all/call_hook.rs | 111 ++++++++++++++++++++++++++---------- 2 files changed, 87 insertions(+), 34 deletions(-) diff --git a/crates/wasmtime/src/func.rs b/crates/wasmtime/src/func.rs index ac8272088a3d..12a85107a213 100644 --- a/crates/wasmtime/src/func.rs +++ b/crates/wasmtime/src/func.rs @@ -998,8 +998,6 @@ impl Func { values_vec: *mut ValRaw, func: &dyn Fn(Caller<'_, T>, &[Val], &mut [Val]) -> Result<(), Trap>, ) -> Result<(), Trap> { - caller.store.0.call_hook(CallHook::CallingHost)?; - // Translate the raw JIT arguments in `values_vec` into a `Val` which // we'll be passing as a slice. The storage for our slice-of-`Val` we'll // be taking from the `Store`. We preserve our slice back into the @@ -1056,7 +1054,6 @@ impl Func { // hostcall to reuse our own storage. val_vec.truncate(0); caller.store.0.save_hostcall_val_storage(val_vec); - caller.store.0.call_hook(CallHook::ReturningFromHost)?; Ok(()) } @@ -2050,7 +2047,12 @@ impl HostFunc { func: impl Fn(Caller<'_, T>, *mut ValRaw) -> Result<(), Trap> + Send + Sync + 'static, ) -> Self { let func = move |caller_vmctx, values: *mut ValRaw| { - Caller::::with(caller_vmctx, |caller| func(caller, values)) + Caller::::with(caller_vmctx, |mut caller| { + caller.store.0.call_hook(CallHook::CallingHost)?; + let result = func(caller.sub_caller(), values)?; + caller.store.0.call_hook(CallHook::ReturningFromHost)?; + Ok(result) + }) }; let (instance, trampoline) = crate::trampoline::create_function(&ty, func, engine) .expect("failed to create function"); diff --git a/tests/all/call_hook.rs b/tests/all/call_hook.rs index e253519f8046..d1bc78e8a133 100644 --- a/tests/all/call_hook.rs +++ b/tests/all/call_hook.rs @@ -6,47 +6,98 @@ use wasmtime::*; fn call_wrapped_func() -> Result<(), Error> { let mut store = Store::::default(); store.call_hook(State::call_hook); - let f = Func::wrap( + + fn verify(state: &State) { + // Calling this func will switch context into wasm, then back to host: + assert_eq!(state.context, vec![Context::Wasm, Context::Host]); + + assert_eq!(state.calls_into_host, state.returns_from_host + 1); + assert_eq!(state.calls_into_wasm, state.returns_from_wasm + 1); + } + + let mut funcs = Vec::new(); + funcs.push(Func::wrap( &mut store, |caller: Caller, a: i32, b: i64, c: f32, d: f64| { - // Calling this func will switch context into wasm, then back to host: - assert_eq!(caller.data().context, vec![Context::Wasm, Context::Host]); - - assert_eq!( - caller.data().calls_into_host, - caller.data().returns_from_host + 1 - ); - assert_eq!( - caller.data().calls_into_wasm, - caller.data().returns_from_wasm + 1 - ); + verify(caller.data()); assert_eq!(a, 1); assert_eq!(b, 2); assert_eq!(c, 3.0); assert_eq!(d, 4.0); }, - ); - - f.call( + )); + funcs.push(Func::new( &mut store, - &[Val::I32(1), Val::I64(2), 3.0f32.into(), 4.0f64.into()], - &mut [], - )?; - - // One switch from vm to host to call f, another in return from f. - assert_eq!(store.data().calls_into_host, 1); - assert_eq!(store.data().returns_from_host, 1); - assert_eq!(store.data().calls_into_wasm, 1); - assert_eq!(store.data().returns_from_wasm, 1); + FuncType::new([ValType::I32, ValType::I64, ValType::F32, ValType::F64], []), + |caller: Caller, params, results| { + verify(caller.data()); + + assert_eq!(params.len(), 4); + assert_eq!(params[0].i32().unwrap(), 1); + assert_eq!(params[1].i64().unwrap(), 2); + assert_eq!(params[2].f32().unwrap(), 3.0); + assert_eq!(params[3].f64().unwrap(), 4.0); + assert_eq!(results.len(), 0); + Ok(()) + }, + )); + funcs.push(unsafe { + Func::new_unchecked( + &mut store, + FuncType::new([ValType::I32, ValType::I64, ValType::F32, ValType::F64], []), + |caller: Caller, space| { + verify(caller.data()); + + assert_eq!((*space.add(0)).i32, 1); + assert_eq!((*space.add(1)).i64, 2); + assert_eq!((*space.add(2)).f32, 3.0f32.to_bits()); + assert_eq!((*space.add(3)).f64, 4.0f64.to_bits()); + Ok(()) + }, + ) + }); - f.typed::<(i32, i64, f32, f64), (), _>(&store)? - .call(&mut store, (1, 2, 3.0, 4.0))?; + let mut n = 0; + for f in funcs.iter() { + f.call( + &mut store, + &[Val::I32(1), Val::I64(2), 3.0f32.into(), 4.0f64.into()], + &mut [], + )?; + n += 1; + + // One switch from vm to host to call f, another in return from f. + assert_eq!(store.data().calls_into_host, n); + assert_eq!(store.data().returns_from_host, n); + assert_eq!(store.data().calls_into_wasm, n); + assert_eq!(store.data().returns_from_wasm, n); + + f.typed::<(i32, i64, f32, f64), (), _>(&store)? + .call(&mut store, (1, 2, 3.0, 4.0))?; + n += 1; + + assert_eq!(store.data().calls_into_host, n); + assert_eq!(store.data().returns_from_host, n); + assert_eq!(store.data().calls_into_wasm, n); + assert_eq!(store.data().returns_from_wasm, n); + + unsafe { + let mut args = [ + Val::I32(1).to_raw(&mut store), + Val::I64(2).to_raw(&mut store), + Val::F32(3.0f32.to_bits()).to_raw(&mut store), + Val::F64(4.0f64.to_bits()).to_raw(&mut store), + ]; + f.call_unchecked(&mut store, args.as_mut_ptr())?; + } + n += 1; - assert_eq!(store.data().calls_into_host, 2); - assert_eq!(store.data().returns_from_host, 2); - assert_eq!(store.data().calls_into_wasm, 2); - assert_eq!(store.data().returns_from_wasm, 2); + assert_eq!(store.data().calls_into_host, n); + assert_eq!(store.data().returns_from_host, n); + assert_eq!(store.data().calls_into_wasm, n); + assert_eq!(store.data().returns_from_wasm, n); + } Ok(()) }