1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
use core::ffi::{c_char, CStr};
pub use kernel_user_link::process::{
process_metadata, PriorityLevel, ProcessMetadata, SpawnFileMapping,
};
use kernel_user_link::{
call_syscall,
syscalls::{SyscallError, SYS_EXIT, SYS_PRIORITY, SYS_SPAWN, SYS_WAIT_PID},
};
/// # Safety
/// No guarantees are made about the state of the system after this function returns.
pub unsafe fn exit(code: i32) -> ! {
unsafe {
call_syscall!(
SYS_EXIT,
code as u64, // code
)
.unwrap();
}
unreachable!("exit syscall should not return")
}
/// # Safety
/// path must be a valid C string.
/// argv must be a valid C string array. ending with a null pointer.
/// File mappings must be valid and present file mappings.
/// The fds used in the file mappings must never be used again by the caller, as the ownership is
/// transferred to the child process.
pub unsafe fn spawn(
path: &CStr,
argv: &[*const c_char],
file_mappings: &[SpawnFileMapping],
) -> Result<u64, SyscallError> {
unsafe {
call_syscall!(
SYS_SPAWN,
path.as_ptr() as u64, // path
argv.as_ptr() as u64, // argv
file_mappings.as_ptr() as u64, // file_mappings
file_mappings.len() as u64 // file_mappings_len
)
}
}
/// # Safety
/// This is generally safe, it will return error if the pid is not valid, but it might wait for a long
/// time depending on the process we are waiting for.
pub unsafe fn wait_for_pid(pid: u64, block: bool) -> Result<i32, SyscallError> {
unsafe {
call_syscall!(
SYS_WAIT_PID,
pid, // pid
block as u64 // block
)
.map(|x| x as i32)
}
}
/// # Safety
/// This is generally safe, it will return error if the pid is not valid, but its marked as unsafe
/// because it's a syscall
pub unsafe fn priority(
pid: u64,
priority_level: Option<PriorityLevel>,
) -> Result<PriorityLevel, SyscallError> {
unsafe {
call_syscall!(
SYS_PRIORITY,
pid, // pid
priority_level.map_or(0, |x| x.to_u64()) // priority_level
)
// this should never fail
.map(|x| PriorityLevel::from_u64(x).unwrap())
}
}