use alloc::sync::Arc;
use core::ffi::{c_char, c_int};
#[cfg(not(feature = "use-hermit-types"))]
use core::mem::size_of;
use ax_errno::{LinuxError, LinuxResult};
use ax_fs::fops::OpenOptions;
use ax_io::{PollState, SeekFrom};
use ax_sync::Mutex;
use super::fd_ops::{FileLike, get_file_like};
use crate::{ctypes, utils::char_ptr_to_str};
pub struct File {
inner: Mutex<ax_fs::fops::File>,
}
pub struct Directory {
inner: Mutex<ax_fs::fops::Directory>,
}
#[cfg(not(feature = "use-hermit-types"))]
#[repr(C, packed)]
struct LinuxDirent64Head {
d_ino: u64,
d_off: i64,
d_reclen: u16,
d_type: u8,
}
#[cfg(not(feature = "use-hermit-types"))]
struct DirBuffer<'a> {
buf: &'a mut [u8],
offset: usize,
}
#[cfg(not(feature = "use-hermit-types"))]
impl<'a> DirBuffer<'a> {
fn new(buf: &'a mut [u8]) -> Self {
Self { buf, offset: 0 }
}
fn used_len(&self) -> usize {
self.offset
}
fn remaining_space(&self) -> usize {
self.buf.len().saturating_sub(self.offset)
}
fn write_entry(&mut self, d_ino: u64, d_off: i64, d_type: u8, name: &[u8]) -> bool {
const NAME_OFFSET: usize = size_of::<LinuxDirent64Head>();
let name_len = name.len().min(255);
let reclen = (NAME_OFFSET + name_len + 1).next_multiple_of(8);
if self.remaining_space() < reclen {
return false;
}
unsafe {
let entry_ptr = self.buf.as_mut_ptr().add(self.offset);
entry_ptr
.cast::<LinuxDirent64Head>()
.write_unaligned(LinuxDirent64Head {
d_ino,
d_off,
d_reclen: reclen as _,
d_type,
});
let name_ptr = entry_ptr.add(NAME_OFFSET);
name_ptr.copy_from_nonoverlapping(name.as_ptr(), name_len);
name_ptr.add(name_len).write(0);
}
self.offset += reclen;
true
}
}
#[cfg(feature = "use-hermit-types")]
use core::mem;
#[cfg(feature = "use-hermit-types")]
struct HermitDirBuffer<'a> {
buf: &'a mut [u8],
offset: usize,
}
#[cfg(feature = "use-hermit-types")]
impl<'a> HermitDirBuffer<'a> {
fn new(buf: &'a mut [u8]) -> Self {
Self { buf, offset: 0 }
}
fn used_len(&self) -> usize {
self.offset
}
fn remaining_space(&self) -> usize {
self.buf.len().saturating_sub(self.offset)
}
fn write_entry(&mut self, d_ino: u64, d_type: u8, name: &[u8]) -> bool {
const NAME_OFFSET: usize = 19;
let name_len = name.len().min(255);
let dirent_len = NAME_OFFSET + name_len + 1;
let reclen = dirent_len.next_multiple_of(mem::align_of::<ctypes::dirent64>());
if self.remaining_space() < reclen {
return false;
}
unsafe {
let entry_ptr = self.buf.as_mut_ptr().add(self.offset);
let d_ino_ptr = entry_ptr.cast::<u64>();
d_ino_ptr.write_unaligned(d_ino);
let d_off_ptr = entry_ptr.add(8).cast::<i64>();
d_off_ptr.write_unaligned(0);
let d_reclen_ptr = entry_ptr.add(16).cast::<u16>();
d_reclen_ptr.write_unaligned(reclen as u16);
let d_type_ptr = entry_ptr.add(18);
d_type_ptr.write(d_type);
let name_ptr = entry_ptr.add(NAME_OFFSET);
name_ptr.copy_from_nonoverlapping(name.as_ptr(), name_len);
name_ptr.add(name_len).write(0); }
self.offset += reclen;
true
}
}
fn file_type_to_d_type(ty: ax_fs::fops::FileType) -> u8 {
match ty {
ax_fs::fops::FileType::Dir => 4, ax_fs::fops::FileType::File => 8, ax_fs::fops::FileType::SymLink => 10, _ => 0, }
}
impl File {
fn new(inner: ax_fs::fops::File) -> Self {
Self {
inner: Mutex::new(inner),
}
}
fn add_to_fd_table(self) -> LinuxResult<c_int> {
super::fd_ops::add_file_like(Arc::new(self))
}
fn from_fd(fd: c_int) -> LinuxResult<Arc<Self>> {
let f = super::fd_ops::get_file_like(fd)?;
f.into_any()
.downcast::<Self>()
.map_err(|_| LinuxError::EINVAL)
}
}
impl Directory {
fn new(inner: ax_fs::fops::Directory) -> Self {
Self {
inner: Mutex::new(inner),
}
}
fn add_to_fd_table(self) -> LinuxResult<c_int> {
super::fd_ops::add_file_like(Arc::new(self))
}
fn from_fd(fd: c_int) -> LinuxResult<Arc<Self>> {
let f = super::fd_ops::get_file_like(fd)?;
f.into_any()
.downcast::<Self>()
.map_err(|_| LinuxError::ENOTDIR)
}
}
impl FileLike for File {
fn read(&self, buf: &mut [u8]) -> LinuxResult<usize> {
Ok(self.inner.lock().read(buf)?)
}
fn write(&self, buf: &[u8]) -> LinuxResult<usize> {
Ok(self.inner.lock().write(buf)?)
}
fn stat(&self) -> LinuxResult<ctypes::stat> {
let metadata = self.inner.lock().get_attr()?;
let ty = metadata.file_type() as u8;
let perm = metadata.perm().bits() as u32;
let st_mode = ((ty as u32) << 12) | perm;
Ok(ctypes::stat {
st_ino: 1,
st_nlink: 1,
st_mode,
st_uid: 1000,
st_gid: 1000,
st_size: metadata.size() as _,
st_blocks: metadata.blocks() as _,
st_blksize: 512,
..Default::default()
})
}
fn into_any(self: Arc<Self>) -> Arc<dyn core::any::Any + Send + Sync> {
self
}
fn poll(&self) -> LinuxResult<PollState> {
Ok(PollState {
readable: true,
writable: true,
})
}
fn set_nonblocking(&self, _nonblocking: bool) -> LinuxResult {
Ok(())
}
}
impl FileLike for Directory {
fn read(&self, _buf: &mut [u8]) -> LinuxResult<usize> {
Err(LinuxError::EISDIR)
}
fn write(&self, _buf: &[u8]) -> LinuxResult<usize> {
Err(LinuxError::EISDIR)
}
fn stat(&self) -> LinuxResult<ctypes::stat> {
let st_mode = 0o040755;
Ok(ctypes::stat {
st_ino: 1,
st_nlink: 1,
st_mode,
st_uid: 1000,
st_gid: 1000,
st_size: 0,
st_blocks: 0,
st_blksize: 512,
..Default::default()
})
}
fn into_any(self: Arc<Self>) -> Arc<dyn core::any::Any + Send + Sync> {
self
}
fn poll(&self) -> LinuxResult<PollState> {
Ok(PollState {
readable: true,
writable: false,
})
}
fn set_nonblocking(&self, _nonblocking: bool) -> LinuxResult {
Ok(())
}
}
fn flags_to_options(flags: c_int, _mode: ctypes::mode_t) -> OpenOptions {
let flags = flags as u32;
let mut options = OpenOptions::new();
match flags & 0b11 {
ctypes::O_RDONLY => options.read(true),
ctypes::O_WRONLY => options.write(true),
_ => {
options.read(true);
options.write(true);
}
};
if flags & ctypes::O_APPEND != 0 {
options.append(true);
}
if flags & ctypes::O_TRUNC != 0 {
options.truncate(true);
}
if flags & ctypes::O_CREAT != 0 {
options.create(true);
}
if flags & ctypes::O_EXEC != 0 {
options.create_new(true);
}
options
}
pub fn sys_open(filename: *const c_char, flags: c_int, mode: ctypes::mode_t) -> c_int {
let filename = char_ptr_to_str(filename);
debug!("sys_open <= {filename:?} {flags:#o} {mode:#o}");
syscall_body!(sys_open, {
let options = flags_to_options(flags, mode);
let filename = filename?;
if (flags as u32) & ctypes::O_DIRECTORY != 0 {
let dir = ax_fs::fops::Directory::open_dir(filename, &options)?;
Directory::new(dir).add_to_fd_table()
} else {
let file = ax_fs::fops::File::open(filename, &options)?;
File::new(file).add_to_fd_table()
}
})
}
#[cfg(not(feature = "use-hermit-types"))]
pub unsafe fn sys_getdents64(fd: c_int, buf: *mut u8, len: usize) -> ctypes::ssize_t {
debug!("sys_getdents64 (Linux) <= {fd} {:#x} {len}", buf as usize);
syscall_body!(sys_getdents64, {
if buf.is_null() || len == 0 {
return Err(LinuxError::EINVAL);
}
let dir = Directory::from_fd(fd).map_err(|_| LinuxError::EBADF)?;
let mut dir = dir.inner.lock();
let out = unsafe { core::slice::from_raw_parts_mut(buf, len) };
let mut dir_buf = DirBuffer::new(out);
let mut entries: [ax_fs::fops::DirEntry; 16] =
core::array::from_fn(|_| ax_fs::fops::DirEntry::default());
loop {
let nr = dir.read_dir(&mut entries)?;
if nr == 0 {
break;
}
for entry in entries.iter().take(nr) {
let d_type = file_type_to_d_type(entry.entry_type());
if !dir_buf.write_entry(1, 0, d_type, entry.name_as_bytes()) {
return Ok(dir_buf.used_len() as ctypes::ssize_t);
}
}
}
Ok(dir_buf.used_len() as ctypes::ssize_t)
})
}
#[cfg(feature = "use-hermit-types")]
pub unsafe fn sys_getdents64(fd: c_int, buf: *mut u8, len: usize) -> ctypes::ssize_t {
debug!("sys_getdents64 (Hermit) <= {fd} {:#x} {len}", buf as usize);
syscall_body!(sys_getdents64, {
if buf.is_null() || len == 0 {
return Err(LinuxError::EINVAL);
}
let dir = Directory::from_fd(fd).map_err(|_| LinuxError::EINVAL)?;
let mut dir = dir.inner.lock();
let out = unsafe { core::slice::from_raw_parts_mut(buf, len) };
let mut dir_buf = HermitDirBuffer::new(out);
let mut entries: [ax_fs::fops::DirEntry; 16] =
core::array::from_fn(|_| ax_fs::fops::DirEntry::default());
loop {
let nr = dir.read_dir(&mut entries)?;
if nr == 0 {
break;
}
for entry in entries.iter().take(nr) {
let d_type = file_type_to_d_type(entry.entry_type());
if !dir_buf.write_entry(1, d_type, entry.name_as_bytes()) {
return Ok(dir_buf.used_len() as ctypes::ssize_t);
}
}
}
Ok(dir_buf.used_len() as ctypes::ssize_t)
})
}
pub fn sys_lseek(fd: c_int, offset: ctypes::off_t, whence: c_int) -> ctypes::off_t {
debug!("sys_lseek <= {fd} {offset} {whence}");
syscall_body!(sys_lseek, {
let pos = match whence {
0 => SeekFrom::Start(offset as _),
1 => SeekFrom::Current(offset as _),
2 => SeekFrom::End(offset as _),
_ => return Err(LinuxError::EINVAL),
};
let off = File::from_fd(fd)?.inner.lock().seek(pos)?;
Ok(off)
})
}
pub unsafe fn sys_stat(path: *const c_char, buf: *mut ctypes::stat) -> c_int {
let path = char_ptr_to_str(path);
debug!("sys_stat <= {:?} {:#x}", path, buf as usize);
syscall_body!(sys_stat, {
if buf.is_null() {
return Err(LinuxError::EFAULT);
}
let mut options = OpenOptions::new();
options.read(true);
let file = ax_fs::fops::File::open(path?, &options)?;
let st = File::new(file).stat()?;
unsafe { *buf = st };
Ok(0)
})
}
pub unsafe fn sys_fstat(fd: c_int, buf: *mut ctypes::stat) -> c_int {
debug!("sys_fstat <= {} {:#x}", fd, buf as usize);
syscall_body!(sys_fstat, {
if buf.is_null() {
return Err(LinuxError::EFAULT);
}
unsafe { *buf = get_file_like(fd)?.stat()? };
Ok(0)
})
}
pub unsafe fn sys_lstat(path: *const c_char, buf: *mut ctypes::stat) -> ctypes::ssize_t {
let path = char_ptr_to_str(path);
debug!("sys_lstat <= {:?} {:#x}", path, buf as usize);
syscall_body!(sys_lstat, {
if buf.is_null() {
return Err(LinuxError::EFAULT);
}
let mut options = OpenOptions::new();
options.read(true);
let file = ax_fs::fops::File::open(path?, &options)?;
let st = File::new(file).stat()?;
unsafe { *buf = st };
Ok(0)
})
}
#[allow(clippy::unnecessary_cast)] pub fn sys_getcwd(buf: *mut c_char, size: usize) -> *mut c_char {
debug!("sys_getcwd <= {:#x} {}", buf as usize, size);
syscall_body!(sys_getcwd, {
if buf.is_null() {
return Ok(core::ptr::null::<c_char>() as _);
}
let dst = unsafe { core::slice::from_raw_parts_mut(buf as *mut u8, size as _) };
let cwd = ax_fs::api::current_dir()?;
let cwd = cwd.as_bytes();
if cwd.len() < size {
dst[..cwd.len()].copy_from_slice(cwd);
dst[cwd.len()] = 0;
Ok(buf)
} else {
Err(LinuxError::ERANGE)
}
})
}
pub fn sys_rename(old: *const c_char, new: *const c_char) -> c_int {
syscall_body!(sys_rename, {
let old_path = char_ptr_to_str(old)?;
let new_path = char_ptr_to_str(new)?;
debug!("sys_rename <= old: {old_path:?}, new: {new_path:?}");
ax_fs::api::rename(old_path, new_path)?;
Ok(0)
})
}