## 一、电路图



## 二、流水线冲突的情况及解决方案

## 1.控制冒险

如果当前指令为跳转指令,在 decode 阶段检出,而此时 fetch 阶段已经取出了原来的 pc+4 为错误的指令地址

解决方案:在 decode 阶段发现这是一个跳转指令则将一个选择信号与目标的指令地址传回 fetch 阶段,并将 fetch 与 decode 之间的流水线寄存器清零(插入气泡)——因为此时里面存的是从错误的地址取出的指令。

电路实现:在 pc 前插入一个二路选择器,控制信号为来自 decode 阶段的跳转指令,选择下一个 pc 是 pc+4 还是需要跳转的指令地址。

## 2.数据冒险

decode 阶段从寄存器中取数时,前几条指令可能存在往相同的寄存器中写入的情况,此时前三条指令尚未 writeback,寄存器中数据尚未更新,则 decode 阶段取出来的数据是错误的。

| 冲突情况           | 具体指令      | 解决方案         | 如何阻塞       | 电路实现          |
|----------------|-----------|--------------|------------|---------------|
| 正处于 execute 阶  | addi、     | 使用转发,将       | 无需阻塞       | ①将前两条指令要      |
| 段的指令的前两条       | xori、ori、 | 前两条指令即       |            | 写入寄存器的地址      |
| 指令中存在写入寄       | andi、     | 将写入寄存器       |            | 与 execute 阶段两 |
| 存器的情况,且不       | lui 、     | 的值直接替换       |            | 个操作数来源的寄      |
| 是从内存中取出的       | jal 、     | execute 阶段的  |            | 存器地址比较,       |
| 值              | add 、     | 操作数          |            | 若一致,②则进行      |
|                | sub 、     |              |            | 转发,直接替换       |
|                | and.      |              |            | execute 的操作数。 |
|                | or , xor, |              |            | (注意写入\$0 寄存   |
|                | auipc     |              |            | 器的值不做转发)      |
| 正处于 execute 阶  | Id        | 此时需要等一       | decode/ex  | ①判断 memory 阶  |
| 段的指令的前一条       |           | 个时钟周期才       | ecute 和    | 段的操作是否为 ld    |
| 指令中存在从内存       |           | 能够获得转发       | fetch/deco | 操作,           |
| 写入寄存器的情        |           | 回来的数据,       | de 寄存器     | 若是②则将         |
| 况。由于转发的数       |           | 让 execute 阶段 | 阻塞         | memory 阶段写入   |
| 据源只能来自流水       |           | 及之后的指令       | execute/m  | 寄存器的地址与       |
| 线寄存器的输出,       |           | 在原地停留一       | emory 寄    | execute 阶段操作数 |
| 需要等前一条指令       |           | 个时钟周期,       | 存器中插       | 来源的寄存器的地      |
| 到 writeback 阶段 |           | 并使 memory    | 入气泡        | 址进行比较,        |
| 才能转发           |           | 阶段执行的下       |            | 若一致,③则插入      |
|                |           | 一条指令为空       |            | 气泡和阻塞         |
|                |           | 指令           |            |               |
| 正处于 decode 阶   | beq, jalr | 此时前一条指       | decode/ex  | ①判断 decode 阶  |
| 段的指令需要从寄       |           | 令将要写入寄       | ecute 寄存   | 段指令是否为 beq    |
| 存器中取出值直接       |           | 存器的值尚未       | 器插入气       | 或 jalr        |
| 进行比较立即数加       |           | 从 alu 中运算得   | 泡,         | ②判断 decode 取  |
| 载等操作, 此时其      |           | 出, 因此让       | fetch/deco | 出的寄存器地址是      |
| 前一条指令的结果       |           | decode 阶段及   | de 寄存器     | 否与 execute 即将 |
| 尚未从 alu 中算出    |           | 其之后的指令       | 阻塞         | 写入寄存器的地址      |
|                |           | 在原地等待一       |            | 一致            |
|                |           | 个周期          |            | 若一致③则进行插      |
|                |           |              |            | 入气泡和阻塞        |