forked from Bananymous/banan-os
256 lines
7.5 KiB
C++
256 lines
7.5 KiB
C++
#include <kernel/Debug.h>
|
|
#include <kernel/InterruptStack.h>
|
|
#include <kernel/Process.h>
|
|
#include <kernel/Scheduler.h>
|
|
#include <kernel/Syscall.h>
|
|
|
|
#include <termios.h>
|
|
|
|
namespace Kernel
|
|
{
|
|
|
|
extern "C" long sys_fork(uintptr_t rsp, uintptr_t rip)
|
|
{
|
|
auto ret = Process::current().sys_fork(rsp, rip);
|
|
if (ret.is_error())
|
|
return -ret.error().get_error_code();
|
|
return ret.value();
|
|
}
|
|
|
|
extern "C" long sys_fork_trampoline();
|
|
|
|
extern "C" long cpp_syscall_handler(int syscall, uintptr_t arg1, uintptr_t arg2, uintptr_t arg3, uintptr_t arg4, uintptr_t arg5, InterruptStack& interrupt_stack)
|
|
{
|
|
ASSERT((interrupt_stack.cs & 0b11) == 0b11);
|
|
|
|
Thread::current().set_return_rsp(interrupt_stack.rsp);
|
|
Thread::current().set_return_rip(interrupt_stack.rip);
|
|
|
|
asm volatile("sti");
|
|
|
|
(void)arg1;
|
|
(void)arg2;
|
|
(void)arg3;
|
|
(void)arg4;
|
|
(void)arg5;
|
|
(void)interrupt_stack;
|
|
|
|
BAN::ErrorOr<long> ret = BAN::Error::from_errno(ENOSYS);
|
|
|
|
switch (syscall)
|
|
{
|
|
case SYS_EXIT:
|
|
ret = Process::current().sys_exit((int)arg1);
|
|
break;
|
|
case SYS_READ:
|
|
ret = Process::current().sys_read((int)arg1, (void*)arg2, (size_t)arg3);
|
|
break;
|
|
case SYS_WRITE:
|
|
ret = Process::current().sys_write((int)arg1, (const void*)arg2, (size_t)arg3);
|
|
break;
|
|
case SYS_TERMID:
|
|
ret = Process::current().sys_termid((char*)arg1);
|
|
break;
|
|
case SYS_CLOSE:
|
|
ret = Process::current().sys_close((int)arg1);
|
|
break;
|
|
case SYS_OPEN:
|
|
ret = Process::current().sys_open((const char*)arg1, (int)arg2, (mode_t)arg3);
|
|
break;
|
|
case SYS_OPENAT:
|
|
ret = Process::current().sys_openat((int)arg1, (const char*)arg2, (int)arg3, (mode_t)arg4);
|
|
break;
|
|
case SYS_SEEK:
|
|
ret = Process::current().sys_seek((int)arg1, (long)arg2, (int)arg3);
|
|
break;
|
|
case SYS_TELL:
|
|
ret = Process::current().sys_tell((int)arg1);
|
|
break;
|
|
case SYS_GET_TERMIOS:
|
|
ret = Process::current().sys_gettermios((::termios*)arg1);
|
|
break;
|
|
case SYS_SET_TERMIOS:
|
|
ret = Process::current().sys_settermios((const ::termios*)arg1);
|
|
break;
|
|
case SYS_FORK:
|
|
ret = sys_fork_trampoline();
|
|
break;
|
|
case SYS_EXEC:
|
|
ret = Process::current().sys_exec((const char*)arg1, (const char* const*)arg2, (const char* const*)arg3);
|
|
break;
|
|
case SYS_SLEEP:
|
|
ret = Process::current().sys_sleep((unsigned int)arg1);
|
|
break;
|
|
case SYS_WAIT:
|
|
ret = Process::current().sys_wait((pid_t)arg1, (int*)arg2, (int)arg3);
|
|
break;
|
|
case SYS_FSTAT:
|
|
ret = Process::current().sys_fstat((int)arg1, (struct stat*)arg2);
|
|
break;
|
|
case SYS_READ_DIR_ENTRIES:
|
|
ret = Process::current().sys_read_dir_entries((int)arg1, (API::DirectoryEntryList*)arg2, (size_t)arg3);
|
|
break;
|
|
case SYS_SET_UID:
|
|
ret = Process::current().sys_setuid((uid_t)arg1);
|
|
break;
|
|
case SYS_SET_GID:
|
|
ret = Process::current().sys_setgid((gid_t)arg1);
|
|
break;
|
|
case SYS_SET_EUID:
|
|
ret = Process::current().sys_seteuid((uid_t)arg1);
|
|
break;
|
|
case SYS_SET_EGID:
|
|
ret = Process::current().sys_setegid((gid_t)arg1);
|
|
break;
|
|
case SYS_SET_REUID:
|
|
ret = Process::current().sys_setreuid((uid_t)arg1, (uid_t)arg2);
|
|
break;
|
|
case SYS_SET_REGID:
|
|
ret = Process::current().sys_setregid((gid_t)arg1, (gid_t)arg2);
|
|
break;
|
|
case SYS_GET_UID:
|
|
ret = Process::current().sys_getuid();
|
|
break;
|
|
case SYS_GET_GID:
|
|
ret = Process::current().sys_getgid();
|
|
break;
|
|
case SYS_GET_EUID:
|
|
ret = Process::current().sys_geteuid();
|
|
break;
|
|
case SYS_GET_EGID:
|
|
ret = Process::current().sys_getegid();
|
|
break;
|
|
case SYS_GET_PWD:
|
|
ret = Process::current().sys_getpwd((char*)arg1, (size_t)arg2);
|
|
break;
|
|
case SYS_SET_PWD:
|
|
ret = Process::current().sys_setpwd((const char*)arg1);
|
|
break;
|
|
case SYS_CLOCK_GETTIME:
|
|
ret = Process::current().sys_clock_gettime((clockid_t)arg1, (timespec*)arg2);
|
|
break;
|
|
case SYS_PIPE:
|
|
ret = Process::current().sys_pipe((int*)arg1);
|
|
break;
|
|
case SYS_DUP:
|
|
ret = Process::current().sys_dup((int)arg1);
|
|
break;
|
|
case SYS_DUP2:
|
|
ret = Process::current().sys_dup2((int)arg1, (int)arg2);
|
|
break;
|
|
case SYS_KILL:
|
|
ret = Process::current().sys_kill((pid_t)arg1, (int)arg2);
|
|
break;
|
|
case SYS_SIGNAL:
|
|
ret = Process::current().sys_signal((int)arg1, (void (*)(int))arg2);
|
|
break;
|
|
case SYS_TCSETPGRP:
|
|
ret = Process::current().sys_tcsetpgrp((int)arg1, (pid_t)arg2);
|
|
break;
|
|
case SYS_GET_PID:
|
|
ret = Process::current().pid();
|
|
break;
|
|
case SYS_GET_PGID:
|
|
ret = Process::current().sys_getpgid((pid_t)arg1);
|
|
break;
|
|
case SYS_SET_PGID:
|
|
ret = Process::current().sys_setpgid((pid_t)arg1, (pid_t)arg2);
|
|
break;
|
|
case SYS_FCNTL:
|
|
ret = Process::current().sys_fcntl((int)arg1, (int)arg2, (int)arg3);
|
|
break;
|
|
case SYS_NANOSLEEP:
|
|
ret = Process::current().sys_nanosleep((const timespec*)arg1, (timespec*)arg2);
|
|
break;
|
|
case SYS_FSTATAT:
|
|
ret = Process::current().sys_fstatat((int)arg1, (const char*)arg2, (struct stat*)arg3, (int)arg4);
|
|
break;
|
|
case SYS_STAT:
|
|
ret = Process::current().sys_stat((const char*)arg1, (struct stat*)arg2, (int)arg3);
|
|
break;
|
|
case SYS_SYNC:
|
|
ret = Process::current().sys_sync((bool)arg1);
|
|
break;
|
|
case SYS_MMAP:
|
|
ret = Process::current().sys_mmap((const sys_mmap_t*)arg1);
|
|
break;
|
|
case SYS_MUNMAP:
|
|
ret = Process::current().sys_munmap((void*)arg1, (size_t)arg2);
|
|
break;
|
|
case SYS_TTY_CTRL:
|
|
ret = Process::current().sys_tty_ctrl((int)arg1, (int)arg2, (int)arg3);
|
|
break;
|
|
case SYS_POWEROFF:
|
|
ret = Process::current().sys_poweroff((int)arg1);
|
|
break;
|
|
case SYS_CHMOD:
|
|
ret = Process::current().sys_chmod((const char*)arg1, (mode_t)arg2);
|
|
break;
|
|
case SYS_CREATE:
|
|
ret = Process::current().sys_create((const char*)arg1, (mode_t)arg2);
|
|
break;
|
|
case SYS_CREATE_DIR:
|
|
ret = Process::current().sys_create_dir((const char*)arg1, (mode_t)arg2);
|
|
break;
|
|
case SYS_UNLINK:
|
|
ret = Process::current().sys_unlink((const char*)arg1);
|
|
break;
|
|
case SYS_READLINK:
|
|
ret = Process::current().sys_readlink((const char*)arg1, (char*)arg2, (size_t)arg3);
|
|
break;
|
|
case SYS_READLINKAT:
|
|
ret = Process::current().sys_readlinkat((int)arg1, (const char*)arg2, (char*)arg3, (size_t)arg4);
|
|
break;
|
|
case SYS_MSYNC:
|
|
ret = Process::current().sys_msync((void*)arg1, (size_t)arg2, (int)arg3);
|
|
break;
|
|
case SYS_PREAD:
|
|
ret = Process::current().sys_pread((int)arg1, (void*)arg2, (size_t)arg3, (off_t)arg4);
|
|
break;
|
|
case SYS_CHOWN:
|
|
ret = Process::current().sys_chown((const char*)arg1, (uid_t)arg2, (gid_t)arg3);
|
|
break;
|
|
case SYS_LOAD_KEYMAP:
|
|
ret = Process::current().sys_load_keymap((const char*)arg1);
|
|
break;
|
|
case SYS_SOCKET:
|
|
ret = Process::current().sys_socket((int)arg1, (int)arg2, (int)arg3);
|
|
break;
|
|
case SYS_BIND:
|
|
ret = Process::current().sys_bind((int)arg1, (const sockaddr*)arg2, (socklen_t)arg3);
|
|
break;
|
|
case SYS_SENDTO:
|
|
ret = Process::current().sys_sendto((const sys_sendto_t*)arg1);
|
|
break;
|
|
case SYS_RECVFROM:
|
|
ret = Process::current().sys_recvfrom((sys_recvfrom_t*)arg1);
|
|
break;
|
|
case SYS_IOCTL:
|
|
ret = Process::current().sys_ioctl((int)arg1, (int)arg2, (void*)arg3);
|
|
break;
|
|
default:
|
|
dwarnln("Unknown syscall {}", syscall);
|
|
break;
|
|
}
|
|
|
|
asm volatile("cli");
|
|
|
|
if (ret.is_error() && ret.error().get_error_code() == ENOTSUP)
|
|
dprintln("ENOTSUP {}", syscall);
|
|
|
|
if (ret.is_error() && ret.error().is_kernel_error())
|
|
Kernel::panic("Kernel error while returning to userspace {}", ret.error());
|
|
|
|
auto& current_thread = Thread::current();
|
|
if (current_thread.can_add_signal_to_execute())
|
|
current_thread.handle_signal();
|
|
|
|
ASSERT(Kernel::Thread::current().state() == Kernel::Thread::State::Executing);
|
|
|
|
if (ret.is_error())
|
|
return -ret.error().get_error_code();
|
|
return ret.value();
|
|
}
|
|
|
|
}
|