* Distributed under the terms of the MIT License.
*/
#include <string.h>
#include <arch_cpu.h>
#include <arch_debug.h>
#include <arch/thread.h>
#include <boot/stage2.h>
#include <commpage.h>
#include <kernel.h>
#include <thread.h>
#include <team.h>
#include <vm/vm_types.h>
#include <vm/VMAddressSpace.h>
#include "RISCV64VMTranslationMap.h"
extern "C" void SVecU();
status_t
arch_thread_init(struct kernel_args *args)
{
return B_OK;
}
status_t
arch_team_init_team_struct(Team *team, bool kernel)
{
return B_OK;
}
status_t
arch_thread_init_thread_struct(Thread *thread)
{
return B_OK;
}
void
arch_thread_init_kthread_stack(Thread* thread, void* _stack, void* _stackTop,
void (*function)(void*), const void* data)
{
memset(&thread->arch_info.context, 0, sizeof(arch_context));
thread->arch_info.context.sp = (addr_t)_stackTop;
thread->arch_info.context.s[0] = 0;
thread->arch_info.context.s[1] = (addr_t)function;
thread->arch_info.context.s[2] = (addr_t)data;
thread->arch_info.context.ra = (addr_t)arch_thread_entry;
memset(&thread->arch_info.fpuContext, 0, sizeof(fpu_context));
}
status_t
arch_thread_init_tls(Thread *thread)
{
thread->user_local_storage =
thread->user_stack_base + thread->user_stack_size;
return B_OK;
}
void
arch_thread_context_switch(Thread *from, Thread *to)
{
dprintf("arch_thread_context_switch(%p(%s), %p(%s))\n", from, from->name,
to, to->name);
*/
auto fromMap = (RISCV64VMTranslationMap*)from->team->address_space->TranslationMap();
auto toMap = (RISCV64VMTranslationMap*)to->team->address_space->TranslationMap();
int cpu = to->cpu->cpu_num;
toMap->ActiveOnCpus().SetBitAtomic(cpu);
fromMap->ActiveOnCpus().ClearBitAtomic(cpu);
save_fpu(&from->arch_info.fpuContext);
restore_fpu(&to->arch_info.fpuContext);
SetSatp(toMap->Satp());
FlushTlbAllAsid(0);
arch_context_switch(&from->arch_info.context, &to->arch_info.context);
}
void
arch_thread_dump_info(void *info)
{
}
status_t
arch_thread_enter_userspace(Thread *thread, addr_t entry, void *arg1,
void *arg2)
{
addr_t commpageAdr = (addr_t)thread->team->commpage_address;
addr_t threadExitAddr;
ASSERT(user_memcpy(&threadExitAddr,
&((addr_t*)commpageAdr)[COMMPAGE_ENTRY_RISCV64_THREAD_EXIT],
sizeof(threadExitAddr)) >= B_OK);
threadExitAddr += commpageAdr;
disable_interrupts();
arch_stack* stackHeader = (arch_stack*)thread->kernel_stack_top - 1;
stackHeader->thread = thread;
iframe frame;
memset(&frame, 0, sizeof(frame));
SstatusReg status{.val = Sstatus()};
status.pie = (1 << modeS);
status.spp = modeU;
frame.status = status.val;
frame.epc = entry;
frame.a0 = (addr_t)arg1;
frame.a1 = (addr_t)arg2;
frame.ra = threadExitAddr;
frame.sp = thread->user_stack_base + thread->user_stack_size;
frame.tp = thread->user_local_storage;
arch_load_user_iframe(stackHeader, &frame);
return B_ERROR;
}
bool
arch_on_signal_stack(Thread *thread)
{
struct iframe* frame = thread->arch_info.userFrame;
if (frame == NULL) {
panic("arch_on_signal_stack(): No user iframe!");
return false;
}
return frame->sp >= thread->signal_stack_base
&& frame->sp < thread->signal_stack_base
+ thread->signal_stack_size;
}
static uint8*
get_signal_stack(Thread* thread, struct iframe* frame,
struct sigaction* action, size_t spaceNeeded)
{
if (
thread->signal_stack_enabled &&
(action->sa_flags & SA_ONSTACK) != 0 && (
frame->sp < thread->signal_stack_base ||
frame->sp >= thread->signal_stack_base + thread->signal_stack_size
)
) {
addr_t stackTop = thread->signal_stack_base
+ thread->signal_stack_size;
return (uint8*)ROUNDDOWN(stackTop - spaceNeeded, 16);
}
return (uint8*)ROUNDDOWN(frame->sp - spaceNeeded, 16);
}
status_t
arch_setup_signal_frame(Thread *thread, struct sigaction *sa,
struct signal_frame_data *signalFrameData)
{
iframe* frame = thread->arch_info.userFrame;
signalFrameData->context.uc_mcontext.x[ 0] = frame->ra;
signalFrameData->context.uc_mcontext.x[ 1] = frame->sp;
signalFrameData->context.uc_mcontext.x[ 2] = frame->gp;
signalFrameData->context.uc_mcontext.x[ 3] = frame->tp;
signalFrameData->context.uc_mcontext.x[ 4] = frame->t0;
signalFrameData->context.uc_mcontext.x[ 5] = frame->t1;
signalFrameData->context.uc_mcontext.x[ 6] = frame->t2;
signalFrameData->context.uc_mcontext.x[ 7] = frame->fp;
signalFrameData->context.uc_mcontext.x[ 8] = frame->s1;
signalFrameData->context.uc_mcontext.x[ 9] = frame->a0;
signalFrameData->context.uc_mcontext.x[10] = frame->a1;
signalFrameData->context.uc_mcontext.x[11] = frame->a2;
signalFrameData->context.uc_mcontext.x[12] = frame->a3;
signalFrameData->context.uc_mcontext.x[13] = frame->a4;
signalFrameData->context.uc_mcontext.x[14] = frame->a5;
signalFrameData->context.uc_mcontext.x[15] = frame->a6;
signalFrameData->context.uc_mcontext.x[16] = frame->a7;
signalFrameData->context.uc_mcontext.x[17] = frame->s2;
signalFrameData->context.uc_mcontext.x[18] = frame->s3;
signalFrameData->context.uc_mcontext.x[19] = frame->s4;
signalFrameData->context.uc_mcontext.x[20] = frame->s5;
signalFrameData->context.uc_mcontext.x[21] = frame->s6;
signalFrameData->context.uc_mcontext.x[22] = frame->s7;
signalFrameData->context.uc_mcontext.x[23] = frame->s8;
signalFrameData->context.uc_mcontext.x[24] = frame->s9;
signalFrameData->context.uc_mcontext.x[25] = frame->s10;
signalFrameData->context.uc_mcontext.x[26] = frame->s11;
signalFrameData->context.uc_mcontext.x[27] = frame->t3;
signalFrameData->context.uc_mcontext.x[28] = frame->t4;
signalFrameData->context.uc_mcontext.x[29] = frame->t5;
signalFrameData->context.uc_mcontext.x[30] = frame->t6;
signalFrameData->context.uc_mcontext.pc = frame->epc;
save_fpu((fpu_context*)&signalFrameData->context.uc_mcontext.f[0]);
signal_get_user_stack(frame->sp, &signalFrameData->context.uc_stack);
dprintf(" thread->signal_stack_enabled: %d\n",
thread->signal_stack_enabled);
if (thread->signal_stack_enabled) {
dprintf(" signal stack: 0x%" B_PRIxADDR " - 0x%" B_PRIxADDR "\n",
thread->signal_stack_base,
thread->signal_stack_base + thread->signal_stack_size
);
}
*/
signalFrameData->syscall_restart_return_value = thread->arch_info.oldA0;
uint8* userStack = get_signal_stack(thread, frame, sa,
sizeof(*signalFrameData));
status_t res = user_memcpy(userStack, signalFrameData,
sizeof(*signalFrameData));
if (res < B_OK)
return res;
addr_t commpageAdr = (addr_t)thread->team->commpage_address;
addr_t signalHandlerAddr;
ASSERT(user_memcpy(&signalHandlerAddr,
&((addr_t*)commpageAdr)[COMMPAGE_ENTRY_RISCV64_SIGNAL_HANDLER],
sizeof(signalHandlerAddr)) >= B_OK);
signalHandlerAddr += commpageAdr;
frame->ra = frame->epc;
frame->sp = (addr_t)userStack;
frame->epc = signalHandlerAddr;
frame->a0 = frame->sp;
return B_OK;
}
int64
arch_restore_signal_frame(struct signal_frame_data* signalFrameData)
{
iframe* frame = thread_get_current_thread()->arch_info.userFrame;
thread_get_current_thread()->arch_info.oldA0
= signalFrameData->syscall_restart_return_value;
frame->ra = signalFrameData->context.uc_mcontext.x[ 0];
frame->sp = signalFrameData->context.uc_mcontext.x[ 1];
frame->gp = signalFrameData->context.uc_mcontext.x[ 2];
frame->tp = signalFrameData->context.uc_mcontext.x[ 3];
frame->t0 = signalFrameData->context.uc_mcontext.x[ 4];
frame->t1 = signalFrameData->context.uc_mcontext.x[ 5];
frame->t2 = signalFrameData->context.uc_mcontext.x[ 6];
frame->fp = signalFrameData->context.uc_mcontext.x[ 7];
frame->s1 = signalFrameData->context.uc_mcontext.x[ 8];
frame->a0 = signalFrameData->context.uc_mcontext.x[ 9];
frame->a1 = signalFrameData->context.uc_mcontext.x[10];
frame->a2 = signalFrameData->context.uc_mcontext.x[11];
frame->a3 = signalFrameData->context.uc_mcontext.x[12];
frame->a4 = signalFrameData->context.uc_mcontext.x[13];
frame->a5 = signalFrameData->context.uc_mcontext.x[14];
frame->a6 = signalFrameData->context.uc_mcontext.x[15];
frame->a7 = signalFrameData->context.uc_mcontext.x[16];
frame->s2 = signalFrameData->context.uc_mcontext.x[17];
frame->s3 = signalFrameData->context.uc_mcontext.x[18];
frame->s4 = signalFrameData->context.uc_mcontext.x[19];
frame->s5 = signalFrameData->context.uc_mcontext.x[20];
frame->s6 = signalFrameData->context.uc_mcontext.x[21];
frame->s7 = signalFrameData->context.uc_mcontext.x[22];
frame->s8 = signalFrameData->context.uc_mcontext.x[23];
frame->s9 = signalFrameData->context.uc_mcontext.x[24];
frame->s10 = signalFrameData->context.uc_mcontext.x[25];
frame->s11 = signalFrameData->context.uc_mcontext.x[26];
frame->t3 = signalFrameData->context.uc_mcontext.x[27];
frame->t4 = signalFrameData->context.uc_mcontext.x[28];
frame->t5 = signalFrameData->context.uc_mcontext.x[29];
frame->t6 = signalFrameData->context.uc_mcontext.x[30];
frame->epc = signalFrameData->context.uc_mcontext.pc;
restore_fpu((fpu_context*)&signalFrameData->context.uc_mcontext.f[0]);
return frame->a0;
}
* arch_fork_arg structure to be passed to arch_restore_fork_frame().
* Also makes sure to return the right value.
*/
void
arch_store_fork_frame(struct arch_fork_arg *arg)
{
dprintf("arch_store_fork_frame()\n");
dprintf(" arg: %p\n", arg);
dprintf(" userFrame: %p\n",
thread_get_current_thread()->arch_info.userFrame);
*/
memcpy(&arg->frame, thread_get_current_thread()->arch_info.userFrame,
sizeof(iframe));
arg->frame.a0 = 0;
}
* arch_fork_arg structure.
* Needs to be called from within the child team, ie. instead of
* arch_thread_enter_uspace() as thread "starter".
* This function does not return to the caller, but will enter userland
* in the child team at the same position where the parent team left of.
*/
void
arch_restore_fork_frame(struct arch_fork_arg *arg)
{
disable_interrupts();
arch_stack* stackHeader = (arch_stack*)thread_get_current_thread()->kernel_stack_top - 1;
stackHeader->thread = thread_get_current_thread();
SstatusReg status{.val = Sstatus()};
status.pie = (1 << modeS);
status.spp = modeU;
arg->frame.status = status.val;
arch_load_user_iframe(stackHeader, &arg->frame);
}