237 lines
5.9 KiB
Rust

use core::{
ffi::{c_char, c_int},
ptr::NonNull,
};
use yggdrasil_rt::io::{FileAttr, FileMode, FileType, RawFd};
use crate::{
error::{CIntZeroResult, TryFromExt},
io,
util::{self, PointerStrExt},
};
use super::{
fcntl::{AT_FDCWD, AT_SYMLINK_NOFOLLOW},
sys_time::{__ygg_timespec_t, timespec},
sys_types::{blkcnt_t, blksize_t, dev_t, gid_t, ino_t, mode_t, nlink_t, off_t, uid_t},
};
#[derive(Clone, Copy, PartialEq, Eq, Debug, Default)]
#[repr(C)]
pub struct stat {
pub st_dev: dev_t,
pub st_ino: ino_t,
pub st_mode: mode_t,
pub st_nlink: nlink_t,
pub st_uid: uid_t,
pub st_gid: gid_t,
pub st_rdev: dev_t,
pub st_size: off_t,
pub st_atim: __ygg_timespec_t,
pub st_mtim: __ygg_timespec_t,
pub st_ctim: __ygg_timespec_t,
pub st_blksize: blksize_t,
pub st_blocks: blkcnt_t,
}
pub const S_IFMT: c_int = 0xF << 16;
pub const S_IFBLK: c_int = 1 << 16;
pub const S_IFCHR: c_int = 2 << 16;
pub const S_IFIFO: c_int = 3 << 16;
pub const S_IFREG: c_int = 4 << 16;
pub const S_IFDIR: c_int = 5 << 16;
pub const S_IFLNK: c_int = 6 << 16;
pub const S_IFSOCK: c_int = 7 << 16;
pub const S_IRWXU: c_int = 0x7 << 6;
pub const S_IRUSR: c_int = 0x4 << 6;
pub const S_IWUSR: c_int = 0x2 << 6;
pub const S_IXUSR: c_int = 0x1 << 6;
pub const S_IRWXG: c_int = 0x7 << 3;
pub const S_IRGRP: c_int = 0x4 << 3;
pub const S_IWGRP: c_int = 0x2 << 3;
pub const S_IXGRP: c_int = 0x1 << 3;
pub const S_IRWXO: c_int = 0x7 << 0;
pub const S_IROTH: c_int = 0x4 << 0;
pub const S_IWOTH: c_int = 0x2 << 0;
pub const S_IXOTH: c_int = 0x1 << 0;
pub const S_ISUID: c_int = 1 << 11;
pub const S_ISGID: c_int = 1 << 10;
pub const S_ISVTX: c_int = 1 << 9;
impl From<FileAttr> for stat {
fn from(value: FileAttr) -> Self {
// TODO no translation for st_dev/st_rdev/st_ino/etc
let mut st_mode = (value.mode.bits() & 0o777) as _;
match value.ty {
FileType::Block => st_mode |= S_IFBLK,
FileType::Char => st_mode |= S_IFCHR,
FileType::File => st_mode |= S_IFREG,
FileType::Directory => st_mode |= S_IFDIR,
FileType::Symlink => st_mode |= S_IFLNK,
}
let st_size: off_t = value.size.try_into().unwrap();
let st_uid = u32::from(value.uid).try_into().unwrap();
let st_gid = u32::from(value.gid).try_into().unwrap();
let st_blksize = value.block_size as _;
let st_blocks = st_size.div_ceil(st_blksize as _).try_into().unwrap();
let st_ino = value.inode.unwrap_or(0) as u64;
let st_ctim = timespec::from(value.ctime);
let st_mtim = timespec::from(value.mtime);
let st_atim = timespec::from(value.atime);
log::info!("{st_mtim:?}, {:?}", value.mtime);
Self {
st_mode,
st_size,
st_uid,
st_gid,
st_blksize,
st_blocks,
st_ino,
st_mtim,
st_atim,
st_ctim,
st_nlink: 1,
st_dev: 0,
st_rdev: 0,
}
}
}
#[no_mangle]
unsafe extern "C" fn chmod(_pathname: *const c_char, _mode: mode_t) -> c_int {
todo!()
}
#[no_mangle]
unsafe extern "C" fn fchmod(_fd: c_int, _mode: mode_t) -> c_int {
todo!()
}
#[no_mangle]
unsafe extern "C" fn fchmodat(
_fd: c_int,
_pathname: *const c_char,
_mode: mode_t,
_opt: c_int,
) -> c_int {
todo!()
}
#[no_mangle]
unsafe extern "C" fn umask(_mode: mode_t) -> mode_t {
todo!()
// let new = FileMode::new((mode as u32) & 0o777);
// let old = process::update_umask(new);
// old.bits() as mode_t
}
// Create stuff
#[no_mangle]
unsafe extern "C" fn mkdir(pathname: *const c_char, mode: mode_t) -> CIntZeroResult {
mkdirat(AT_FDCWD, pathname, mode)
}
#[no_mangle]
unsafe extern "C" fn mkdirat(atfd: c_int, pathname: *const c_char, mode: mode_t) -> CIntZeroResult {
let pathname = pathname.ensure_str();
let atfd = util::at_fd(atfd)?;
let mode = FileMode::new((mode & 0o777) as u32);
io::create_directory(atfd, pathname, mode)?;
CIntZeroResult::SUCCESS
}
#[no_mangle]
unsafe extern "C" fn mkfifo(_pathname: *const c_char, _mode: mode_t) -> c_int {
unimplemented!()
}
#[no_mangle]
unsafe extern "C" fn mkfifoat(_atfd: c_int, _pathname: *const c_char, _mode: mode_t) -> c_int {
unimplemented!()
}
#[no_mangle]
unsafe extern "C" fn mknod(_pathname: *const c_char, _mode: mode_t, _dev: dev_t) -> c_int {
unimplemented!()
}
#[no_mangle]
unsafe extern "C" fn mknodat(
_atfd: c_int,
_pathname: *const c_char,
_mode: mode_t,
_dev: dev_t,
) -> c_int {
unimplemented!()
}
// File status
#[no_mangle]
unsafe extern "C" fn fstat(fd: c_int, statbuf: *mut stat) -> CIntZeroResult {
let fd = RawFd::e_try_from(fd)?;
let attr = io::get_metadata(Some(fd), "", false)?;
if let Some(statbuf) = NonNull::new(statbuf) {
statbuf.write(attr.into());
}
CIntZeroResult::SUCCESS
}
#[no_mangle]
unsafe extern "C" fn fstatat(
atfd: c_int,
pathname: *const c_char,
statbuf: *mut stat,
opt: c_int,
) -> CIntZeroResult {
let pathname = pathname.ensure_str();
let atfd = util::at_fd(atfd)?;
let follow = opt & AT_SYMLINK_NOFOLLOW == 0;
let attr = io::get_metadata(atfd, pathname, follow)?;
if let Some(statbuf) = statbuf.as_mut() {
*statbuf = attr.into();
}
CIntZeroResult::SUCCESS
}
#[no_mangle]
unsafe extern "C" fn lstat(pathname: *const c_char, statbuf: *mut stat) -> CIntZeroResult {
fstatat(AT_FDCWD, pathname, statbuf, AT_SYMLINK_NOFOLLOW)
}
#[no_mangle]
unsafe extern "C" fn stat(pathname: *const c_char, statbuf: *mut stat) -> CIntZeroResult {
fstatat(AT_FDCWD, pathname, statbuf, 0)
}
// File time updates
#[no_mangle]
unsafe extern "C" fn futimens(_fd: c_int, _times: *const __ygg_timespec_t) -> c_int {
todo!()
}
#[no_mangle]
unsafe extern "C" fn utimensat(
_atfd: c_int,
_pathname: *const c_char,
_times: *const __ygg_timespec_t,
_opt: c_int,
) -> c_int {
todo!()
}