mirror of
https://codeberg.org/ziglang/zig.git
synced 2025-12-06 13:54:21 +00:00
Linux: Add syscall bindings, enhance documentation.
- Add syscall bindings/structures for the `futex2` family. The documentation is taken from the syscall definitions. - Add documnentation for the `cachestat` bindings and structures. Taken from work I did in Cosmopolitian libc. - Add binding for `map_shadow_stack`. No documentation for this one, since the kernel devs didn't bother to do it ¯\_(ツ)_/¯.
This commit is contained in:
parent
cca021c211
commit
3200fae9c5
1 changed files with 157 additions and 0 deletions
|
|
@ -323,6 +323,113 @@ pub fn futex_wake(uaddr: *const i32, futex_op: u32, val: i32) usize {
|
|||
return syscall3(.futex, @intFromPtr(uaddr), futex_op, @as(u32, @bitCast(val)));
|
||||
}
|
||||
|
||||
/// Given an array of `futex_waitv`, wait on each uaddr.
|
||||
/// The thread wakes if a futex_wake() is performed at any uaddr.
|
||||
/// The syscall returns immediately if any waiter has *uaddr != val.
|
||||
/// timeout is an optional timeout value for the operation.
|
||||
/// Each waiter has individual flags.
|
||||
/// The `flags` argument for the syscall should be used solely for specifying
|
||||
/// the timeout as realtime, if needed.
|
||||
/// Flags for private futexes, sizes, etc. should be used on the
|
||||
/// individual flags of each waiter.
|
||||
///
|
||||
/// Returns the array index of one of the woken futexes.
|
||||
/// No further information is provided: any number of other futexes may also
|
||||
/// have been woken by the same event, and if more than one futex was woken,
|
||||
/// the retrned index may refer to any one of them.
|
||||
/// (It is not necessaryily the futex with the smallest index, nor the one
|
||||
/// most recently woken, nor...)
|
||||
pub fn futex2_waitv(
|
||||
/// List of futexes to wait on.
|
||||
waiters: [*]futex_waitv,
|
||||
/// Length of `waiters`.
|
||||
nr_futexes: u32,
|
||||
/// Flag for timeout (monotonic/realtime).
|
||||
flags: u32,
|
||||
/// Optional absolute timeout.
|
||||
timeout: ?*const timespec,
|
||||
/// Clock to be used for the timeout, realtime or monotonic.
|
||||
clockid: i32,
|
||||
) usize {
|
||||
return syscall6(
|
||||
.futex_waitv,
|
||||
@intFromPtr(waiters),
|
||||
nr_futexes,
|
||||
flags,
|
||||
@intFromPtr(timeout),
|
||||
@bitCast(@as(isize, clockid)),
|
||||
);
|
||||
}
|
||||
|
||||
/// Wait on a futex.
|
||||
/// Identical to `FUTEX.WAIT`, except it is part of the futex2 family of calls.
|
||||
pub fn futex2_wait(
|
||||
/// Address of the futex to wait on.
|
||||
uaddr: *const anyopaque,
|
||||
/// Value of `uaddr`.
|
||||
val: usize,
|
||||
/// Bitmask.
|
||||
mask: usize,
|
||||
/// `FUTEX2` flags.
|
||||
flags: u32,
|
||||
/// Optional absolute timeout.
|
||||
timeout: *const timespec,
|
||||
/// Clock to be used for the timeout, realtime or monotonic.
|
||||
clockid: i32,
|
||||
) usize {
|
||||
return syscall6(
|
||||
.futex_wait,
|
||||
@intFromPtr(uaddr),
|
||||
val,
|
||||
mask,
|
||||
flags,
|
||||
@intFromPtr(timeout),
|
||||
@bitCast(@as(isize, clockid)),
|
||||
);
|
||||
}
|
||||
|
||||
/// Wake a number of futexes.
|
||||
/// Identical to `FUTEX.WAKE`, except it is part of the futex2 family of calls.
|
||||
pub fn futex2_wake(
|
||||
/// Address of the futex(es) to wake.
|
||||
uaddr: [*]const anyopaque,
|
||||
/// Bitmask
|
||||
mask: usize,
|
||||
/// Number of the futexes to wake.
|
||||
nr: i32,
|
||||
/// `FUTEX2` flags.
|
||||
flags: u32,
|
||||
) usize {
|
||||
return syscall4(
|
||||
.futex_wake,
|
||||
@intFromPtr(uaddr),
|
||||
mask,
|
||||
@bitCast(@as(isize, nr)),
|
||||
flags,
|
||||
);
|
||||
}
|
||||
|
||||
/// Requeue a waiter from one futex to another.
|
||||
/// Identical to `FUTEX.CMP_REQUEUE`, except it is part of the futex2 family of calls.
|
||||
pub fn futex2_requeue(
|
||||
/// Array describing the source and destination futex.
|
||||
waiters: [*]futex_waitv,
|
||||
/// Unsed.
|
||||
flags: u32,
|
||||
/// Number of futexes to wake.
|
||||
nr_wake: i32,
|
||||
/// Number of futexes to requeue.
|
||||
nr_requeue: i32,
|
||||
) usize {
|
||||
return syscall4(
|
||||
.futex_requeue,
|
||||
@intFromPtr(waiters),
|
||||
flags,
|
||||
@bitCast(@as(isize, nr_wake)),
|
||||
@bitCast(@as(isize, nr_requeue)),
|
||||
);
|
||||
}
|
||||
|
||||
pub fn getcwd(buf: [*]u8, size: usize) usize {
|
||||
return syscall2(.getcwd, @intFromPtr(buf), size);
|
||||
}
|
||||
|
|
@ -1901,10 +2008,17 @@ pub fn ptrace(
|
|||
);
|
||||
}
|
||||
|
||||
/// Query the page cache statistics of a file.
|
||||
pub fn cachestat(
|
||||
/// The open file descriptor to retrieve statistics from.
|
||||
fd: fd_t,
|
||||
/// The byte range in `fd` to query.
|
||||
/// When `len > 0`, the range is `[off..off + len]`.
|
||||
/// When `len` == 0, the range is from `off` to the end of `fd`.
|
||||
cstat_range: *const cache_stat_range,
|
||||
/// The structure where page cache statistics are stored.
|
||||
cstat: *cache_stat,
|
||||
/// Currently unused, and must be set to `0`.
|
||||
flags: u32,
|
||||
) usize {
|
||||
return syscall4(
|
||||
|
|
@ -1916,6 +2030,10 @@ pub fn cachestat(
|
|||
);
|
||||
}
|
||||
|
||||
pub fn map_shadow_stack(addr: u64, size: u64, flags: u32) usize {
|
||||
return syscall3(.map_shadow_stack, addr, size, flags);
|
||||
}
|
||||
|
||||
pub const E = switch (native_arch) {
|
||||
.mips, .mipsel => @import("linux/errno/mips.zig").E,
|
||||
.sparc, .sparcel, .sparc64 => @import("linux/errno/sparc.zig").E,
|
||||
|
|
@ -2016,6 +2134,19 @@ pub const FUTEX = struct {
|
|||
pub const PRIVATE_FLAG = 128;
|
||||
|
||||
pub const CLOCK_REALTIME = 256;
|
||||
|
||||
/// Max numbers of elements in a `futex_waitv` array.
|
||||
pub const WAITV_MAX = 128;
|
||||
};
|
||||
|
||||
pub const FUTEX2 = struct {
|
||||
pub const SIZE_U8 = 0x00;
|
||||
pub const SIZE_U16 = 0x01;
|
||||
pub const SIZE_U32 = 0x02;
|
||||
pub const SIZE_U64 = 0x03;
|
||||
pub const NUMA = 0x04;
|
||||
|
||||
pub const PRIVATE = FUTEX.PRIVATE_FLAG;
|
||||
};
|
||||
|
||||
pub const PROT = struct {
|
||||
|
|
@ -6100,15 +6231,41 @@ pub const PTRACE = struct {
|
|||
pub const GET_SYSCALL_INFO = 0x420e;
|
||||
};
|
||||
|
||||
/// A waiter for vectorized wait.
|
||||
pub const futex_waitv = extern struct {
|
||||
// Expected value at uaddr
|
||||
val: u64,
|
||||
/// User address to wait on.
|
||||
uaddr: u64,
|
||||
/// Flags for this waiter.
|
||||
flags: u32,
|
||||
/// Reserved memeber to preserve alignment.
|
||||
/// Should be 0.
|
||||
__reserved: u32,
|
||||
};
|
||||
|
||||
pub const cache_stat_range = extern struct {
|
||||
off: u64,
|
||||
len: u64,
|
||||
};
|
||||
|
||||
pub const cache_stat = extern struct {
|
||||
/// Number of cached pages.
|
||||
cache: u64,
|
||||
/// Number of dirty pages.
|
||||
dirty: u64,
|
||||
/// Number of pages marked for writeback.
|
||||
writeback: u64,
|
||||
/// Number of pages evicted from the cache.
|
||||
evicted: u64,
|
||||
/// Number of recently evicted pages.
|
||||
/// A page is recently evicted if its last eviction was recent enough that its
|
||||
/// reentry to the cache would indicate that it is actively being used by the
|
||||
/// system, and that there is memory pressure on the system.
|
||||
recently_evicted: u64,
|
||||
};
|
||||
|
||||
pub const SHADOW_STACK = struct {
|
||||
/// Set up a restore token in the shadow stack.
|
||||
pub const SET_TOKEN: u64 = 1 << 0;
|
||||
};
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue