Remove dependency on self in call-related functions

This commit is contained in:
Isaac Clayton 2022-06-09 10:40:46 +02:00
parent 7266dff537
commit 47520f0ca1

View file

@ -6,8 +6,8 @@ use anyhow::{anyhow, Error};
use serde::{de::DeserializeOwned, Serialize}; use serde::{de::DeserializeOwned, Serialize};
use wasi_common::{dir, file}; use wasi_common::{dir, file};
use wasmtime::IntoFunc;
use wasmtime::{Caller, Config, Engine, Instance, Linker, Module, Store, TypedFunc}; use wasmtime::{Caller, Config, Engine, Instance, Linker, Module, Store, TypedFunc};
use wasmtime::{IntoFunc, Memory};
use wasmtime_wasi::{Dir, WasiCtx, WasiCtxBuilder}; use wasmtime_wasi::{Dir, WasiCtx, WasiCtxBuilder};
pub struct WasiResource(u32); pub struct WasiResource(u32);
@ -106,12 +106,15 @@ impl WasiPluginBuilder {
name: &str, name: &str,
function: impl Fn(A) -> R + Send + Sync + 'static, function: impl Fn(A) -> R + Send + Sync + 'static,
) -> Result<Self, Error> { ) -> Result<Self, Error> {
self.linker self.linker.func_wrap(
.func_wrap("env", name, move |ptr: u32, len: u32| { "env",
name,
move |ctx: Caller<'_, WasiCtx>, ptr: u32, len: u32| {
// TODO: insert serialization code // TODO: insert serialization code
function(todo!()); function(todo!());
7u32 7u32
})?; },
)?;
Ok(self) Ok(self)
} }
@ -266,45 +269,56 @@ impl Wasi {
/// Takes an item, allocates a buffer, serializes the argument to that buffer, /// Takes an item, allocates a buffer, serializes the argument to that buffer,
/// and returns a (ptr, len) pair to that buffer. /// and returns a (ptr, len) pair to that buffer.
async fn serialize_to_buffer<T: Serialize>(&mut self, item: T) -> Result<(u32, u32), Error> { async fn serialize_to_buffer<T: Serialize>(
alloc_buffer: TypedFunc<u32, u32>,
plugin_memory: &mut Memory,
mut store: &mut Store<WasiCtx>,
item: T,
) -> Result<(u32, u32), Error> {
// serialize the argument using bincode // serialize the argument using bincode
let item = bincode::serialize(&item)?; let item = bincode::serialize(&item)?;
let buffer_len = item.len() as u32; let buffer_len = item.len() as u32;
// allocate a buffer and write the argument to that buffer // allocate a buffer and write the argument to that buffer
let buffer_ptr = self let buffer_ptr = alloc_buffer.call_async(&mut store, buffer_len).await?;
.alloc_buffer plugin_memory.write(&mut store, buffer_ptr as usize, &item)?;
.call_async(&mut self.store, buffer_len)
.await?;
let plugin_memory = self
.instance
.get_memory(&mut self.store, "memory")
.ok_or_else(|| anyhow!("Could not grab slice of plugin memory"))?;
plugin_memory.write(&mut self.store, buffer_ptr as usize, &item)?;
Ok((buffer_ptr, buffer_len)) Ok((buffer_ptr, buffer_len))
} }
/// Takes a ptr to a (ptr, len) pair and returns the corresponding deserialized buffer /// Takes `ptr to a `(ptr, len)` pair, and returns `(ptr, len)`.
fn deserialize_from_buffer<R: DeserializeOwned>(&mut self, buffer: u32) -> Result<R, Error> { fn deref_buffer(
plugin_memory: &mut Memory,
store: &mut Store<WasiCtx>,
buffer: u32,
) -> Result<(u32, u32), Error> {
// create a buffer to read the (ptr, length) pair into // create a buffer to read the (ptr, length) pair into
// this is a total of 4 + 4 = 8 bytes. // this is a total of 4 + 4 = 8 bytes.
let raw_buffer = &mut [0; 8]; let raw_buffer = &mut [0; 8];
let plugin_memory = self plugin_memory.read(store, buffer as usize, raw_buffer)?;
.instance
.get_memory(&mut self.store, "memory")
.ok_or_else(|| anyhow!("Could not grab slice of plugin memory"))?;
plugin_memory.read(&mut self.store, buffer as usize, raw_buffer)?;
// use these bytes (wasm stores things little-endian) // use these bytes (wasm stores things little-endian)
// to get a pointer to the buffer and its length // to get a pointer to the buffer and its length
let b = raw_buffer; let b = raw_buffer;
let buffer_ptr = u32::from_le_bytes([b[0], b[1], b[2], b[3]]) as usize; let buffer_ptr = u32::from_le_bytes([b[0], b[1], b[2], b[3]]);
let buffer_len = u32::from_le_bytes([b[4], b[5], b[6], b[7]]) as usize; let buffer_len = u32::from_le_bytes([b[4], b[5], b[6], b[7]]);
return Ok((buffer_ptr, buffer_len));
}
/// Takes a `(ptr, len)` pair and returns the corresponding deserialized buffer.
fn deserialize_from_buffer<R: DeserializeOwned>(
plugin_memory: &mut Memory,
store: &mut Store<WasiCtx>,
buffer_ptr: u32,
buffer_len: u32,
) -> Result<R, Error> {
let buffer_ptr = buffer_ptr as usize;
let buffer_len = buffer_len as usize;
let buffer_end = buffer_ptr + buffer_len; let buffer_end = buffer_ptr + buffer_len;
// read the buffer at this point into a byte array // read the buffer at this point into a byte array
// deserialize the byte array into the provided serde type // deserialize the byte array into the provided serde type
let result = &plugin_memory.data(&mut self.store)[buffer_ptr..buffer_end]; let result = &plugin_memory.data(store)[buffer_ptr..buffer_end];
let result = bincode::deserialize(result)?; let result = bincode::deserialize(result)?;
// TODO: this is handled wasm-side, but I'd like to double-check // TODO: this is handled wasm-side, but I'd like to double-check
@ -337,22 +351,31 @@ impl Wasi {
// dbg!(&handle.name); // dbg!(&handle.name);
// dbg!(serde_json::to_string(&arg)).unwrap(); // dbg!(serde_json::to_string(&arg)).unwrap();
let mut plugin_memory = self
.instance
.get_memory(&mut self.store, "memory")
.ok_or_else(|| anyhow!("Could not grab slice of plugin memory"))?;
// write the argument to linear memory // write the argument to linear memory
// this returns a (ptr, lentgh) pair // this returns a (ptr, lentgh) pair
let arg_buffer = self.serialize_to_buffer(arg).await?; let arg_buffer =
Self::serialize_to_buffer(self.alloc_buffer, &mut plugin_memory, &mut self.store, arg)
// get the webassembly function we want to actually call .await?;
// TODO: precompute handle
// let fun_name = format!("__{}", handle);
// let fun = self
// .instance
// .get_typed_func::<(u32, u32), u32, _>(&mut self.store, &fun_name)?;
let fun = handle.function;
// call the function, passing in the buffer and its length // call the function, passing in the buffer and its length
// this returns a ptr to a (ptr, lentgh) pair // this returns a ptr to a (ptr, lentgh) pair
let result_buffer = fun.call_async(&mut self.store, arg_buffer).await?; let result_buffer = handle
.function
.call_async(&mut self.store, arg_buffer)
.await?;
let (result_buffer_ptr, result_buffer_len) =
Self::deref_buffer(&mut plugin_memory, &mut self.store, result_buffer)?;
self.deserialize_from_buffer(result_buffer) Self::deserialize_from_buffer(
&mut plugin_memory,
&mut self.store,
result_buffer_ptr,
result_buffer_len,
)
} }
} }