1
0
mirror of https://github.com/rcore-os/rCore.git synced 2024-11-27 10:13:28 +04:00
rCore/kernel/src/syscall/mod.rs

408 lines
15 KiB
Rust
Raw Normal View History

2019-02-22 10:10:24 +04:00
//! System call
use alloc::{string::String, sync::Arc, vec::Vec};
2019-02-28 06:59:52 +04:00
use core::{slice, str, fmt};
2019-02-22 10:10:24 +04:00
use bitflags::bitflags;
use rcore_memory::VMError;
2019-02-22 19:48:05 +04:00
use rcore_fs::vfs::{FileType, FsError, INode, Metadata};
2019-02-22 10:10:24 +04:00
use crate::arch::interrupt::TrapFrame;
use crate::sync::Condvar;
2019-02-22 10:10:24 +04:00
use crate::process::*;
use crate::thread;
use crate::util;
use crate::arch::cpu;
2019-03-17 20:18:03 +04:00
use crate::arch::syscall::*;
2019-02-22 10:10:24 +04:00
use self::fs::*;
2019-02-22 19:48:05 +04:00
use self::mem::*;
2019-02-22 10:10:24 +04:00
use self::proc::*;
use self::time::*;
use self::net::*;
use self::misc::*;
use self::custom::*;
2019-02-22 10:10:24 +04:00
mod fs;
2019-02-22 19:48:05 +04:00
mod mem;
2019-02-22 10:10:24 +04:00
mod proc;
mod time;
mod net;
mod misc;
mod custom;
2019-02-22 10:10:24 +04:00
/// System call dispatcher
// This #[deny(unreachable_patterns)] checks if each match arm is defined
// See discussion in https://github.com/oscourse-tsinghua/rcore_plus/commit/17e644e54e494835f1a49b34b80c2c4f15ed0dbe.
#[deny(unreachable_patterns)]
2019-02-22 10:10:24 +04:00
pub fn syscall(id: usize, args: [usize; 6], tf: &mut TrapFrame) -> isize {
2019-03-10 11:23:15 +04:00
let cid = cpu::id();
let pid = {
process().pid.clone()
};
let tid = processor().tid();
if !pid.is_init() {
// we trust pid 0 process
2019-03-17 19:05:09 +04:00
debug!("{}:{}:{} syscall id {} begin", cid, pid, tid, id);
}
// use syscall numbers in Linux x86_64
// See https://filippo.io/linux-syscall-table/
// And https://fedora.juszkiewicz.com.pl/syscalls.html.
2019-03-17 19:05:09 +04:00
let ret = match id {
2019-02-22 10:10:24 +04:00
// file
2019-03-17 20:18:03 +04:00
SYS_READ => sys_read(args[0], args[1] as *mut u8, args[2]),
SYS_WRITE => sys_write(args[0], args[1] as *const u8, args[2]),
SYS_CLOSE => sys_close(args[0]),
SYS_FSTAT => sys_fstat(args[0], args[1] as *mut Stat),
SYS_LSEEK => sys_lseek(args[0], args[1] as i64, args[2] as u8),
SYS_MMAP => sys_mmap(args[0], args[1], args[2], args[3], args[4] as i32, args[5]),
SYS_MPROTECT => sys_mprotect(args[0], args[1], args[2]),
SYS_MUNMAP => sys_munmap(args[0], args[1]),
SYS_PREAD64 => sys_pread(args[0], args[1] as *mut u8, args[2], args[3]),
SYS_PWRITE64 => sys_pwrite(args[0], args[1] as *const u8, args[2], args[3]),
SYS_READV => sys_readv(args[0], args[1] as *const IoVec, args[2]),
SYS_WRITEV => sys_writev(args[0], args[1] as *const IoVec, args[2]),
SYS_SCHED_YIELD => sys_yield(),
SYS_NANOSLEEP => sys_nanosleep(args[0] as *const TimeSpec),
SYS_GETPID => sys_getpid(),
SYS_SOCKET => sys_socket(args[0], args[1], args[2]),
SYS_CONNECT => sys_connect(args[0], args[1] as *const SockAddr, args[2]),
SYS_ACCEPT => sys_accept(args[0], args[1] as *mut SockAddr, args[2] as *mut u32),
SYS_SENDTO => sys_sendto(args[0], args[1] as *const u8, args[2], args[3], args[4] as *const SockAddr, args[5]),
SYS_RECVFROM => sys_recvfrom(args[0], args[1] as *mut u8, args[2], args[3], args[4] as *mut SockAddr, args[5] as *mut u32),
// SYS_SENDMSG => sys_sendmsg(),
// SYS_RECVMSG => sys_recvmsg(),
SYS_SHUTDOWN => sys_shutdown(args[0], args[1]),
SYS_BIND => sys_bind(args[0], args[1] as *const SockAddr, args[2]),
SYS_LISTEN => sys_listen(args[0], args[1]),
SYS_GETSOCKNAME => sys_getsockname(args[0], args[1] as *mut SockAddr, args[2] as *mut u32),
SYS_GETPEERNAME => sys_getpeername(args[0], args[1] as *mut SockAddr, args[2] as *mut u32),
SYS_SETSOCKOPT => sys_setsockopt(args[0], args[1], args[2], args[3] as *const u8, args[4]),
SYS_GETSOCKOPT => sys_getsockopt(args[0], args[1], args[2], args[3] as *mut u8, args[4] as *mut u32),
SYS_CLONE => sys_clone(args[0], args[1], args[2] as *mut u32, args[3] as *mut u32, args[4], tf),
SYS_EXECVE => sys_exec(args[0] as *const u8, args[1] as *const *const u8, args[2] as *const *const u8, tf),
SYS_EXIT => sys_exit(args[0] as usize),
SYS_WAIT4 => sys_wait4(args[0] as isize, args[1] as *mut i32), // TODO: wait4
SYS_KILL => sys_kill(args[0], args[1]),
SYS_UNAME => sys_uname(args[0] as *mut u8),
SYS_FSYNC => sys_fsync(args[0]),
SYS_FDATASYNC => sys_fdatasync(args[0]),
SYS_TRUNCATE => sys_truncate(args[0] as *const u8, args[1]),
SYS_FTRUNCATE => sys_ftruncate(args[0], args[1]),
SYS_GETCWD => sys_getcwd(args[0] as *mut u8, args[1]),
SYS_CHDIR => sys_chdir(args[0] as *const u8),
SYS_GETTIMEOFDAY => sys_gettimeofday(args[0] as *mut TimeVal, args[1] as *const u8),
// SYS_GETRLIMIT => sys_getrlimit(),
SYS_GETRUSAGE => sys_getrusage(args[0], args[1] as *mut RUsage),
SYS_SYSINFO => sys_sysinfo(args[0] as *mut SysInfo),
SYS_GETPPID => sys_getppid(),
SYS_SETPRIORITY => sys_set_priority(args[0]),
// SYS_SETRLIMIT => sys_setrlimit(),
SYS_SYNC => sys_sync(),
2019-03-17 20:18:03 +04:00
SYS_REBOOT => sys_reboot(args[0] as u32, args[1] as u32, args[2] as u32, args[3] as *const u8),
SYS_GETTID => sys_gettid(),
SYS_FUTEX => sys_futex(args[0], args[1] as u32, args[2] as i32, args[3] as *const TimeSpec),
SYS_SCHED_GETAFFINITY => sys_sched_getaffinity(args[0], args[1], args[2] as *mut u32),
SYS_GETDENTS64 => sys_getdents64(args[0], args[1] as *mut LinuxDirent64, args[2]),
SYS_CLOCK_GETTIME => sys_clock_gettime(args[0], args[1] as *mut TimeSpec),
SYS_EXIT_GROUP => sys_exit_group(args[0]),
SYS_OPENAT => sys_open(args[1] as *const u8, args[2], args[3]), // TODO: handle `dfd`
SYS_MKDIRAT => sys_mkdir(args[1] as *const u8, args[2]), // TODO: handle `dfd`
// SYS_MKNODAT => sys_mknod(),
SYS_NEWFSTATAT => sys_stat(args[1] as *const u8, args[2] as *mut Stat), // TODO: handle `dfd`, `flag`
SYS_UNLINKAT => sys_unlink(args[1] as *const u8), // TODO: handle `dfd`, `flag`
SYS_RENAMEAT => sys_rename(args[1] as *const u8, args[3] as *const u8), // TODO: handle `olddfd`, `newdfd`
SYS_LINKAT => sys_link(args[1] as *const u8, args[3] as *const u8), // TODO: handle `olddfd`, `newdfd`, `flags`
SYS_FACCESSAT => sys_access(args[1] as *const u8, args[2]), // TODO: handle `dfd`
SYS_ACCEPT4 => sys_accept(args[0], args[1] as *mut SockAddr, args[2] as *mut u32), // use accept for accept4
SYS_DUP3 => sys_dup2(args[0], args[1]), // TODO: handle `flags`
SYS_PIPE2 => sys_pipe(args[0] as *mut u32), // TODO: handle `flags`
// custom temporary syscall
2019-03-17 20:18:03 +04:00
SYS_MAP_PCI_DEVICE => sys_map_pci_device(args[0], args[1]),
SYS_GET_PADDR => sys_get_paddr(args[0] as *const u64, args[1] as *mut u64, args[2]),
2019-02-22 10:10:24 +04:00
// for musl: empty impl
2019-03-17 20:18:03 +04:00
SYS_BRK => {
2019-02-22 19:48:05 +04:00
warn!("sys_brk is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_RT_SIGACTION => {
2019-02-22 19:48:05 +04:00
warn!("sys_sigaction is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_RT_SIGPROCMASK => {
2019-02-22 19:48:05 +04:00
warn!("sys_sigprocmask is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_IOCTL => {
warn!("sys_ioctl is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_MADVISE => {
warn!("sys_madvise is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_SETITIMER => {
warn!("sys_setitimer is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_FCNTL => {
warn!("sys_fcntl is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_UMASK => {
2019-03-02 17:25:36 +04:00
warn!("sys_umask is unimplemented");
Ok(0o777)
}
2019-03-17 20:18:03 +04:00
SYS_GETUID => {
warn!("sys_getuid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_GETGID => {
warn!("sys_getgid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_SETUID => {
warn!("sys_setuid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_GETEUID => {
warn!("sys_geteuid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_GETEGID => {
warn!("sys_getegid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_SETSID => {
warn!("sys_setsid is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_SIGALTSTACK => {
2019-02-22 19:48:05 +04:00
warn!("sys_sigaltstack is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_SET_TID_ADDRESS => {
2019-02-22 10:10:24 +04:00
warn!("sys_set_tid_address is unimplemented");
2019-03-08 15:04:39 +04:00
Ok(thread::current().id())
2019-02-22 10:10:24 +04:00
}
2019-03-17 20:18:03 +04:00
SYS_UTIMENSAT => {
2019-03-08 11:54:03 +04:00
warn!("sys_utimensat is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_EPOLL_CREATE1 => {
warn!("sys_epoll_create1 is unimplemented");
Err(SysError::ENOSYS)
}
2019-03-17 20:18:03 +04:00
SYS_PRLIMIT64 => {
warn!("sys_prlimit64 is unimplemented");
Ok(0)
}
2019-02-22 10:10:24 +04:00
_ => {
2019-03-17 19:05:09 +04:00
#[cfg(target_arch = "x86_64")]
let x86_64_ret = x86_64_syscall(id, args, tf);
#[cfg(not(target_arch = "x86_64"))]
let x86_64_ret = None;
if let Some(ret) = x86_64_ret {
ret
} else {
error!("unknown syscall id: {}, args: {:x?}", id, args);
crate::trap::error(tf);
}
2019-02-22 10:10:24 +04:00
}
};
if !pid.is_init() {
// we trust pid 0 process
2019-03-17 19:05:09 +04:00
debug!("{}:{}:{} syscall id {} ret with {:x?}", cid, pid, tid, id, ret);
}
2019-02-22 10:10:24 +04:00
match ret {
2019-03-08 15:04:39 +04:00
Ok(code) => code as isize,
2019-02-22 10:10:24 +04:00
Err(err) => -(err as isize),
}
}
2019-03-17 19:05:09 +04:00
#[cfg(target_arch = "x86_64")]
fn x86_64_syscall(id: usize, args: [usize; 6], tf: &mut TrapFrame) -> Option<SysResult> {
let ret = match id {
2019-03-17 20:18:03 +04:00
SYS_OPEN => sys_open(args[0] as *const u8, args[1], args[2]),
SYS_STAT => sys_stat(args[0] as *const u8, args[1] as *mut Stat),
SYS_LSTAT => sys_lstat(args[0] as *const u8, args[1] as *mut Stat),
SYS_POLL => sys_poll(args[0] as *mut PollFd, args[1], args[2]),
SYS_ACCESS => sys_access(args[0] as *const u8, args[1]),
SYS_PIPE => sys_pipe(args[0] as *mut u32),
SYS_SELECT => sys_select(args[0], args[1] as *mut u32, args[2] as *mut u32, args[3] as *mut u32, args[4] as *const TimeVal),
SYS_DUP2 => sys_dup2(args[0], args[1]),
// SYS_PAUSE => sys_pause(),
2019-03-17 19:05:09 +04:00
SYS_FORK => sys_fork(tf),
// use fork for vfork
2019-03-17 20:18:03 +04:00
SYS_VFORK => sys_fork(tf),
SYS_RENAME => sys_rename(args[0] as *const u8, args[1] as *const u8),
SYS_MKDIR => sys_mkdir(args[0] as *const u8, args[1]),
SYS_RMDIR => sys_rmdir(args[0] as *const u8),
2019-03-17 20:18:03 +04:00
SYS_LINK => sys_link(args[0] as *const u8, args[1] as *const u8),
SYS_UNLINK => sys_unlink(args[0] as *const u8),
SYS_ARCH_PRCTL => sys_arch_prctl(args[0] as i32, args[1], tf),
SYS_TIME => sys_time(args[0] as *mut u64),
SYS_ALARM => {
2019-03-17 19:05:09 +04:00
warn!("sys_alarm is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_READLINK => {
2019-03-17 19:05:09 +04:00
warn!("sys_readlink is unimplemented");
Err(SysError::ENOENT)
}
2019-03-17 20:18:03 +04:00
SYS_CHOWN => {
2019-03-17 19:05:09 +04:00
warn!("sys_chown is unimplemented");
Ok(0)
}
2019-03-17 20:18:03 +04:00
SYS_EPOLL_CREATE => {
2019-03-17 19:05:09 +04:00
warn!("sys_epoll_create is unimplemented");
Err(SysError::ENOSYS)
}
_ => {
return None;
}
};
Some(ret)
}
2019-03-08 15:04:39 +04:00
pub type SysResult = Result<usize, SysError>;
2019-02-22 10:10:24 +04:00
2019-02-28 06:59:52 +04:00
#[allow(dead_code)]
2019-02-22 10:10:24 +04:00
#[repr(isize)]
#[derive(Debug)]
pub enum SysError {
2019-02-28 06:59:52 +04:00
EUNDEF = 0,
EPERM = 1,
ENOENT = 2,
ESRCH = 3,
EINTR = 4,
EIO = 5,
ENXIO = 6,
E2BIG = 7,
ENOEXEC = 8,
EBADF = 9,
ECHILD = 10,
EAGAIN = 11,
ENOMEM = 12,
EACCES = 13,
EFAULT = 14,
ENOTBLK = 15,
EBUSY = 16,
EEXIST = 17,
EXDEV = 18,
ENODEV = 19,
ENOTDIR = 20,
EISDIR = 21,
EINVAL = 22,
ENFILE = 23,
EMFILE = 24,
ENOTTY = 25,
ETXTBSY = 26,
EFBIG = 27,
ENOSPC = 28,
ESPIPE = 29,
EROFS = 30,
EMLINK = 31,
EPIPE = 32,
EDOM = 33,
ERANGE = 34,
EDEADLK = 35,
ENAMETOOLONG = 36,
ENOLCK = 37,
ENOSYS = 38,
ENOTEMPTY = 39,
ENOTSOCK = 80,
ENOPROTOOPT = 92,
EPFNOSUPPORT = 96,
EAFNOSUPPORT = 97,
ENOBUFS = 105,
EISCONN = 106,
ENOTCONN = 107,
ECONNREFUSED = 111,
2019-02-28 06:59:52 +04:00
}
2019-02-22 10:10:24 +04:00
2019-02-28 06:59:52 +04:00
#[allow(non_snake_case)]
impl fmt::Display for SysError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
2019-03-16 11:44:57 +04:00
use self::SysError::*;
2019-02-28 06:59:52 +04:00
write!(f, "{}",
match self {
EPERM => "Operation not permitted",
ENOENT => "No such file or directory",
ESRCH => "No such process",
EINTR => "Interrupted system call",
EIO => "I/O error",
ENXIO => "No such device or address",
E2BIG => "Argument list too long",
ENOEXEC => "Exec format error",
EBADF => "Bad file number",
ECHILD => "No child processes",
EAGAIN => "Try again",
ENOMEM => "Out of memory",
EACCES => "Permission denied",
EFAULT => "Bad address",
ENOTBLK => "Block device required",
EBUSY => "Device or resource busy",
EEXIST => "File exists",
EXDEV => "Cross-device link",
ENODEV => "No such device",
ENOTDIR => "Not a directory",
EISDIR => "Is a directory",
EINVAL => "Invalid argument",
ENFILE => "File table overflow",
EMFILE => "Too many open files",
ENOTTY => "Not a typewriter",
ETXTBSY => "Text file busy",
EFBIG => "File too large",
ENOSPC => "No space left on device",
ESPIPE => "Illegal seek",
EROFS => "Read-only file system",
EMLINK => "Too many links",
EPIPE => "Broken pipe",
EDOM => "Math argument out of domain of func",
ERANGE => "Math result not representable",
EDEADLK => "Resource deadlock would occur",
ENAMETOOLONG => "File name too long",
ENOLCK => "No record locks available",
ENOSYS => "Function not implemented",
ENOTEMPTY => "Directory not empty",
2019-03-10 11:23:15 +04:00
ENOTSOCK => "Socket operation on non-socket",
ENOPROTOOPT => "Protocol not available",
EPFNOSUPPORT => "Protocol family not supported",
EAFNOSUPPORT => "Address family not supported by protocol",
ENOBUFS => "No buffer space available",
EISCONN => "Transport endpoint is already connected",
ENOTCONN => "Transport endpoint is not connected",
ECONNREFUSED => "Connection refused",
2019-02-28 06:59:52 +04:00
_ => "Unknown error",
},
)
}
2019-02-22 10:10:24 +04:00
}
impl From<VMError> for SysError {
fn from(_: VMError) -> Self {
SysError::EFAULT
}
}
const SPIN_WAIT_TIMES: usize = 100;
2019-03-19 14:35:40 +04:00
pub fn spin_and_wait<T>(condvars: &[&Condvar], mut action: impl FnMut() -> Option<T>) -> T {
for i in 0..SPIN_WAIT_TIMES {
if let Some(result) = action() {
return result;
}
}
loop {
if let Some(result) = action() {
return result;
}
Condvar::wait_any(&condvars);
}
}