/****************************************************************************** COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011. This file is part of Sortix. Sortix is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Sortix is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Sortix. If not, see . syscall.h Handles system calls from userspace safely. ******************************************************************************/ #include "platform.h" #include "syscall.h" #include "syscallnum.h" #include "panic.h" #include "thread.h" #include "scheduler.h" #include "log.h" // DEBUG namespace Sortix { namespace Syscall { extern "C" { CPU::SyscallRegisters* syscall_state_ptr; unsigned system_was_incomplete; size_t SYSCALL_MAX = SYSCALL_MAX_NUM; void* syscall_list[SYSCALL_MAX_NUM]; } int BadSyscall() { // TODO: Send signal, set errno, or crash/abort process? return -1; } void Init() { for ( size_t i = 0; i < SYSCALL_MAX; i++ ) { syscall_list[i] = (void*) BadSyscall; } } void Register(size_t index, void* funcptr) { if ( SYSCALL_MAX <= index ) { PanicF("attempted to register syscall 0x%p to index %zu, but " "SYSCALL_MAX = %zu", funcptr, index, SYSCALL_MAX); } syscall_list[index] = funcptr; } void Incomplete() { Thread* thread = CurrentThread(); system_was_incomplete = 1; CPU::InterruptRegisters* regs = InterruptRegs(); thread->SaveRegisters(regs); Scheduler::SetThreadState(thread, Thread::State::BLOCKING); Scheduler::Switch(regs); } void AsIs() { system_was_incomplete = 1; } void ScheduleResumption(Thread* thread) { Scheduler::SetThreadState(thread, Thread::State::RUNNABLE); } extern "C" size_t resume_syscall(void* scfunc, size_t scsize, size_t* scstate); void Resume(CPU::InterruptRegisters* regs) { Thread* thread = CurrentThread(); syscall_state_ptr = (CPU::SyscallRegisters*) regs; ASSERT(thread->scfunc); size_t* scstate = thread->scstate; size_t scsize = thread->scsize; void* scfunc = thread->scfunc; system_was_incomplete = 0; size_t result = resume_syscall(scfunc, scsize, scstate); bool incomplete = (system_was_incomplete); system_was_incomplete = 1; if ( !incomplete ) { syscall_state_ptr->result = result; return; } Incomplete(); } CPU::InterruptRegisters* InterruptRegs() { return (CPU::InterruptRegisters*) syscall_state_ptr; } CPU::SyscallRegisters* SyscallRegs() { return syscall_state_ptr; } } }