而为了让生活更美好,**_entry函数使用汇编指令初始化栈后,跳转到使用C语言编写的start(kernel/start.c:20)。
start函数的主要工作就是设置CSRs(control
and state registers),从而切换到S-mode(supervisor
mode),并将执行流设置成main(kernel/main.c:10)。start函数实现的非常巧妙,其通过设置CSRs,伪造一个异常处理保存的上下文,其上下文的特权级是S-mode,PC是main。执行mret**指令后,通过恢复上下文,完成特权级和执行流的转换
// set up to take exceptions and traps while in the kernel. void trapinithart(void) { w_stvec((uint64)kernelvec); // 设置stvec指向kernelvec,kernelvec是一个.S文件,当有中断产生时,会通过ecall进入管理员模式,同时由ecall指令设置stvec指向trampoline,这是硬件层面做的吗?在代码里并没x有看到任何像w_stvec(trampolone)的代码。 }
// Set up first user process. void userinit(void) { structproc *p;
p = allocproc(); initproc = p;
// allocate one user page and copy init's instructions // and data into it. uvminit(p->pagetable, initcode, sizeof(initcode)); p->sz = PGSIZE;
// prepare for the very first "return" from kernel to user. p->trapframe->epc = 0; // user program counter p->trapframe->sp = PGSIZE; // user stack pointer
// Look in the process table for an UNUSED proc. // If found, initialize state required to run in the kernel, // and return with p->lock held. // If there are no free procs, or a memory allocation fails, return 0. staticstruct proc* allocproc(void) { structproc *p;
// An empty user page table. p->pagetable = proc_pagetable(p); if(p->pagetable == 0){ freeproc(p); release(&p->lock); return0; }
// Set up new context to start executing at forkret, // which returns to user space. memset(&p->context, 0, sizeof(p->context)); p->context.ra = (uint64)forkret; p->context.sp = p->kstack + PGSIZE;
return p; }
int growproc(int n) { uint sz; structproc *p = myproc();
// map the trampoline code (for system call return) // at the highest user virtual address. // only the supervisor uses it, on the way // to/from user space, so not PTE_U. if(mappages(pagetable, TRAMPOLINE, PGSIZE, (uint64)trampoline, PTE_R | PTE_X) < 0){ uvmfree(pagetable, 0); return0; }
// map the trapframe just below TRAMPOLINE, for trampoline.S. if(mappages(pagetable, TRAPFRAME, PGSIZE, (uint64)(p->trapframe), PTE_R | PTE_W) < 0){ uvmunmap(pagetable, TRAMPOLINE, 1, 0); uvmfree(pagetable, 0); return0; }