use tracing::{field, instrument, trace_span};
use wasmer::{AsStoreMut, AsStoreRef, FunctionEnvMut, Memory, WasmPtr};
use wasmer_wasix_types::wasi::{
Errno, Event, EventFdReadwrite, Eventrwflags, Eventtype, ExitCode, Fd, Filesize, Filestat,
Filetype, Snapshot0Event, Snapshot0Filestat, Snapshot0Subscription, Snapshot0Whence,
Subscription, Whence,
};
use crate::{
mem_error_to_wasi,
os::task::thread::WasiThread,
state::{PollEventBuilder, PollEventSet},
syscalls::types,
syscalls::{self, handle_rewind},
Memory32, MemorySize, WasiEnv, WasiError,
};
#[instrument(level = "trace", skip_all, ret)]
pub fn fd_filestat_get(
mut ctx: FunctionEnvMut<WasiEnv>,
fd: Fd,
buf: WasmPtr<Snapshot0Filestat, Memory32>,
) -> Errno {
let env = ctx.data();
let memory = unsafe { env.memory_view(&ctx) };
let result = syscalls::fd_filestat_get_old::<Memory32>(ctx.as_mut(), fd, buf);
result
}
#[instrument(level = "trace", skip_all, ret)]
pub fn path_filestat_get(
mut ctx: FunctionEnvMut<WasiEnv>,
fd: Fd,
flags: types::LookupFlags,
path: WasmPtr<u8, Memory32>,
path_len: u32,
buf: WasmPtr<Snapshot0Filestat, Memory32>,
) -> Errno {
let env = ctx.data();
let memory = unsafe { env.memory_view(&ctx) };
let result =
syscalls::path_filestat_get_old::<Memory32>(ctx.as_mut(), fd, flags, path, path_len, buf);
result
}
#[instrument(level = "trace", skip_all, ret)]
pub fn fd_seek(
ctx: FunctionEnvMut<WasiEnv>,
fd: Fd,
offset: types::FileDelta,
whence: Snapshot0Whence,
newoffset: WasmPtr<Filesize, Memory32>,
) -> Result<Errno, WasiError> {
let new_whence = match whence {
Snapshot0Whence::Cur => Whence::Cur,
Snapshot0Whence::End => Whence::End,
Snapshot0Whence::Set => Whence::Set,
Snapshot0Whence::Unknown => return Ok(Errno::Inval),
};
syscalls::fd_seek::<Memory32>(ctx, fd, offset, new_whence, newoffset)
}
#[instrument(level = "trace", skip_all, fields(timeout_ms = field::Empty, fd_guards = field::Empty, seen = field::Empty), ret)]
pub fn poll_oneoff<M: MemorySize>(
mut ctx: FunctionEnvMut<WasiEnv>,
in_: WasmPtr<Snapshot0Subscription, Memory32>,
out_: WasmPtr<Snapshot0Event, Memory32>,
nsubscriptions: u32,
nevents: WasmPtr<u32, Memory32>,
) -> Result<Errno, WasiError> {
wasi_try_ok!(WasiEnv::process_signals_and_exit(&mut ctx)?);
let env = ctx.data();
let memory = unsafe { env.memory_view(&ctx) };
let mut subscriptions = Vec::new();
let in_origs = wasi_try_mem_ok!(in_.slice(&memory, nsubscriptions));
let in_origs = wasi_try_mem_ok!(in_origs.read_to_vec());
for in_orig in in_origs {
subscriptions.push((
None,
PollEventSet::default(),
Into::<Subscription>::into(in_orig),
));
}
let process_events = |ctx: &FunctionEnvMut<'_, WasiEnv>, triggered_events: Vec<Event>| {
let env = ctx.data();
let memory = unsafe { env.memory_view(&ctx) };
let mut events_seen: u32 = 0;
let event_array = wasi_try_mem!(out_.slice(&memory, nsubscriptions));
for event in triggered_events {
let event = Snapshot0Event {
userdata: event.userdata,
error: event.error,
type_: Eventtype::FdRead,
fd_readwrite: match event.type_ {
Eventtype::FdRead => unsafe { event.u.fd_readwrite },
Eventtype::FdWrite => unsafe { event.u.fd_readwrite },
Eventtype::Clock => EventFdReadwrite {
nbytes: 0,
flags: Eventrwflags::empty(),
},
Eventtype::Unknown => return Errno::Inval,
},
};
wasi_try_mem!(event_array.index(events_seen as u64).write(event));
events_seen += 1;
}
let out_ptr = nevents.deref(&memory);
wasi_try_mem!(out_ptr.write(events_seen));
Errno::Success
};
wasi_try_mem_ok!(nevents.write(&memory, 0));
syscalls::poll_oneoff_internal::<M, _>(ctx, subscriptions, process_events)
}