# 《计算机组成原理》实验报告

| 年级、专业、班级     | 2023 级计算机科学与技术 06 班<br>2023 级计算机科学与技术 04 班 | 姓名    | 饶格奇<br>刘雨霜 |  |  |
|--------------|--------------------------------------------|-------|------------|--|--|
| 13214 113132 | 2023 级计算机科学与技术 04 班                        | 77.11 | 李隆征        |  |  |
| 实验题目         | 实验二处理器译码实验                                 |       |            |  |  |
| 实验时间         | 2025 年 4 月 27 日                            | 实验地点  | DS1410     |  |  |
|              |                                            |       | □验证性       |  |  |
| 实验成绩         | 优秀/良好/中等                                   | 实验性质  | ☑设计性       |  |  |
|              |                                            |       | □综合性       |  |  |

### 教师评价:

□算法/实验过程正确; □源程序/实验内容提交; □程序结构/实验步骤合理; □实验结果正确; □语法、语义正确; □报告规范;

其他:

评价教师: 任骜

#### 实验目的

- (1)掌握单周期 CPU 控制器的工作原理及其设计方法。
- (2)掌握单周期 CPU 各个控制信号的作用和生成过程。
- (3)掌握单周期 CPU 执行指令的过程。
- (4)掌握取指、译码阶段数据通路执行过程。

报告完成时间: 2025 年 4 月 27 日

## 1 实验内容

- 1. PC。D 触发器结构,用于储存 PC(一个周期)。需<mark>实现 2 个输入</mark>,分别为 clk, rst, 分别连接时钟和复位信号;需<mark>实现 2 个输出</mark>,分别为 pc,  $inst\_ce$ , 分别连接指令存储器的 addra, ena端口。其中 addra 位数依据 coe 文件中指令数定义;
- 2. 加法器。用于计算下一条指令地址,需<mark>实现 2 个输入,1 个输出</mark>,输入值分别为当前指令地址 PC、32'h4;
- 3. Controller。其中包含两部分:
  - (a). main\_decoder。负责判断指令类型,并生成相应的控制信号。需实现 1 个输入,为指令inst 的高 6 位 op,输出分为 2 部分,控制信号有多个,可作为多个输出,也作为一个多位输出,具体参照参考指导书进行设计; aluop,传输至 alu\_decoder,使 alu\_decoder配合 inst 低 6 位 funct,进行 ALU 模块控制信号的译码。
  - (b). alu\_decoder。负责 ALU 模块控制信号的译码。需<mark>实现 2 个输入,1 个输出,</mark>输入分别 为 funct, aluop;输出位 alucontrol 信号。
  - (c). 除上述两个组件,需设计 controller 文件调用两个 decoder,对应实现 *op,funct* 输入信号,并传入调用模块;对应实现控制信号及 *alucontrol*,并连接至调用模块相应端口。
- 4. 指令存储器。使用 Block Memory Generator IP 构造。(参考指导书) 注意: Basic 中 Generate address interface with 32 bits 选项不选中; PortA Options 中 Enable Port Type 选择为 Use ENA Pin
- 5. 时钟分频器。将板载 100Mhz 频率降低为 1hz,连接 PC、指令存储器时钟信号 clk。(参考数字逻辑实验)

注意: Xilinx Clocking Wizard IP 可分的最低频率为 4.687Mhz, 因而只能使用自实现分频模块进行分频

# 2 实验设计

#### 2.1 控制器 (Controller)

#### 2.1.1 功能描述

控制器的任务就是依据指令的 opcode 和 funct 字段译码,生成所有使能与多路选择信号,驱动数据通路各模块在一个时钟周期内正确完成对应指令的执行。

#### 2.1.2 接口定义

表 1: maindec 接口定义

| 信号名             | 方向     | 位宽    | 功能描述 |
|-----------------|--------|-------|------|
| Inst_part_31_26 | Output | 6-bit |      |
| jump            | Input  | 1-bit |      |
| branch          | Output | 1-bit |      |
| alusrc          | Output | 1-bit |      |
| memwrite        | Output | 1-bit |      |
| memtoreg        | Output | 1-bit |      |
| regwrite        | Output | 1-bit |      |
| regdst          | Output | 1-bit |      |
| ALop            | Output | 2-bit |      |

表 2: aludec 接口定义

| 信号名        | 方向     | 位宽    | 功能描述 |
|------------|--------|-------|------|
| func       | Input  | 6-bit |      |
| aluop      | Input  | 2-bit |      |
| alucontrol | Output | 3-bit |      |

#### 2.1.3 逻辑控制

- R 型指令 (opcode=000000): 设定控制信号为执行寄存器运算。
- lw (opcode=100011):设定为读取内存,并回写到寄存器。
- sw (opcode=101011):设定为寄存器数据写入内存,不进行寄存器回写。
- beq (opcode=000100): 设定为进行分支判断,使用减法 (ALU 减运算)。
- addi (opcode=001000): 设定为立即数加法运算。
- jump (opcode=000010): 设定为无条件跳转。

