b0f4fee4c4
Phase II.X.W: extend the AcpiVerb enum with two new variants for the S3 round-trip: * AcpiVerb::SetS3WakingVector (= 5): acpid writes the kernel's S3 resume trampoline address to FACS. Mirrors Linux 7.1's acpi_set_firmware_waking_vector. The 8-byte write payload is the address in little-endian. A zero payload is a sentinel for 'use the kernel's default trampoline address' (s3_trampoline symbol). * AcpiVerb::EnterS3 (= 6): acpid requests the kernel to enter S3. The kernel's stop::enter_s3() reads the SLP_TYP value from S3_SLP_TYP and does the PM1 register write. The actual S3 entry happens via acpid writing to /scheme/sys/kstop. Hardware-agnostic: works on any x86_64 system with standard ACPI S3 support (Dell, HP, Lenovo, LG Gram 14). On Modern Standby-only systems (LG Gram 16 (2025)), the kernel never enters S3 so these verbs are no-ops.
568 lines
18 KiB
Rust
568 lines
18 KiB
Rust
use bitflags::bitflags as inner_bitflags;
|
|
use core::{mem, ops::Deref, slice};
|
|
|
|
macro_rules! bitflags {
|
|
(
|
|
$(#[$outer:meta])*
|
|
pub struct $BitFlags:ident: $T:ty {
|
|
$(
|
|
$(#[$inner:ident $($args:tt)*])*
|
|
const $Flag:ident = $value:expr;
|
|
)+
|
|
}
|
|
) => {
|
|
// First, use the inner bitflags
|
|
inner_bitflags! {
|
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Hash, Debug, Clone, Copy, Default)]
|
|
$(#[$outer])*
|
|
pub struct $BitFlags: $T {
|
|
$(
|
|
$(#[$inner $($args)*])*
|
|
const $Flag = $value;
|
|
)+
|
|
}
|
|
}
|
|
|
|
impl $BitFlags {
|
|
#[deprecated = "use the safe `from_bits_retain` method instead"]
|
|
pub unsafe fn from_bits_unchecked(bits: $T) -> Self {
|
|
Self::from_bits_retain(bits)
|
|
}
|
|
}
|
|
|
|
// Secondly, re-export all inner constants
|
|
// (`pub use self::Struct::*` doesn't work)
|
|
$(
|
|
$(#[$inner $($args)*])*
|
|
pub const $Flag: $BitFlags = $BitFlags::$Flag;
|
|
)+
|
|
}
|
|
}
|
|
|
|
pub const CLOCK_REALTIME: usize = 1;
|
|
pub const CLOCK_MONOTONIC: usize = 4;
|
|
|
|
bitflags! {
|
|
pub struct EventFlags: usize {
|
|
const EVENT_NONE = 0;
|
|
const EVENT_READ = 1;
|
|
const EVENT_WRITE = 2;
|
|
}
|
|
}
|
|
|
|
pub const F_DUPFD: usize = 0;
|
|
pub const F_GETFD: usize = 1;
|
|
pub const F_SETFD: usize = 2;
|
|
pub const F_GETFL: usize = 3;
|
|
pub const F_SETFL: usize = 4;
|
|
pub const F_DUPFD_CLOEXEC: usize = 1030;
|
|
|
|
pub const FUTEX_WAIT: usize = 0;
|
|
pub const FUTEX_WAKE: usize = 1;
|
|
pub const FUTEX_REQUEUE: usize = 2;
|
|
pub const FUTEX_WAIT64: usize = 3;
|
|
|
|
// TODO: Split SendFdFlags into caller flags and flags that the scheme receives?
|
|
bitflags::bitflags! {
|
|
#[derive(Clone, Copy, Debug)]
|
|
pub struct SendFdFlags: usize {
|
|
/// If set, the kernel will enforce that the file descriptors are exclusively owned.
|
|
///
|
|
/// That is, there will no longer exist any other reference to those FDs when removed from
|
|
/// the file table (sendfd always removes the FDs from the file table, but without this
|
|
/// flag, it can be retained by SYS_DUPing them first).
|
|
const EXCLUSIVE = 1;
|
|
|
|
/// If set, the file descriptors will be cloned and *not* removed from the sender's file table.
|
|
/// By default, `SYS_SENDFD` moves the file descriptors, removing them from the sender.
|
|
const CLONE = 2;
|
|
}
|
|
}
|
|
bitflags::bitflags! {
|
|
#[derive(Clone, Copy, Debug)]
|
|
pub struct FobtainFdFlags: usize {
|
|
/// If set, the SYS_CALL payload specifies the destination file descriptor slots, otherwise the lowest
|
|
/// available slots will be selected, and placed in the usize pointed to by SYS_CALL
|
|
/// payload.
|
|
const MANUAL_FD = 1;
|
|
|
|
/// If set, the file descriptors received are guaranteed to be exclusively owned (by the file
|
|
/// table the obtainer is running in).
|
|
const EXCLUSIVE = 2;
|
|
|
|
/// If set, the file descriptors received will be placed into the *upper* file table.
|
|
const UPPER_TBL = 4;
|
|
|
|
/// If set, the received file descriptors are marked as close-on-exec.
|
|
const CLOEXEC = 8;
|
|
|
|
// No, cloexec won't be stored in the kernel in the future, when the stable ABI is moved to
|
|
// relibc, so no flag for that!
|
|
}
|
|
}
|
|
bitflags::bitflags! {
|
|
#[derive(Clone, Copy, Debug)]
|
|
pub struct RecvFdFlags: usize {
|
|
/// If set, the SYS_CALL payload specifies the destination file descriptor slots, otherwise the lowest
|
|
/// available slots will be selected, and placed in the usize pointed to by SYS_CALL
|
|
/// payload.
|
|
const MANUAL_FD = 1;
|
|
|
|
/// If set, the file descriptors received will be placed into the *upper* file table.
|
|
const UPPER_TBL = 2;
|
|
|
|
/// If set, the received file descriptors are marked as close-on-exec.
|
|
const CLOEXEC = 4;
|
|
}
|
|
}
|
|
bitflags::bitflags! {
|
|
#[derive(Clone, Copy, Debug)]
|
|
pub struct FmoveFdFlags: usize {
|
|
/// If set, the kernel will enforce that the file descriptors are exclusively owned.
|
|
///
|
|
/// That is, there will no longer exist any other reference to those FDs when removed from
|
|
/// the file table (SYS_CALL always removes the FDs from the file table, but without this
|
|
/// flag, it can be retained by SYS_DUPing them first).
|
|
const EXCLUSIVE = 1;
|
|
|
|
/// If set, the file descriptors will be cloned and *not* removed from the sender's file table.
|
|
/// By default, sendfd moves the file descriptors, removing them from the sender.
|
|
const CLONE = 2;
|
|
}
|
|
}
|
|
|
|
bitflags! {
|
|
pub struct MapFlags: usize {
|
|
// TODO: Downgrade PROT_NONE to global constant? (bitflags specifically states zero flags
|
|
// can cause buggy behavior).
|
|
const PROT_NONE = 0x0000_0000;
|
|
|
|
const PROT_EXEC = 0x0001_0000;
|
|
const PROT_WRITE = 0x0002_0000;
|
|
const PROT_READ = 0x0004_0000;
|
|
|
|
const MAP_SHARED = 0x0001;
|
|
const MAP_PRIVATE = 0x0002;
|
|
|
|
const MAP_FIXED = 0x0004;
|
|
const MAP_FIXED_NOREPLACE = 0x000C;
|
|
|
|
/// For *userspace-backed mmaps*, return from the mmap call before all pages have been
|
|
/// provided by the scheme. This requires the scheme to be trusted, as the current context
|
|
/// can block indefinitely, if the scheme does not respond to the page fault handler's
|
|
/// request, as it tries to map the page by requesting it from the scheme.
|
|
///
|
|
/// In some cases however, such as the program loader, the data needs to be trusted as much
|
|
/// with or without MAP_LAZY, and if so, mapping lazily will not cause insecureness by
|
|
/// itself.
|
|
///
|
|
/// For kernel-backed mmaps, this flag has no effect at all. It is unspecified whether
|
|
/// kernel mmaps are lazy or not.
|
|
const MAP_LAZY = 0x0010;
|
|
}
|
|
}
|
|
bitflags! {
|
|
pub struct MunmapFlags: usize {
|
|
/// Indicates whether the funmap call must implicitly do an msync, for the changes to
|
|
/// become visible later.
|
|
///
|
|
/// This flag will currently be set if and only if MAP_SHARED | PROT_WRITE are set.
|
|
const NEEDS_SYNC = 1;
|
|
}
|
|
}
|
|
|
|
pub const MODE_TYPE: u16 = 0xF000;
|
|
pub const MODE_DIR: u16 = 0x4000;
|
|
pub const MODE_FILE: u16 = 0x8000;
|
|
pub const MODE_SYMLINK: u16 = 0xA000;
|
|
pub const MODE_FIFO: u16 = 0x1000;
|
|
pub const MODE_CHR: u16 = 0x2000;
|
|
pub const MODE_SOCK: u16 = 0xC000;
|
|
|
|
pub const MODE_PERM: u16 = 0x0FFF;
|
|
pub const MODE_SETUID: u16 = 0o4000;
|
|
pub const MODE_SETGID: u16 = 0o2000;
|
|
|
|
pub const O_RDONLY: usize = 0x0001_0000;
|
|
pub const O_WRONLY: usize = 0x0002_0000;
|
|
pub const O_RDWR: usize = 0x0003_0000;
|
|
pub const O_NONBLOCK: usize = 0x0004_0000;
|
|
pub const O_APPEND: usize = 0x0008_0000;
|
|
pub const O_SHLOCK: usize = 0x0010_0000;
|
|
pub const O_EXLOCK: usize = 0x0020_0000;
|
|
pub const O_ASYNC: usize = 0x0040_0000;
|
|
pub const O_FSYNC: usize = 0x0080_0000;
|
|
pub const O_CLOEXEC: usize = 0x0100_0000;
|
|
pub const O_CREAT: usize = 0x0200_0000;
|
|
pub const O_TRUNC: usize = 0x0400_0000;
|
|
pub const O_EXCL: usize = 0x0800_0000;
|
|
pub const O_DIRECTORY: usize = 0x1000_0000;
|
|
pub const O_STAT: usize = 0x2000_0000;
|
|
pub const O_SYMLINK: usize = 0x4000_0000;
|
|
pub const O_NOFOLLOW: usize = 0x8000_0000;
|
|
pub const O_ACCMODE: usize = O_RDONLY | O_WRONLY | O_RDWR;
|
|
pub const O_FCNTL_MASK: usize = O_NONBLOCK | O_APPEND | O_ASYNC | O_FSYNC;
|
|
|
|
/// Remove directory instead of unlinking file.
|
|
pub const AT_REMOVEDIR: usize = 0x200;
|
|
|
|
// The top 48 bits of PTRACE_* are reserved, for now
|
|
|
|
// NOT ABI STABLE!
|
|
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
|
|
#[repr(usize)]
|
|
pub enum ContextStatus {
|
|
Runnable,
|
|
Blocked,
|
|
NotYetStarted,
|
|
Dead,
|
|
ForceKilled,
|
|
Stopped,
|
|
UnhandledExcp,
|
|
#[default]
|
|
Other, // reserved
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(usize)]
|
|
pub enum ContextVerb {
|
|
Stop = 1,
|
|
Unstop = 2,
|
|
Interrupt = 3,
|
|
ForceKill = usize::MAX,
|
|
}
|
|
impl ContextVerb {
|
|
pub fn try_from_raw(raw: usize) -> Option<Self> {
|
|
Some(match raw {
|
|
1 => Self::Stop,
|
|
2 => Self::Unstop,
|
|
3 => Self::Interrupt,
|
|
usize::MAX => Self::ForceKill,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
// NOT ABI STABLE!
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(u8)]
|
|
pub enum AddrSpaceVerb {
|
|
MmapMin = 255,
|
|
}
|
|
impl AddrSpaceVerb {
|
|
pub fn try_from_raw(verb: u8) -> Option<Self> {
|
|
Some(match verb {
|
|
255 => Self::MmapMin,
|
|
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
// NOT ABI STABLE!
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(u8)]
|
|
pub enum ProcSchemeVerb {
|
|
RegsInt = 250,
|
|
RegsFloat = 251,
|
|
RegsEnv = 252,
|
|
SchedAffinity = 253,
|
|
Start = 254,
|
|
Iopl = 255,
|
|
}
|
|
impl ProcSchemeVerb {
|
|
pub fn try_from_raw(verb: u8) -> Option<Self> {
|
|
Some(match verb {
|
|
250 => Self::RegsInt,
|
|
251 => Self::RegsFloat,
|
|
252 => Self::RegsEnv,
|
|
253 => Self::SchedAffinity,
|
|
254 => Self::Start,
|
|
255 => Self::Iopl,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
pub enum FileTableVerb {
|
|
Close = 1,
|
|
Dup2 = 2,
|
|
CloseCloExec = 3,
|
|
}
|
|
impl FileTableVerb {
|
|
pub fn try_from_raw(value: u8) -> Option<Self> {
|
|
Some(match value {
|
|
1 => Self::Close,
|
|
2 => Self::Dup2,
|
|
3 => Self::CloseCloExec,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
// NOT ABI-STABLE!
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(u64)]
|
|
pub enum AcpiVerb {
|
|
// copies the rsdt/xsdt to the payload buffer (the number of bytes that fit), and returns the
|
|
// rsdt/xsdt length regardless
|
|
ReadRxsdt = 1,
|
|
// no payload, just returns 0 or 1
|
|
CheckShutdown = 2,
|
|
/// Red Bear OS extension (Phase I): acpid requests the kernel
|
|
/// enter s2idle (Modern Standby / S0ix). The kernel sets
|
|
/// `S2IDLE_REQUESTED`; the idle path calls `mwait_loop()`. Read
|
|
/// payload (1 byte) returns the *previous* value of the flag.
|
|
/// Write payload is opaque (ignored by current kernel).
|
|
/// Mirrors Linux 7.1 `s2idle_enter()` in
|
|
/// `kernel/power/suspend.c:91`. Hardware-agnostic — works on
|
|
/// any platform with Modern Standby firmware (Dell, HP, Lenovo,
|
|
/// LG Gram, etc.), not just LG Gram.
|
|
EnterS2Idle = 3,
|
|
/// Red Bear OS extension (Phase I): acpid signals s2idle
|
|
/// exit. Kernel clears `S2IDLE_REQUESTED`. Read payload (1
|
|
/// byte) always returns 0. Mirrors Linux 7.1 `s2idle_wake()` in
|
|
/// `kernel/power/suspend.c:133`. Hardware-agnostic.
|
|
ExitS2Idle = 4,
|
|
/// Red Bear OS extension (Phase II.X.W): acpid writes the
|
|
/// 64-bit kernel S3 resume trampoline address to
|
|
/// FACS.xfirmware_waking_vector. The kernel's
|
|
/// `arch/x86_shared/s3_resume.rs` trampoline is at this
|
|
/// address; the platform firmware jumps to it on S3 wake.
|
|
/// Write payload: 8-byte little-endian u64 (the trampoline
|
|
/// address). Read payload: 0 (no error condition; this verb
|
|
/// is one-way). Mirrors Linux 7.1's
|
|
/// `acpi_set_firmware_waking_vector` in ACPICA.
|
|
/// Hardware-agnostic: works on any x86_64 system with
|
|
/// standard ACPI S3 support (Dell, HP, Lenovo, LG Gram 14).
|
|
SetS3WakingVector = 5,
|
|
/// Red Bear OS extension (Phase II.X.W): acpid requests
|
|
/// the kernel to enter S3. The kernel's kstop handler
|
|
/// dispatches on the "s3" string arg, dispatches on the
|
|
/// SLP_TYP byte, and does the PM1 register write. The
|
|
/// acpid has already done the AML prep (`_TTS(3)`, `_PTS(3)`,
|
|
/// `_SST(3)`) and written the trampoline address to FACS
|
|
/// via `SetS3WakingVector`. No payload needed.
|
|
/// Mirrors Linux 7.1's `enter_sleep_state` /
|
|
/// `acpi_hw_legacy_sleep` for S3.
|
|
EnterS3 = 6,
|
|
}
|
|
impl AcpiVerb {
|
|
pub const fn try_from_raw(value: u64) -> Option<Self> {
|
|
Some(match value {
|
|
1 => Self::ReadRxsdt,
|
|
2 => Self::CheckShutdown,
|
|
3 => Self::EnterS2Idle,
|
|
4 => Self::ExitS2Idle,
|
|
5 => Self::SetS3WakingVector,
|
|
6 => Self::EnterS3,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(usize)]
|
|
pub enum SchemeSocketCall {
|
|
ObtainFd = 0,
|
|
MoveFd = 1,
|
|
}
|
|
impl SchemeSocketCall {
|
|
pub fn try_from_raw(raw: usize) -> Option<Self> {
|
|
Some(match raw {
|
|
0 => Self::ObtainFd,
|
|
1 => Self::MoveFd,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
#[repr(usize)]
|
|
#[non_exhaustive]
|
|
pub enum FsCall {
|
|
Connect = 0,
|
|
}
|
|
impl FsCall {
|
|
pub fn try_from_raw(raw: usize) -> Option<Self> {
|
|
Some(match raw {
|
|
0 => Self::Connect,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
bitflags! {
|
|
pub struct PtraceFlags: u64 {
|
|
/// Stop before a syscall is handled. Send PTRACE_FLAG_IGNORE to not
|
|
/// handle the syscall.
|
|
const PTRACE_STOP_PRE_SYSCALL = 0x0000_0000_0000_0001;
|
|
/// Stop after a syscall is handled.
|
|
const PTRACE_STOP_POST_SYSCALL = 0x0000_0000_0000_0002;
|
|
/// Stop after exactly one instruction. TODO: This may not handle
|
|
/// fexec/signal boundaries. Should it?
|
|
const PTRACE_STOP_SINGLESTEP = 0x0000_0000_0000_0004;
|
|
/// Stop before a signal is handled. Send PTRACE_FLAG_IGNORE to not
|
|
/// handle signal.
|
|
const PTRACE_STOP_SIGNAL = 0x0000_0000_0000_0008;
|
|
/// Stop on a software breakpoint, such as the int3 instruction for
|
|
/// x86_64.
|
|
const PTRACE_STOP_BREAKPOINT = 0x0000_0000_0000_0010;
|
|
/// Stop just before exiting for good.
|
|
const PTRACE_STOP_EXIT = 0x0000_0000_0000_0020;
|
|
|
|
const PTRACE_STOP_MASK = 0x0000_0000_0000_00FF;
|
|
|
|
|
|
/// Sent when a child is cloned, giving you the opportunity to trace it.
|
|
/// If you don't catch this, the child is started as normal.
|
|
const PTRACE_EVENT_CLONE = 0x0000_0000_0000_0100;
|
|
|
|
/// Sent when current-addrspace is changed, allowing the tracer to reopen the memory file.
|
|
const PTRACE_EVENT_ADDRSPACE_SWITCH = 0x0000_0000_0000_0200;
|
|
|
|
const PTRACE_EVENT_MASK = 0x0000_0000_0000_0F00;
|
|
|
|
/// Special meaning, depending on the event. Usually, when fired before
|
|
/// an action, it will skip performing that action.
|
|
const PTRACE_FLAG_IGNORE = 0x0000_0000_0000_1000;
|
|
|
|
const PTRACE_FLAG_MASK = 0x0000_0000_0000_F000;
|
|
}
|
|
}
|
|
impl Deref for PtraceFlags {
|
|
type Target = [u8];
|
|
fn deref(&self) -> &Self::Target {
|
|
// Same as to_ne_bytes but in-place
|
|
unsafe {
|
|
slice::from_raw_parts(&self.bits() as *const _ as *const u8, mem::size_of::<u64>())
|
|
}
|
|
}
|
|
}
|
|
|
|
pub const SEEK_SET: usize = 0;
|
|
pub const SEEK_CUR: usize = 1;
|
|
pub const SEEK_END: usize = 2;
|
|
|
|
pub const SIGCHLD: usize = 17;
|
|
pub const SIGTSTP: usize = 20;
|
|
pub const SIGTTIN: usize = 21;
|
|
pub const SIGTTOU: usize = 22;
|
|
|
|
pub const ADDRSPACE_OP_MMAP: usize = 0;
|
|
pub const ADDRSPACE_OP_MUNMAP: usize = 1;
|
|
pub const ADDRSPACE_OP_MPROTECT: usize = 2;
|
|
pub const ADDRSPACE_OP_TRANSFER: usize = 3;
|
|
|
|
bitflags! {
|
|
pub struct MremapFlags: usize {
|
|
const FIXED = 1;
|
|
const FIXED_REPLACE = 3;
|
|
/// Alias's memory region at `old_address` to `new_address` such that both regions share
|
|
/// the same frames.
|
|
const KEEP_OLD = 1 << 2;
|
|
// TODO: MAYMOVE, DONTUNMAP
|
|
}
|
|
}
|
|
bitflags! {
|
|
pub struct RwFlags: u32 {
|
|
const NONBLOCK = 1;
|
|
const APPEND = 2;
|
|
// TODO: sync/dsync
|
|
// TODO: O_DIRECT?
|
|
}
|
|
}
|
|
bitflags! {
|
|
pub struct SigcontrolFlags: usize {
|
|
/// Prevents the kernel from jumping the context to the signal trampoline, but otherwise
|
|
/// has absolutely no effect on which signals are blocked etc. Meant to be used for
|
|
/// short-lived critical sections inside libc.
|
|
const INHIBIT_DELIVERY = 1;
|
|
}
|
|
}
|
|
bitflags! {
|
|
pub struct CallFlags: usize {
|
|
// reserved
|
|
const RSVD0 = 1 << 0;
|
|
const RSVD1 = 1 << 1;
|
|
const RSVD2 = 1 << 2;
|
|
const RSVD3 = 1 << 3;
|
|
const RSVD4 = 1 << 4;
|
|
const RSVD5 = 1 << 5;
|
|
const RSVD6 = 1 << 6;
|
|
const RSVD7 = 1 << 7;
|
|
|
|
/// Remove the fd from the caller's file table before sending the message.
|
|
const CONSUME = 1 << 8;
|
|
|
|
const WRITE = 1 << 9;
|
|
const READ = 1 << 10;
|
|
|
|
/// Indicates the request is a bulk fd passing request.
|
|
const FD = 1 << 11;
|
|
/// Flags for the fd passing request.
|
|
const FD_EXCLUSIVE = 1 << 12;
|
|
const FD_CLONE = 1 << 13;
|
|
const FD_UPPER = 1 << 14;
|
|
const FD_CLOEXEC = 1 << 15;
|
|
|
|
/// Call is a standard fs call, with metadata defined in `StdFsCallMeta`
|
|
const STD_FS = 1 << 16;
|
|
|
|
/// Call is taking multiple fds as an argument
|
|
const MULTIPLE_FDS = 1 << 17;
|
|
}
|
|
}
|
|
|
|
#[repr(u8)]
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
|
|
pub enum StdFsCallKind {
|
|
// TODO: remove old syscalls
|
|
Fchmod = 1,
|
|
Fchown = 2,
|
|
Getdents = 3,
|
|
Fstat = 4,
|
|
Fstatvfs = 5,
|
|
Fsync = 6,
|
|
Ftruncate = 7,
|
|
Futimens = 8,
|
|
// 9 reserved in fscall RFC
|
|
// Unlinkat = 10,
|
|
Relpathat = 11,
|
|
Lock = 12,
|
|
Unlock = 13,
|
|
GetLock = 14,
|
|
}
|
|
|
|
impl StdFsCallKind {
|
|
pub fn try_from_raw(raw: u8) -> Option<Self> {
|
|
use StdFsCallKind::*;
|
|
|
|
// TODO: Use a library where this match can be automated.
|
|
Some(match raw {
|
|
1 => Fchmod,
|
|
2 => Fchown,
|
|
3 => Getdents,
|
|
4 => Fstat,
|
|
5 => Fstatvfs,
|
|
6 => Fsync,
|
|
7 => Ftruncate,
|
|
8 => Futimens,
|
|
// 9 reserved in fscall RFC
|
|
// 10 => Unlinkat,
|
|
11 => Relpathat,
|
|
12 => Lock,
|
|
13 => Unlock,
|
|
14 => GetLock,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
/// The tag for the fd number in the upper file descriptor table.
|
|
pub const UPPER_FDTBL_TAG: usize = 1 << (usize::BITS - 2);
|
|
|
|
/// The identifier for registering event timeout
|
|
pub const EVENT_TIMEOUT_ID: usize = usize::MAX - 2;
|