ex1
参照nextfree,然后需要对齐,直接调用boot map的代码
ex2
先单核启动的(BSP),再是APs,AP从实模式开始,入口:boot aps,就像bootloader加载一样,然后boot aps把AP启动代码复制到一个实模式下的内存地址。不同的是,我们可以控制AP从哪里开始执行,复制到0x7000(MPENTRY_PADDR),不过小于640K的空闲地址都可以用。
然后,boot aps依次激活AP,通过发STARTUP给对应LAPIC,包括 AP应当开始运行的CS:IP地址(MPENTRY_PADDR)。mpentry的入口代码和boot相似,把AP放进保护模式,允许paging,然后调用C函数mp main。aps等待AP的信号(在cpu_status field of its struct CpuInfo),然后激活下一个。
修改page init:直接加入一个判断i不等于MPENTRY_PADDR/PGSIZE
ex3
for遍历NCPU
根据注释里面的第一项把kstack top i往下的k stack size大小 map到physical memory percpu_kstacks的地方
ex4
把所有的ts都改成thiscpu
esp的位置:参照mem init的栈顶向下长度
位移运算3位是因为每个descriptor的大小是8字节
哦还有忘记写了,需要吧boot map large改成boot否则会一直爆重启...
ex5
照着文档要求加放锁即可
ex6
sched_yield() in the new kern/sched.c 是选择下一个环境,按顺序循环搜索env数组,选择刚刚运行的下一个RUNNABLE的env, 然后调用calls env_run() 跳到那个环境
不可以在两个cpu运行同一个环境,正在运行的env会变成RUNNING状态
使用一个for循环遍历env即可,如果iter回到最初的i就退出,再检查一下i是否可以
注意要把init文件里面增加可用cpu数量
TODO
ex7
按照给的写就可以了
注意权限位的bit运算!!(#TODO
ex8
缺页中断发生时,会执行env_pgfault_upcall指定位置的代码。当执行env_pgfault_upcall指定位置的代码时,栈已经转到异常栈,并且压入了UTrapframe结构
不要忘记更新syscall
ex9
In this case, you should start the new stack frame just under the current tf->tf_esp rather than at UXSTACKTOP. You should first push an empty 32-bit word, then a struct UTrapframe.
ex10
当handler执行完后,如何返回到错误时的状态
trap-time esp
trap-time eflags
trap-time eip
trap-time eax start of struct PushRegs
trap-time ecx
trap-time edx
trap-time ebx
trap-time esp
trap-time ebp
trap-time esi
trap-time edi end of struct PushRegs
tf_err (error code)
fault_va <-- %esp when handler is run
这里很难写,网上的攻略根本看不懂,花了相当长的时间
ex11
handler 是传入的用户自定义页错误处理函数指针。
_pgfault_upcall 是一个全局变量,在 lib/pfentry.S 中完成的初始化。它是页错误处理的总入口,页错误除了运行 page fault handler,还需要切换回正常栈。
_pgfault_handler 被赋值为handler,会在 _pgfault_upcall 中被调用,是页错误处理的一部分。
ex12
fork()参照 user/dumbfork.c:但是
1.设置 page fault handler,即 page fault upcall 调用的函数
2.duppage 的范围不同,fork() 不需要复制内核区域的映射
3.为子进程设置 page fault upcall,之所以这么做,是因为 sys_exofork() 并不会复制父进程的 e->env_pgfault_upcall 给子进程。