Currently, the use of the downcast method means that you have to use one of the hard-coded types. But Enarx needs to define its own `WasiFile` implementations. This works fine, except the resulting files cannot be used in poll because they aren't part of the hard-coded list. Replace this with an accessor method for the pollable type in `WasiFile`. Because we provide a default implementation of the method and manually implement it on all the hard-coded types, this is backwards compatible. Signed-off-by: Nathaniel McCallum <nathaniel@profian.com>
248 lines
7.8 KiB
Rust
248 lines
7.8 KiB
Rust
use cap_fs_ext::MetadataExt;
|
|
use fs_set_times::{SetTimes, SystemTimeSpec};
|
|
use is_terminal::IsTerminal;
|
|
use std::any::Any;
|
|
use std::convert::TryInto;
|
|
use std::io;
|
|
use system_interface::{
|
|
fs::{FileIoExt, GetSetFdFlags},
|
|
io::ReadReady,
|
|
};
|
|
use wasi_common::{
|
|
file::{Advice, FdFlags, FileType, Filestat, WasiFile},
|
|
Error, ErrorExt,
|
|
};
|
|
|
|
pub struct File(cap_std::fs::File);
|
|
|
|
impl File {
|
|
pub fn from_cap_std(file: cap_std::fs::File) -> Self {
|
|
File(file)
|
|
}
|
|
}
|
|
|
|
#[async_trait::async_trait]
|
|
impl WasiFile for File {
|
|
fn as_any(&self) -> &dyn Any {
|
|
self
|
|
}
|
|
#[cfg(unix)]
|
|
fn pollable(&self) -> Option<rustix::fd::BorrowedFd> {
|
|
Some(self.0.as_fd())
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
fn pollable(&self) -> Option<io_extras::os::windows::RawHandleOrSocket> {
|
|
Some(self.0.as_raw_handle_or_socket())
|
|
}
|
|
async fn datasync(&mut self) -> Result<(), Error> {
|
|
self.0.sync_data()?;
|
|
Ok(())
|
|
}
|
|
async fn sync(&mut self) -> Result<(), Error> {
|
|
self.0.sync_all()?;
|
|
Ok(())
|
|
}
|
|
async fn get_filetype(&mut self) -> Result<FileType, Error> {
|
|
let meta = self.0.metadata()?;
|
|
Ok(filetype_from(&meta.file_type()))
|
|
}
|
|
async fn get_fdflags(&mut self) -> Result<FdFlags, Error> {
|
|
let fdflags = self.0.get_fd_flags()?;
|
|
Ok(from_sysif_fdflags(fdflags))
|
|
}
|
|
async fn set_fdflags(&mut self, fdflags: FdFlags) -> Result<(), Error> {
|
|
if fdflags.intersects(
|
|
wasi_common::file::FdFlags::DSYNC
|
|
| wasi_common::file::FdFlags::SYNC
|
|
| wasi_common::file::FdFlags::RSYNC,
|
|
) {
|
|
return Err(Error::invalid_argument().context("cannot set DSYNC, SYNC, or RSYNC flag"));
|
|
}
|
|
let set_fd_flags = self.0.new_set_fd_flags(to_sysif_fdflags(fdflags))?;
|
|
self.0.set_fd_flags(set_fd_flags)?;
|
|
Ok(())
|
|
}
|
|
async fn get_filestat(&mut self) -> Result<Filestat, Error> {
|
|
let meta = self.0.metadata()?;
|
|
Ok(Filestat {
|
|
device_id: meta.dev(),
|
|
inode: meta.ino(),
|
|
filetype: filetype_from(&meta.file_type()),
|
|
nlink: meta.nlink(),
|
|
size: meta.len(),
|
|
atim: meta.accessed().map(|t| Some(t.into_std())).unwrap_or(None),
|
|
mtim: meta.modified().map(|t| Some(t.into_std())).unwrap_or(None),
|
|
ctim: meta.created().map(|t| Some(t.into_std())).unwrap_or(None),
|
|
})
|
|
}
|
|
async fn set_filestat_size(&mut self, size: u64) -> Result<(), Error> {
|
|
self.0.set_len(size)?;
|
|
Ok(())
|
|
}
|
|
async fn advise(&mut self, offset: u64, len: u64, advice: Advice) -> Result<(), Error> {
|
|
self.0.advise(offset, len, convert_advice(advice))?;
|
|
Ok(())
|
|
}
|
|
async fn allocate(&mut self, offset: u64, len: u64) -> Result<(), Error> {
|
|
self.0.allocate(offset, len)?;
|
|
Ok(())
|
|
}
|
|
async fn set_times(
|
|
&mut self,
|
|
atime: Option<wasi_common::SystemTimeSpec>,
|
|
mtime: Option<wasi_common::SystemTimeSpec>,
|
|
) -> Result<(), Error> {
|
|
self.0
|
|
.set_times(convert_systimespec(atime), convert_systimespec(mtime))?;
|
|
Ok(())
|
|
}
|
|
async fn read_vectored<'a>(&mut self, bufs: &mut [io::IoSliceMut<'a>]) -> Result<u64, Error> {
|
|
let n = self.0.read_vectored(bufs)?;
|
|
Ok(n.try_into()?)
|
|
}
|
|
async fn read_vectored_at<'a>(
|
|
&mut self,
|
|
bufs: &mut [io::IoSliceMut<'a>],
|
|
offset: u64,
|
|
) -> Result<u64, Error> {
|
|
let n = self.0.read_vectored_at(bufs, offset)?;
|
|
Ok(n.try_into()?)
|
|
}
|
|
async fn write_vectored<'a>(&mut self, bufs: &[io::IoSlice<'a>]) -> Result<u64, Error> {
|
|
let n = self.0.write_vectored(bufs)?;
|
|
Ok(n.try_into()?)
|
|
}
|
|
async fn write_vectored_at<'a>(
|
|
&mut self,
|
|
bufs: &[io::IoSlice<'a>],
|
|
offset: u64,
|
|
) -> Result<u64, Error> {
|
|
let n = self.0.write_vectored_at(bufs, offset)?;
|
|
Ok(n.try_into()?)
|
|
}
|
|
async fn seek(&mut self, pos: std::io::SeekFrom) -> Result<u64, Error> {
|
|
Ok(self.0.seek(pos)?)
|
|
}
|
|
async fn peek(&mut self, buf: &mut [u8]) -> Result<u64, Error> {
|
|
let n = self.0.peek(buf)?;
|
|
Ok(n.try_into()?)
|
|
}
|
|
async fn num_ready_bytes(&self) -> Result<u64, Error> {
|
|
Ok(self.0.num_ready_bytes()?)
|
|
}
|
|
fn isatty(&mut self) -> bool {
|
|
self.0.is_terminal()
|
|
}
|
|
}
|
|
|
|
pub fn filetype_from(ft: &cap_std::fs::FileType) -> FileType {
|
|
use cap_fs_ext::FileTypeExt;
|
|
if ft.is_dir() {
|
|
FileType::Directory
|
|
} else if ft.is_symlink() {
|
|
FileType::SymbolicLink
|
|
} else if ft.is_socket() {
|
|
if ft.is_block_device() {
|
|
FileType::SocketDgram
|
|
} else {
|
|
FileType::SocketStream
|
|
}
|
|
} else if ft.is_block_device() {
|
|
FileType::BlockDevice
|
|
} else if ft.is_char_device() {
|
|
FileType::CharacterDevice
|
|
} else if ft.is_file() {
|
|
FileType::RegularFile
|
|
} else {
|
|
FileType::Unknown
|
|
}
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
use io_lifetimes::{AsHandle, BorrowedHandle};
|
|
#[cfg(windows)]
|
|
impl AsHandle for File {
|
|
fn as_handle(&self) -> BorrowedHandle<'_> {
|
|
self.0.as_handle()
|
|
}
|
|
}
|
|
|
|
#[cfg(windows)]
|
|
use io_extras::os::windows::{AsRawHandleOrSocket, RawHandleOrSocket};
|
|
#[cfg(windows)]
|
|
impl AsRawHandleOrSocket for File {
|
|
#[inline]
|
|
fn as_raw_handle_or_socket(&self) -> RawHandleOrSocket {
|
|
self.0.as_raw_handle_or_socket()
|
|
}
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
use io_lifetimes::{AsFd, BorrowedFd};
|
|
|
|
#[cfg(unix)]
|
|
impl AsFd for File {
|
|
fn as_fd(&self) -> BorrowedFd<'_> {
|
|
self.0.as_fd()
|
|
}
|
|
}
|
|
pub fn convert_systimespec(t: Option<wasi_common::SystemTimeSpec>) -> Option<SystemTimeSpec> {
|
|
match t {
|
|
Some(wasi_common::SystemTimeSpec::Absolute(t)) => {
|
|
Some(SystemTimeSpec::Absolute(t.into_std()))
|
|
}
|
|
Some(wasi_common::SystemTimeSpec::SymbolicNow) => Some(SystemTimeSpec::SymbolicNow),
|
|
None => None,
|
|
}
|
|
}
|
|
|
|
pub fn to_sysif_fdflags(f: wasi_common::file::FdFlags) -> system_interface::fs::FdFlags {
|
|
let mut out = system_interface::fs::FdFlags::empty();
|
|
if f.contains(wasi_common::file::FdFlags::APPEND) {
|
|
out |= system_interface::fs::FdFlags::APPEND;
|
|
}
|
|
if f.contains(wasi_common::file::FdFlags::DSYNC) {
|
|
out |= system_interface::fs::FdFlags::DSYNC;
|
|
}
|
|
if f.contains(wasi_common::file::FdFlags::NONBLOCK) {
|
|
out |= system_interface::fs::FdFlags::NONBLOCK;
|
|
}
|
|
if f.contains(wasi_common::file::FdFlags::RSYNC) {
|
|
out |= system_interface::fs::FdFlags::RSYNC;
|
|
}
|
|
if f.contains(wasi_common::file::FdFlags::SYNC) {
|
|
out |= system_interface::fs::FdFlags::SYNC;
|
|
}
|
|
out
|
|
}
|
|
pub fn from_sysif_fdflags(f: system_interface::fs::FdFlags) -> wasi_common::file::FdFlags {
|
|
let mut out = wasi_common::file::FdFlags::empty();
|
|
if f.contains(system_interface::fs::FdFlags::APPEND) {
|
|
out |= wasi_common::file::FdFlags::APPEND;
|
|
}
|
|
if f.contains(system_interface::fs::FdFlags::DSYNC) {
|
|
out |= wasi_common::file::FdFlags::DSYNC;
|
|
}
|
|
if f.contains(system_interface::fs::FdFlags::NONBLOCK) {
|
|
out |= wasi_common::file::FdFlags::NONBLOCK;
|
|
}
|
|
if f.contains(system_interface::fs::FdFlags::RSYNC) {
|
|
out |= wasi_common::file::FdFlags::RSYNC;
|
|
}
|
|
if f.contains(system_interface::fs::FdFlags::SYNC) {
|
|
out |= wasi_common::file::FdFlags::SYNC;
|
|
}
|
|
out
|
|
}
|
|
pub fn convert_advice(advice: Advice) -> system_interface::fs::Advice {
|
|
match advice {
|
|
Advice::Normal => system_interface::fs::Advice::Normal,
|
|
Advice::Sequential => system_interface::fs::Advice::Sequential,
|
|
Advice::Random => system_interface::fs::Advice::Random,
|
|
Advice::WillNeed => system_interface::fs::Advice::WillNeed,
|
|
Advice::DontNeed => system_interface::fs::Advice::DontNeed,
|
|
Advice::NoReuse => system_interface::fs::Advice::NoReuse,
|
|
}
|
|
}
|