MIT6.1810: xv6 book Chapter4: Traps and system calls 笔记

三种使CPU暂停指令转而执行内核代码的事件:

1.系统调用 ecall

2.exception指令异常

3.设备interrupt

统称为trap

xv6的trap处理分为四个阶段:RISC-V CPU硬件行为,C代码前的汇编指令,处理trap的C函数,系统调用或设备驱动服务路线

处理陷入的内核代码被称为handler ,其中的首批指令通常用汇编语言编写,有时被称为vector

RISC-V trap machinery

每个RISC-V CPU都有一系列硬件控制寄存器,下面列举其中重要的几个:

  • stvec(Supervisor Trap Vector Base Address Register) 内核将其trap handler写在这里,RISC-V跳转到stvec中的地址处理陷入(从用户态陷入:uservec;从内核态陷入:kernelvec)
  • sepc(Supervisor Exception Program Counter) 陷入发生时,RISC-V将pc存在这里,从陷入返回时将sepc复制给pc
  • scause 描述陷入的原因
  • sscratch(Supervisor Scratch Register) 帮助避免在保存用户寄存器前覆盖它们
  • sstatusSIE bit 设备中断是否启用;SPP bit trap来自user mode/supervisor mode
  • satp(Supervisor Address Translation and Protection) 包含了指向page table的物理内存地址

Traps from user space

代码流程:

trampoline.S/uservec -> trap.c/usertrap() -> trampoline.S/userret

xv6中,uservec执行时,要存储32个寄存器的值,此时没有空闲的寄存器

RISC-V提供了寄存器sscratch,下面代码表示a0 获得 sscratch 中预先存放的 TRAPFRAME 地址,sscratch 保存了用户态的 a0 值

assembly 复制代码
# save user a0 in sscratch so
# a0 can be used to get at TRAPFRAME.
csrw sscratch, a0

uservec和userret用汇编语言写的原因是很难用C语言保存所有寄存器的值,以及在切换页表时使C代码继续生效

Code: System call arguments

有些系统调用需要指针参数来读写用户内存,为防止用户指令传递恶意地址指针,xv6实现了一系列确保指针安全传递的函数

e.g. syscall.c/fetchstr()调用了vm.c/copyinstr(),后者调用walkaddr()确保传递的虚拟地址在用户页表中可用

Traps from kernel space

代码流程:

kernelvec.S -> trap.c/kerneltrap() -> kernelvec.S

Real world

现实中部分操作系统会将内核内存映射到每个进程的用户页表,这样可以省去用户态到内核态的页表切换;系统调用的实现也可以利用当前进程的用户页表而直接解引用用户指针。

相关推荐
jimbo_lee1 小时前
yocto 用法(随手笔记,记录以备不时之需)
笔记·yocto
胡图图不糊涂^_^3 小时前
测试用例篇——设计测试用例的方法
笔记·学习·测试用例·判定表法·正交法生成用例测试·等价类·边界值
IT19954 小时前
Dify笔记-知识库创建后设置和召回测试
笔记·dify
飞翔中文网4 小时前
Java学习笔记之抽象类
java·笔记·学习
中屹指纹浏览器6 小时前
2026指纹浏览器行为指纹对抗技术详解:从算法识别到真人模拟全方案
经验分享·笔记
2301_809051146 小时前
Linux 数据库开发 学习笔记
笔记·学习·数据库开发
很楠爱上6 小时前
Node.js 模块化学习笔记
笔记·学习·node.js
mnasd6 小时前
RockyLinux 9.5 部署 Kubernetes1.35 集群
笔记
東隅已逝,桑榆非晚7 小时前
编译和链接
c语言·笔记