diff --git a/crates/plugin_runtime/src/wasi.rs b/crates/plugin_runtime/src/wasi.rs index 63a07788f4..feb7281265 100644 --- a/crates/plugin_runtime/src/wasi.rs +++ b/crates/plugin_runtime/src/wasi.rs @@ -91,31 +91,51 @@ impl WasiPluginBuilder { Self::new(wasi_ctx) } - pub fn host_function( + pub fn host_function( mut self, name: &str, function: impl Fn(A) -> R + Send + Sync + 'static, ) -> Result { - self.linker.func_wrap( + self.linker.func_wrap2_async( "env", name, move |mut caller: Caller<'_, WasiCtxAlloc>, ptr: u32, len: u32| { - let mut plugin_memory = match caller.get_export("memory") { - Some(Extern::Memory(mem)) => mem, - _ => return Err(Trap::new("Could not grab slice of plugin memory")), - }; - let args = Wasi::deserialize_from_buffer(&mut plugin_memory, &caller, ptr, len)?; + // TODO: use try block once avaliable + let result: Result<(Memory, Vec), Trap> = (|| { + // grab a handle to the memory + let mut plugin_memory = match caller.get_export("memory") { + Some(Extern::Memory(mem)) => mem, + _ => return Err(Trap::new("Could not grab slice of plugin memory"))?, + }; - let result = function(args); - let buffer = Wasi::serialize_to_buffer( - caller.data().alloc_buffer(), - &mut plugin_memory, - &mut caller, - result, - ) - .await; + // get the args passed from Guest + let args = + Wasi::deserialize_from_buffer(&mut plugin_memory, &caller, ptr, len)?; - Ok(7u32) + // Call the Host-side function + let result: R = function(args); + + // Serialize the result back to guest + let result = Wasi::serialize(result).map_err(|_| { + Trap::new("Could not serialize value returned from function") + })?; + Ok((plugin_memory, result)) + })(); + + Box::new(async move { + let (mut plugin_memory, result) = result?; + + // todo!(); + let (ptr, len) = Wasi::serialize_to_buffer( + caller.data().alloc_buffer(), + &mut plugin_memory, + &mut caller, + result, + ) + .await?; + + Ok(7u32) + }) }, )?; Ok(self) @@ -318,19 +338,22 @@ impl Wasi { // This isn't a problem because Wasm stops executing after the function returns, // so the heap is still valid for our inspection when we want to pull things out. + fn serialize(item: A) -> Result, Error> { + // serialize the argument using bincode + let item = bincode::serialize(&item)?; + Ok(item) + } + /// Takes an item, allocates a buffer, serializes the argument to that buffer, /// and returns a (ptr, len) pair to that buffer. - async fn serialize_to_buffer( + async fn serialize_to_buffer( alloc_buffer: TypedFunc, plugin_memory: &mut Memory, mut store: impl AsContextMut, - item: A, + item: Vec, ) -> Result<(u32, u32), Error> { - // serialize the argument using bincode - let item = bincode::serialize(&item)?; - let buffer_len = item.len() as u32; - // allocate a buffer and write the argument to that buffer + let buffer_len = item.len() as u32; let buffer_ptr = alloc_buffer.call_async(&mut store, buffer_len).await?; plugin_memory.write(&mut store, buffer_ptr as usize, &item)?; Ok((buffer_ptr, buffer_len)) @@ -413,7 +436,7 @@ impl Wasi { self.store.data().alloc_buffer(), &mut plugin_memory, &mut self.store, - arg, + Self::serialize(arg)?, ) .await?;