用户态到内核态切换相对比较简单。用户态切换到内核态时,需要切换堆栈,堆栈地址在当前task的tss段中,因此需要先建立好一个task,并为内核态的ss和esp赋好值。
仍然使用中断切换到内核态,中断处理函数中,首先将保存在新堆栈中的老堆栈的esp取出来,然后在老堆栈的esp指针指向位置之上建立一个trapframe(不包括ss和esp值),然后将新堆栈中的这一堆内容拷贝过来,然后给tf赋值为新堆栈的trapframe的地址,之后直接返回到trap_asm就ok了。
在使用过程中,发现切换不了。使用gdb跟踪发现,不断发生通用保护异常13,才发现idt中为用户态切换到内核态而建的中断的中断描述符的dpl仍然为0,没有改成3,导致执行该中断时不断发生通用保护异常。将该中断描述符的dpl设置为DPL_USER(3)后,切换正常:
struct trapframe *newtf = (struct trapframe*)(tf->esp - sizeof(struct trapframe) + 8); memmove(newtf, tf, sizeof(struct trapframe)-8); // newtf has no ss // newtf->ss = SS_KDATA; newtf->cs = SS_KTEXT; newtf->ds = SS_KDATA; newtf->es = SS_KDATA; newtf->fs = SS_KDATA; newtf->gs = SS_KDATA; tf = newtf;
建立中断描述符表的代码改一下:
int i; for (i=0; i<256; i++) setintrgate(&idt[i], SS_KTEXT, vectors[i], DPL_KERNEL); setintrgate(&idt[TRAPNO_U2K], SS_KTEXT, vectors[TRAPNO_U2K], DPL_USER);