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 提供支持
在本页

这有帮助吗?

lab6: 进程调度

在前两章中,我们已经分别实现了内核进程和用户进程,并且让他们正确运行了起来。我们同时也实现了一个简单的调度算法,FIFO调度算法,来对我们的进程进行调度。但是,单单如此就够了吗?显然,我们可以让ucore支持更加丰富的调度算法,从而满足各方面的调度需求。在本章里,我们要实现一个调度框架,从而方便我们实现各种各样的调度算法。

在本次实验中,我们在init/init.c中加入了对sched_init函数的调用。这个函数主要完成调度器和特定调度算法的绑定。初始化后,我们在调度函数中就可以使用相应的接口了。这也是在C语言环境下对于面向对象编程模式的一种模仿。这样之后,我们只需要关注于实现调度类的接口即可,操作系统也同样不关心调度类具体的实现,方便了新调度算法的开发。

在ucore中,进程有如下几个状态:

  • PROC_UNINIT:这个状态表示进程刚刚被分配相应的进程控制块,但还没有初始化,需要进一步的初始化才能进入PROC_RUNNABLE的状态。

  • PROC_SLEEPING:这个状态表示进程正在等待某个事件的发生,通常由于等待锁的释放,或者主动交出CPU资源(do_sleep)。这个状态下的进程是不会被调度的。

  • PROC_RUNNABLE:这个状态表示进程已经准备好要执行了,只需要操作系统给他分配相应的CPU资源就可以运行。

  • PROC_ZOMBIE:这个状态表示进程已经退出,相应的资源被回收(大部分),almost dead。

一个进程的生命周期一般由如下过程组成:

1. 刚刚开始初始化,进程处在PROC_UNINIT的状态 2. 进程已经完成初始化,时刻准备执行,进入PROC_RUNNABLE状态 3. 在调度的时候,调度器选中该进程进行执行,进程处在running的状态 4.(1) 正在运行的进程由于wait等系统调用被阻塞,进入PROC_SLEEPING,等待相应的资源或者信号。 4.(2) 另一种可能是正在运行的进程被外部中断打断,此时进程变为PROC_RUNNABLE状态,等待下次被调用 5. 等待的事件发生,进程又变成PROC_RUNNABLE状态 6. 重复3~6,直到进程执行完毕,通过exit进入PROC_ZOMBIE状态,由父进程对他的资源进行回收,释放进程控制块。至此,这个进程的生命周期彻底结束

下面我们来看一看如何实现内核对于进程的调度。

项目组成

lab6
├── 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
│   │   ├── ide.c
│   │   ├── ide.h
│   │   ├── intr.c
│   │   ├── intr.h
│   │   ├── kbdreg.h
│   │   ├── picirq.c
│   │   └── picirq.h
│   ├── fs
│   │   ├── fs.h
│   │   ├── swapfs.c
│   │   └── swapfs.h
│   ├── init
│   │   ├── entry.S
│   │   └── init.c
│   ├── libs
│   │   ├── readline.c
│   │   └── stdio.c
│   ├── mm
│   │   ├── default_pmm.c
│   │   ├── default_pmm.h
│   │   ├── kmalloc.c
│   │   ├── kmalloc.h
│   │   ├── memlayout.h
│   │   ├── mmu.h
│   │   ├── pmm.c
│   │   ├── pmm.h
│   │   ├── swap.c
│   │   ├── swap.h
│   │   ├── swap_fifo.c
│   │   ├── swap_fifo.h
│   │   ├── vmm.c
│   │   └── vmm.h
│   ├── process
│   │   ├── entry.S
│   │   ├── proc.c
│   │   ├── proc.h
│   │   └── switch.S
│   ├── schedule
│   │   ├── default_sched.h
│   │   ├── default_sched_c
│   │   ├── default_sched_stride.c
│   │   ├── sched.c
│   │   └── sched.h
│   ├── sync
│   │   └── sync.h
│   ├── syscall
│   │   ├── syscall.c
│   │   └── syscall.h
│   └── trap
│       ├── trap.c
│       ├── trap.h
│       └── trapentry.S
├── libs
│   ├── atomic.h
│   ├── defs.h
│   ├── elf.h
│   ├── error.h
│   ├── hash.c
│   ├── list.h
│   ├── printfmt.c
│   ├── rand.c
│   ├── riscv.h
│   ├── sbi.h
│   ├── skew_heap.h
│   ├── stdarg.h
│   ├── stdio.h
│   ├── stdlib.h
│   ├── string.c
│   ├── string.h
│   └── unistd.h
├── tools
│   ├── boot.ld
│   ├── function.mk
│   ├── gdbinit
│   ├── grade.sh
│   ├── kernel.ld
│   ├── sign.c
│   ├── user.ld
│   └── vector.c
└── user
    ├── badarg.c
    ├── badsegment.c
    ├── divzero.c
    ├── exit.c
    ├── faultread.c
    ├── faultreadkernel.c
    ├── forktest.c
    ├── forktree.c
    ├── hello.c
    ├── libs
    │   ├── initcode.S
    │   ├── panic.c
    │   ├── stdio.c
    │   ├── syscall.c
    │   ├── syscall.h
    │   ├── ulib.c
    │   ├── ulib.h
    │   └── umain.c
    ├── matrix.c
    ├── pgdir.c
    ├── priority.c
    ├── softint.c
    ├── spin.c
    ├── testbss.c
    ├── waitkill.c
    └── yield.c

16 directories, 105 files
上一页项目组成和执行流下一页再次认识进程切换

最后更新于4年前

这有帮助吗?