2024-02-12 21:47:40 +02:00
|
|
|
#include <BAN/Bitcast.h>
|
2023-03-13 15:32:46 +02:00
|
|
|
#include <kernel/Debug.h>
|
2023-07-21 11:01:19 +03:00
|
|
|
#include <kernel/InterruptStack.h>
|
2023-04-12 17:51:36 +03:00
|
|
|
#include <kernel/Process.h>
|
2023-07-31 22:28:18 +03:00
|
|
|
#include <kernel/Scheduler.h>
|
2023-03-13 15:32:46 +02:00
|
|
|
#include <kernel/Syscall.h>
|
|
|
|
|
2023-05-16 19:22:10 +03:00
|
|
|
#include <termios.h>
|
|
|
|
|
2024-08-01 17:20:20 +03:00
|
|
|
#define DUMP_ALL_SYSCALLS 0
|
|
|
|
|
2023-03-13 15:32:46 +02:00
|
|
|
namespace Kernel
|
|
|
|
{
|
|
|
|
|
2024-03-22 14:48:33 +02:00
|
|
|
extern "C" long sys_fork(uintptr_t sp, uintptr_t ip)
|
2023-05-28 18:08:26 +03:00
|
|
|
{
|
2024-03-22 14:48:33 +02:00
|
|
|
auto ret = Process::current().sys_fork(sp, ip);
|
2023-06-04 17:57:24 +03:00
|
|
|
if (ret.is_error())
|
|
|
|
return -ret.error().get_error_code();
|
|
|
|
return ret.value();
|
|
|
|
}
|
|
|
|
|
2023-05-28 18:08:26 +03:00
|
|
|
extern "C" long sys_fork_trampoline();
|
|
|
|
|
2024-02-12 21:47:40 +02:00
|
|
|
using SyscallHandler = BAN::ErrorOr<long> (Process::*)(uintptr_t, uintptr_t, uintptr_t, uintptr_t, uintptr_t);
|
|
|
|
|
|
|
|
static const SyscallHandler s_syscall_handlers[] = {
|
|
|
|
#define O(enum, name) BAN::bit_cast<SyscallHandler>(&Process::sys_ ## name),
|
|
|
|
__SYSCALL_LIST(O)
|
|
|
|
#undef O
|
|
|
|
};
|
|
|
|
|
2024-08-01 17:20:20 +03:00
|
|
|
static constexpr const char* s_syscall_names[] {
|
|
|
|
#define O(enum, name) #enum,
|
|
|
|
__SYSCALL_LIST(O)
|
|
|
|
#undef O
|
|
|
|
};
|
|
|
|
|
2024-04-03 02:23:23 +03:00
|
|
|
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)
|
2023-03-13 15:32:46 +02:00
|
|
|
{
|
2024-04-03 02:23:23 +03:00
|
|
|
ASSERT(GDT::is_user_segment(interrupt_stack->cs));
|
2023-08-01 14:23:50 +03:00
|
|
|
|
2023-04-21 10:40:24 +03:00
|
|
|
asm volatile("sti");
|
|
|
|
|
2023-06-12 14:16:48 +03:00
|
|
|
BAN::ErrorOr<long> ret = BAN::Error::from_errno(ENOSYS);
|
|
|
|
|
2024-08-22 14:17:16 +03:00
|
|
|
const char* process_path = nullptr;
|
|
|
|
if (Process::current().userspace_info().argc > 0 && Process::current().userspace_info().argv)
|
|
|
|
process_path = Process::current().userspace_info().argv[0];
|
|
|
|
if (process_path == nullptr)
|
|
|
|
process_path = "<null>";
|
|
|
|
|
|
|
|
#if DUMP_ALL_SYSCALLS
|
|
|
|
dprintln("{} pid {}: {}", process_path, Process::current().pid(), s_syscall_names[syscall]);
|
|
|
|
#endif
|
|
|
|
|
2024-02-12 21:47:40 +02:00
|
|
|
if (syscall < 0 || syscall >= __SYSCALL_COUNT)
|
|
|
|
dwarnln("No syscall {}", syscall);
|
|
|
|
else if (syscall == SYS_FORK)
|
2023-05-28 18:08:26 +03:00
|
|
|
ret = sys_fork_trampoline();
|
2024-02-12 21:47:40 +02:00
|
|
|
else
|
2024-09-22 17:15:07 +03:00
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic warning "-Wmaybe-uninitialized"
|
2024-02-12 21:47:40 +02:00
|
|
|
ret = (Process::current().*s_syscall_handlers[syscall])(arg1, arg2, arg3, arg4, arg5);
|
2024-09-22 17:15:07 +03:00
|
|
|
#pragma GCC diagnostic pop
|
2023-03-13 15:32:46 +02:00
|
|
|
|
2023-04-21 10:40:24 +03:00
|
|
|
asm volatile("cli");
|
|
|
|
|
2024-08-01 17:20:20 +03:00
|
|
|
#if DUMP_ALL_SYSCALLS
|
2024-08-22 14:17:16 +03:00
|
|
|
if (ret.is_error())
|
|
|
|
dprintln("{} pid {}: {}: {}", process_path, Process::current().pid(), s_syscall_names[syscall], ret.error());
|
2024-08-01 17:20:20 +03:00
|
|
|
else
|
2024-08-22 14:17:16 +03:00
|
|
|
dprintln("{} pid {}: {}: {}", process_path, Process::current().pid(), s_syscall_names[syscall], ret.value());
|
|
|
|
#else
|
|
|
|
if (ret.is_error() && ret.error().get_error_code() == ENOTSUP)
|
|
|
|
dwarnln("{} pid {}: {}: ENOTSUP", process_path, Process::current().pid(), s_syscall_names[syscall]);
|
2024-08-01 17:20:20 +03:00
|
|
|
#endif
|
2024-01-03 02:08:01 +02:00
|
|
|
|
2023-08-29 00:13:21 +03:00
|
|
|
if (ret.is_error() && ret.error().is_kernel_error())
|
|
|
|
Kernel::panic("Kernel error while returning to userspace {}", ret.error());
|
|
|
|
|
2023-12-06 13:02:17 +02:00
|
|
|
auto& current_thread = Thread::current();
|
|
|
|
if (current_thread.can_add_signal_to_execute())
|
|
|
|
current_thread.handle_signal();
|
|
|
|
|
2023-08-03 10:42:14 +03:00
|
|
|
ASSERT(Kernel::Thread::current().state() == Kernel::Thread::State::Executing);
|
2023-09-23 02:28:25 +03:00
|
|
|
|
2023-06-12 14:16:48 +03:00
|
|
|
if (ret.is_error())
|
|
|
|
return -ret.error().get_error_code();
|
|
|
|
return ret.value();
|
2023-03-13 15:32:46 +02:00
|
|
|
}
|
|
|
|
|
2024-01-24 14:43:46 +02:00
|
|
|
}
|