ALU 译码器 (aludec) 根据主译码器给出的 aluop 信号和指令低 6 位 (inst[5:0]) 中的 funct 字段进一步确定具体的 ALU 操作类型,生成精确的 ALU 控制信号 (alucontrol):

- 当 aluop=00 (例如 lw, sw, addi): 执行加法运算。
- 当 aluop=01 (例如 beq): 执行减法运算。
- 当 aluop=10 (R 型指令):根据 funct 字段选择执行加法、减法、与、或、比较等具体操作。

此外,为了简化数据通路,本实验未额外添加复杂的优化逻辑,仅采用标准单周期控制方案。

## 2.2 存储器 (Block Memory)

存储器的功能是存储 CPU 执行的指令,并根据地址(PC)输出对应地址的指令内容。

### 2.2.1 类型选择

存储器类型选择 Single Port ROM。

#### 2.2.2 参数设置

端口宽度设置为 32-bit,加载 coe 文件。



图 1: 存储器 Basic 界面



图 2: 存储器 PartAOptions 界面



图 3: 存储器 OtherOptions 界面

# 3 实验过程记录

# 4 实验结果及分析

需要仿真图一张,要求仿真图中包含 pc、指令存储器输出、8 种控制信号,PC 自增无误。

#### 4.1 仿真图

### 4.2 控制台输出

## A Controller 代码

#### A.1 controller

```
'timescale 1ns / 1ps
module controller(
input [5:0] Inst_part_31_26,
input [5:0] Inst_part_5_0,
output wire jump, branch, alusrc, memwrite, memtoreg, regwrite, regdst,
output wire [2:0] Alcontrol
    );
wire [1:0] Alop;
maindec MD(
.Inst_part_31_26(Inst_part_31_26),
.jump(jump),
.branch(branch),
.alusrc(alusrc),
.memwrite(memwrite),
.memtoreg(memtoreg),
.regwrite(regwrite),
.regdst(regdst),
.Alop(Alop)
);
aludec AL(
.Inst_part_5_0(Inst_part_5_0),
.Alop(Alop),
.Alcontrol(Alcontrol)
);
endmodule
```

### A.2 maindec

```
'timescale 1ns / 1ps
module maindec(
input [5:0] Inst_part_31_26,
output wire jump,branch,alusrc,memwrite,memtoreg,regwrite,regdst,
output wire [1:0] Alop
    );
reg [1:0] Alop_reg;
reg [6:0] Signs;
assign {regwrite,regdst,alusrc,branch,memwrite,memtoreg,jump}=Signs;
assign Alop=Alop_reg;
always @(*) begin
    case (Inst_part_31_26)
        6'b000000 : begin
                                  //R-type
             Signs <= 7 'b1100000;
             Alop_reg <= 2 'b10;
        end
        6'b100011 : begin
                                  //lw
             Signs <= 7' b1010010;
             Alop_reg <= 2 'b00;
        end
        6'b101011 : begin
                                  //sw
             Signs <= 7 'b0x101x0;
             Alop_reg <= 2, b00;
        end
        6'b000100 : begin
                                  //beq
             Signs <= 7' b0x010x0;
             Alop_reg <= 2 'b01;
        end
        6'b001000 : begin
                                  //addi
             Signs <= 7' b1010000;
             Alop_reg <= 2 'b00;
        end
        6'b000010 : begin
                                  //jump
             Signs <= 7 'b0xxx0x1;
             Alop_reg <= 2 'bxx;
        end
        default : begin
             Signs <= 7, b0000000;
             Alop_reg <= 2, b00;
        end
    endcase
end
```

endmodule

7

#### A.3 aludec

```
'timescale 1ns / 1ps
module aludec(
input wire [5:0] Inst_part_5_0,
input wire [1:0] Alop,
output wire [2:0] Alcontrol
    );
assign Alcontrol=(Alop==2', b00)?(3', b010):
                                                //lw or sw
                  (Alop==2'b01)?(3'b110):
                                                //beq
                  (Alop==2'b10)?(
                  (Inst_part_5_0 == 6'b100000)?(3'b010): //R-type
                  (Inst_part_5_0==6'b100010)?(3'b110):
                  (Inst_part_5_0==6',b100100)?(3',b000):
                  (Inst_part_5_0==6',b100101)?(3',b001):
                  (Inst_part_5_0==6'b101010)?(3'b111):(3'bxxx)
                  ):(3'bxxx);
```

endmodule

## B PC 代码

```
'timescale 1ns / 1ps
module pc(
input clk,
input rst,
output wire [31:0]pc,
output wire inst_ce
    );
reg [31:0] PC;
reg Inst;
always @(posedge clk) begin
    if(rst) begin
         PC <= 32 'b0;
         Inst <= 1 ' b0;</pre>
    end
    else begin
         Inst <= 1 ' b1;</pre>
         if (PC>=32'hffffff) begin
             PC <= 0;
         end
         else begin
             PC <= pc +4 , b0100;
         end
```

```
end
end
assign inst_ce=Inst;
assign pc=PC;
endmodule
```