开发一个RISC-V上的操作系统(八)—— 抢占式多任务(Preemptive Multitasking)

目录

一、抢占式多任务

二、代码实现

三、上板测试


本节的代码在仓库的 06_Preemptive_Muti_Task 目录下,仓库链接:riscv_os: 一个RISC-V上的简易操作系统

本文代码的运行调试会在前面开发的RISC-V处理器上进行,仓库链接:cpu_prj: 一个基于RISC-V指令集的CPU实现

在第五节的内容中,我们实现了协作式多任务,这一节我们要实现的是抢占式多任务。

一、抢占式多任务

我们先回忆一下协作式和抢占式多任务的概念。

  • 协作式多任务(Cooperative Mutitasking):协作式环境下,下一个任务被调度的前提是当前任务主动放弃处理器。
  • 抢占式多任务(Preemptive Multitasking):抢占式环境下,操作系统完全决定任务调度方案,操作系统可以剥夺当前任务对处理器的使用,并且将处理器提供给其他任务

可以看到,在协作式多任务中,需要用户程序主动放弃处理器,其他程序才能运行,这种调度方式十分不合理(万一有一个用户程序不懂谦让,那整个系统就乱了)。

而在抢占式多任务中,任务的调度完全由操作系统决定,采用统一的调度方式,这样就防止了某些用户程序不释放处理器。

在这一节,我们会利用第七节实现的硬件定时器来实现简单的抢占式调度(每个任务运行一个 tick 的时间)。

二、代码实现

大致步骤如下,当中断发生时,进入 trap_handler 函数,如果是定时器中断则进入 timer_handler 函数,在 timer_handler 函数中调用了 schedule 函数,从而实现任务切换的效果:

开发一个RISC-V上的操作系统(八)—— 抢占式多任务(Preemptive Multitasking)_第1张图片

timer_handler 函数代码如下:

void timer_handler()
{
        timer_write_reg(TIMER_CTRL, (timer_read_reg(TIMER_CTRL) & ~(TIMER_INT_PENDING)));
        _tick++;
        printf("tick: %d\n", _tick);

        timer_load(TIMER_INTERVAL);

        schedule();
}

和之前不同的是,context 上下文还需要额外保存 mepc 值,switch_to 函数最后的返回由 ret 改成了 mret(这样switch_to 函数就不需要返回,直接通过 mret 跳转到下一个任务上下文的 mepc 处执行)。 

# void switch_to(struct context *next);
# a0: pointer to the context of the next task
.globl switch_to
.align 4
switch_to:
        # switch mscratch to point to the context of the next task
        csrw    mscratch, a0
        # set mepc to the pc of the next task
        lw      a1, 124(a0)
        csrw    mepc, a1

        # Restore all GP registers
        # Use t6 to point to the context of the new task
        mv      t6, a0
        reg_restore t6

        # Do actual context switching.
        # Notice this will enable global interrupt
        mret

.end

三、上板测试

将程序烧录到板子上后,打开串口调试助手,现象如下:

开发一个RISC-V上的操作系统(八)—— 抢占式多任务(Preemptive Multitasking)_第2张图片

可以看到任务随着 tick 的增加而切换。

你可能感兴趣的:(RISC-V上的操作系统设计,risc-v,linux)