ucore step by step
tutorial
tutorial
  • 欢迎来到ucore step-by-step的世界
  • lab0: 预备起
    • 溯源: ucore的历史
    • 概览: 指导书的结构
    • 开搞: 实验环境搭建
  • lab0.5: 比麻雀更小的麻雀(最小可执行内核)
    • 内存布局,OpenSBI,elf和bin
    • 链接脚本和入口点
    • "真正的"入口点
    • 从SBI到stdio
    • Just make it
    • 项目组成和执行流
  • lab1: 断, 都可以断
    • 掉进兔子洞(中断入口点)
    • 中断处理程序
    • 滴答滴答(时钟中断)
    • 项目组成和执行流
  • lab2: 物理内存和页表
    • 内核初始映射
    • 物理内存管理
    • 页面分配算法
    • 项目组成和执行流
  • lab3: 缺页异常和页面置换
    • 扫清外围
    • 使用多级页表
    • 页面置换机制
    • FIFO置换算法
    • 项目组成和执行流
  • lab4: 进程管理
    • 进程与线程
    • 相关数据结构
    • 进程模块初始化
    • 进程切换
    • 项目组成和执行流
  • lab5: 用户程序
    • 用户程序
    • system call!
    • 中断处理
    • 项目组成和执行流
  • lab6: 进程调度
    • 再次认识进程切换
    • 调度算法框架
    • 项目组成和执行流
  • lab7: 同步互斥
    • 同步互斥的基本概念
    • 信号量
    • 条件变量与管程
    • 项目组成和执行流
  • lab8: 文件系统
    • 文件系统抽象层VFS
    • 硬盘文件系统SFS
    • 设备即文件
    • 从zhong duan 到 zhong duan
    • 项目组成和执行流
  • 练习题
    • lab1
    • lab2
    • lab3
    • lab4
    • lab5
    • lab6
    • lab7
    • lab8
  • 附录
    • makefile
由 GitBook 提供支持
在本页
  • 项目组成
  • 硬件驱动层
  • 初始化
  • 中断处理
  • 执行流

这有帮助吗?

  1. lab1: 断, 都可以断

项目组成和执行流

项目组成

lab1
├── Makefile
├── kern
│   ├── debug
│   │   ├── assert.h
│   │   ├── kdebug.c
│   │   ├── kdebug.h
│   │   ├── kmonitor.c
│   │   ├── kmonitor.h
│   │   ├── panic.c
│   │   └── stab.h
│   ├── driver
│   │   ├── clock.c
│   │   ├── clock.h
│   │   ├── console.c
│   │   ├── console.h
│   │   ├── intr.c
│   │   └── intr.h
│   ├── init
│   │   ├── entry.S
│   │   └── init.c
│   ├── libs
│   │   └── stdio.c
│   ├── mm
│   │   ├── memlayout.h
│   │   ├── mmu.h
│   │   ├── pmm.c
│   │   └── pmm.h
│   └── trap
│       ├── trap.c
│       ├── trap.h
│       └── trapentry.S
├── lab1.md
├── libs
│   ├── defs.h
│   ├── error.h
│   ├── printfmt.c
│   ├── readline.c
│   ├── riscv.h
│   ├── sbi.c
│   ├── sbi.h
│   ├── stdarg.h
│   ├── stdio.h
│   ├── string.c
│   └── string.h
├── readme.md
└── tools
    ├── function.mk
    ├── gdbinit
    ├── grade.sh
    ├── kernel.ld
    ├── sign.c
    └── vector.c

9 directories, 43 files

只介绍新增的或变动较大的文件。

硬件驱动层

kern/driver/clock.c(h): 通过OpenSBI的接口, 可以读取当前时间(rdtime), 设置时钟事件(sbi_set_timer),是时钟中断必需的硬件支持。

kern/driver/intr.c(h): 中断也需要CPU的硬件支持,这里提供了设置中断使能位的接口(其实只封装了一句riscv指令)。

初始化

kern/init/init.c: 需要调用中断机制的初始化函数。

中断处理

kern/trap/trapentry.S: 我们把中断入口点设置为这段汇编代码。这段汇编代码把寄存器的数据挪来挪去,进行上下文切换。

kern/trap/trap.c(h): 分发不同类型的中断给不同的handler, 完成上下文切换之后对中断的具体处理,例如外设中断要处理外设发来的信息,时钟中断要触发特定的事件。中断处理初始化的函数也在这里,主要是把中断向量表(stvec)设置成所有中断都要跳到trapentry.S进行处理。

执行流

内核初始化函数kern_init()的执行流:(从kern/init/entry.S进入) -> 输出一些信息说明正在初始化 -> 设置中断向量表(stvec)跳转到的地方为kern/trap/trapentry.S里的一个标记 ->在kern/driver/clock.c设置第一个时钟事件,使能时钟中断->设置全局的S mode中断使能位-> 现在开始不断地触发时钟中断

产生一次时钟中断的执行流:set_sbi_timer()通过OpenSBI的时钟事件触发一个中断,跳转到kern/trap/trapentry.S的__alltraps标记 -> 保存当前执行流的上下文,并通过函数调用,切换为kern/trap/trap.c的中断处理函数trap()的上下文,进入trap()的执行流。切换前的上下文作为一个结构体,传递给trap()作为函数参数 -> kern/trap/trap.c按照中断类型进行分发(trap_dispatch(), interrupt_handler())->执行时钟中断对应的处理语句,累加计数器,设置下一次时钟中断->完成处理,返回到kern/trap/trapentry.S->恢复原先的上下文,中断处理结束。

上一页滴答滴答(时钟中断)下一页lab2: 物理内存和页表

最后更新于4年前

这有帮助吗?