

# Project: 单周期 CPU

Author: 刘雪洁 3240102846

## Part 1: 单周期 CPU 数据通路设计

### Register File

#### 功能

管理寄存器堆，对寄存器进行读写操作

#### 设计

#### 相关头文件

```
parameter xLen = 64;
parameter ADDR_WIDTH = xLen;
parameter DATA_WIDTH = xLen;
parameter MASK_WIDTH = DATA_WIDTH/8;
typedef logic [ADDR_WIDTH-1:0] addr_t;
typedef logic [DATA_WIDTH-1:0] data_t;
typedef logic [MASK_WIDTH-1:0] mask_t;
typedef logic [31:0] inst_t;
typedef logic [4:0] reg_ind_t;
```

#### 源代码

```
`include "core_struct.vh"
module RegFile (
    input clk,
    input rst,
    input we,
    input CorePack::reg_ind_t read_addr_1,
    input CorePack::reg_ind_t read_addr_2,
    input CorePack::reg_ind_t write_addr,
    input CorePack::data_t write_data,
    output CorePack::data_t read_data_1,
    output CorePack::data_t read_data_2
);
import CorePack::*;

integer i;
data_t register [1:31]; // x1 - x31, x0 keeps zero
```

```
// fill your code
assign read_data_1 = (read_addr_1 == 0) ? 0 : register[read_addr_1];
assign read_data_2 = (read_addr_2 == 0) ? 0 : register[read_addr_2];

always @(posedge clk or posedge rst) begin
    if(rst) begin
        for(i = 1; i <= 31; i = i + 1) begin
            register[i] <= 0;
        end
    end
    else begin
        if(we && (write_addr != 0)) begin
            register[write_addr] <= write_data;
        end
    end
end

endmodule
```

## ALU

### 功能

根据运算类型对数据进行算术和逻辑运算

### 数据来源

a 和 b 可能来自寄存器/立即数

### 设计

### 相关头文件

```
typedef enum logic [3:0] {
    ALU_ADD, ALU_SUB, ALU_AND, ALU_OR,
    ALU_XOR, ALU_SLT, ALU_SLTU, ALU_SLL,
    ALU_SRL, ALU_SRA, ALU_ADDW, ALU_SUBW,
    ALU_SLLW, ALU_SRLW, ALU_SRAW, ALU_DEFAULT
} alu_op_enum;
```

### 源代码

```
`include "core_struct.vh"
module ALU (
    input CorePack::data_t a,
    input CorePack::data_t b,
```

```

input CorePack::alu_op_enum alu_op,
output CorePack::data_t res
);

import CorePack::*;

// fill your code
logic [31 : 0] temp32;
// verilator lint_off LATCH
always_comb begin
    case(alu_op)
        ALU_ADD: res = a + b;
        ALU_SUB: res = a - b;
        ALU_AND: res = a & b;
        ALU_OR: res = a | b;
        ALU_XOR: res = a ^ b;
        ALU_SLT: res = ($signed(a) < $signed(b)) ? 64'b1 : 64'b0;           // $signed()
转化为有符号数
        ALU_SLTU: res = (a < b) ? 64'b1 : 64'b0;                      // verilog 默认当做无符号
数比较
        ALU_SLL: res = a << b[5:0];                                // 最多移位不超过64位, 只占6bit
        ALU_SRL: res = a >> b[5:0];
        ALU_SRA: res = $signed(a) >>> b[5:0];
        ALU_ADDW: begin
            temp32 = $signed(a[31:0]) + $signed(b[31:0]);
            res = {{32{temp32[31]}},temp32};
        end
        ALU_SUBW: begin
            temp32 = a[31:0] - b[31:0];
            res = {{32{temp32[31]}},temp32};
        end
        ALU_SLLW: begin
            temp32 = a[31:0] << b[4:0];                          // 最多移位不超过32位, 只占5bit
            res = {{32{temp32[31]}},temp32};
        end
        ALU_SRLW: begin
            temp32 = a[31:0] >> b[4:0];                      // verilog 默认逻辑右移
            res = {{32{temp32[31]}},temp32};
        end
        ALU_SRAW: begin
            temp32 = $signed(a[31:0]) >>> b[4:0];
            res = {{32{temp32[31]}},temp32};
        end
        default: res = '0;
    endcase
end
// verilator lint_on LATCH
endmodule

```

## 注意点

### 有符号数 & 无符号数

- RICS\_V指令中的计算默认操作数为**有符号数**, 如果是字型计算, 需要符号扩展; 而 Verilog 中默认为**无符号数**,
- Verilog 中 `>>` 表示逻辑右移, 表示算数右移要使用 `>>>` 并搭配 `$signed()`

## 位宽

- RISC-V 中计算指令按操作数位宽分有64位 (默认) 和32位 (指令最后有w) 两种
- 32位字型计算只有加 (减) 、移位: `add(i)w,subw,sll(i)w,srl(i)w,sra(i)w`

## 移位操作

- 双字: 低6位决定
- 字: 低5位决定

---

## Cmp

### 功能

通过逻辑和算术运算来判断 B 型指令是否满足跳转条件

### 数据来源

寄存器 (B 型指令结构决定)

### 设计

### 相关头文件

```
typedef enum logic [2:0] {
    CMP_NO,   CMP_EQ,   CMP_NE,   CMP_LT,
    CMP_GE,   CMP_LTU,  CMP_GEU,  CMP7
} cmp_op_enum;
```

### 源代码

```
`include "core_struct.vh"
module Cmp (
    input CorePack::data_t a,
    input CorePack::data_t b,
    input CorePack::cmp_op_enum cmp_op,
    output cmp_res
);

import CorePack::*;

// fill your code
logic res;
```

```
always_comb begin
    case(cmp_op)
        CMP_NO: res = 0;
        CMP_EQ: res = (a == b);
        CMP_NE: res = (a != b);
        CMP_LT: res = ($signed(a) < $signed(b));
        CMP_GE: res = ($signed(a) >= $signed(b));
        CMP_LTU: res = (a < b);
        CMP_GEU: res = (a >= b);
        CMP7: res = 0;
        default: res = 0;
    endcase
end

assign cmp_res = res;

endmodule
```

## Data package

### 功能

将写入内存的数据改为64位/8B

### 设计

#### 相关头文件

```
typedef enum logic [2:0] {
    MEM_NO, MEM_D, MEM_W, MEM_H,
    MEM_B, MEM_UB, MEM_UH, MEM_UW
} mem_op_enum;
```

#### 源代码

```
`include "core_struct.vh"

module DataPkg(
    input CorePack::mem_op_enum mem_op,
    input CorePack::data_t reg_data,
    input CorePack::addr_t dmem_waddr,
    output CorePack::data_t dmem_wdata
);

    import CorePack::*;

    // Data package
```

```
// fill your code

logic [2:0] offset;
assign offset = dmem_waddr[2:0];

always_comb begin
    dmem_wdata = '0;
    case(mem_op)
        MEM_NO: dmem_wdata = '0;
        MEM_D: dmem_wdata = reg_data;
        MEM_W, MEM_UW:begin
            case(offset[2])
                1'b0: dmem_wdata[31:0] = reg_data[31:0];
                1'b1: dmem_wdata[63:32] = reg_data[31:0];
            endcase
        end
        MEM_H, MEM_UH:begin
            case(offset[2:1])
                2'b00: dmem_wdata[15:0] = reg_data[15:0];
                2'b01: dmem_wdata[31:16] = reg_data[15:0];
                2'b10: dmem_wdata[47:32] = reg_data[15:0];
                2'b11: dmem_wdata[63:48] = reg_data[15:0];
            endcase
        end
        MEM_B, MEM_UB:begin
            case(offset)
                3'b000: dmem_wdata[7:0] = reg_data[7:0];
                3'b001: dmem_wdata[15:8] = reg_data[7:0];
                3'b010: dmem_wdata[23:16] = reg_data[7:0];
                3'b011: dmem_wdata[31:24] = reg_data[7:0];
                3'b100: dmem_wdata[39:32] = reg_data[7:0];
                3'b101: dmem_wdata[47:40] = reg_data[7:0];
                3'b110: dmem_wdata[55:48] = reg_data[7:0];
                3'b111: dmem_wdata[63:56] = reg_data[7:0];
            endcase
        end
        default: dmem_wdata = '0;
    endcase
end

endmodule
```

## 注意点

- store指令不区分有符号数和无符号数，直接截断，不做扩展，多余位是0
- 偏移量直接截取低3位，即为地址除以8的余数

## Data Mask Generation

### 功能

传回的数据中可能包含大量无意义仅用于占位的 0， 使用掩码 (Mask) 来告知内存哪些位是有意义的，只有掩码中为 1 的字节才会被载入内存。

## 设计

```
`include "core_struct.vh"

module MaskGen(
    input CorePack::mem_op_enum mem_op,
    input CorePack::addr_t dmem_waddr,
    output CorePack::mask_t dmem_wmask
);

import CorePack::*;

// Mask generation
// fill your code
logic [2:0] offset;
assign offset = dmem_waddr[2:0];

always_comb begin
    case(mem_op)
        MEM_NO: dmem_wmask = '0;
        MEM_D: dmem_wmask = '1;
        MEM_W, MEM_UW:begin
            case(offset[2])
                1'b0: dmem_wmask = 8'b00001111;
                1'b1: dmem_wmask = 8'b11110000;
            endcase
        end
        MEM_H, MEM_UH:begin
            case(offset[2:1])
                2'b00: dmem_wmask = 8'b00000011;
                2'b01: dmem_wmask = 8'b00001100;
                2'b10: dmem_wmask = 8'b00110000;
                2'b11: dmem_wmask = 8'b11000000;
            endcase
        end
        MEM_B, MEM_UB:begin
            case(offset)
                3'b000: dmem_wmask = 8'b00000001;
                3'b001: dmem_wmask = 8'b00000010;
                3'b010: dmem_wmask = 8'b00000100;
                3'b011: dmem_wmask = 8'b00001000;
                3'b100: dmem_wmask = 8'b00010000;
                3'b101: dmem_wmask = 8'b00100000;
                3'b110: dmem_wmask = 8'b01000000;
                3'b111: dmem_wmask = 8'b10000000;
            endcase
        end
        default: dmem_wmask = '0;
    endcase
end
```

```
end  
endmodule
```

## Data Trunction

### 功能

将从内存中读取到的数据改为64位/8B

### 设计

```
`include "core_struct.vh"  
  
module DataTrunc (  
    input CorePack::data_t dmem_rdata,  
    input CorePack::mem_op_enum mem_op,  
    input CorePack::addr_t dmem_raddr,  
    output CorePack::data_t read_data  
);  
  
import CorePack::*;  
  
// Data trunction  
// fill your code  
logic [2:0] offset;  
assign offset = dmem_raddr[2:0];  
  
always_comb begin  
    case(mem_op)  
        MEM_NO: read_data = '0;  
        MEM_D: read_data = dmem_rdata;  
        MEM_W: begin  
            case(offset[2])  
                1'b0: read_data = {{32{dmem_rdata[31]}}, dmem_rdata[31:0]};  
                1'b1: read_data = {{32{dmem_rdata[63]}}, dmem_rdata[63:32]};  
            endcase  
        end  
        MEM_UW:begin  
            case(offset[2])  
                1'b0: read_data = {32'b0, dmem_rdata[31:0]};  
                1'b1: read_data = {32'b0, dmem_rdata[63:32]};  
            endcase  
        end  
        MEM_H: begin  
            case(offset[2:1])  
                2'b00: read_data = {{48{dmem_rdata[15]}}, dmem_rdata[15:0]};  
                2'b01: read_data = {{48{dmem_rdata[31]}}, dmem_rdata[31:16]};  
                2'b10: read_data = {{48{dmem_rdata[47]}}, dmem_rdata[47:32]};  
                2'b11: read_data = {{48{dmem_rdata[63]}}, dmem_rdata[63:48]};  
            endcase  
        end  
    endcase  
end
```

```

        endcase
    end
    MEM_UH:begin
        case(offset[2:1])
            2'b00: read_data = {48'b0, dmem_rdata[15:0]};
            2'b01: read_data = {48'b0, dmem_rdata[31:16]};
            2'b10: read_data = {48'b0, dmem_rdata[47:32]};
            2'b11: read_data = {48'b0, dmem_rdata[63:48]};
        endcase
    end
    MEM_B: begin
        case(offset)
            3'b000: read_data = {{56{dmem_rdata[7]}}, dmem_rdata[7:0]};
            3'b001: read_data = {{56{dmem_rdata[15]}}, dmem_rdata[15:8]};
            3'b010: read_data = {{56{dmem_rdata[23]}}, dmem_rdata[23:16]};
            3'b011: read_data = {{56{dmem_rdata[31]}}, dmem_rdata[31:24]};
            3'b100: read_data = {{56{dmem_rdata[39]}}, dmem_rdata[39:32]};
            3'b101: read_data = {{56{dmem_rdata[47]}}, dmem_rdata[47:40]};
            3'b110: read_data = {{56{dmem_rdata[55]}}, dmem_rdata[55:48]};
            3'b111: read_data = {{56{dmem_rdata[63]}}, dmem_rdata[63:56]};
        endcase
    end
    MEM_UB:begin
        case(offset)
            3'b000: read_data = {56'b0, dmem_rdata[7:0]};
            3'b001: read_data = {56'b0, dmem_rdata[15:8]};
            3'b010: read_data = {56'b0, dmem_rdata[23:16]};
            3'b011: read_data = {56'b0, dmem_rdata[31:24]};
            3'b100: read_data = {56'b0, dmem_rdata[39:32]};
            3'b101: read_data = {56'b0, dmem_rdata[47:40]};
            3'b110: read_data = {56'b0, dmem_rdata[55:48]};
            3'b111: read_data = {56'b0, dmem_rdata[63:56]};
        endcase
    end
    default: read_data = '0;
endcase
end

endmodule

```

## 注意点

- load 指令区分有符号数和无符号数 有符号数符号扩展，无符号数零扩展

## 数据通路 Core

### 功能

集成上面实现的这些功能模块

# 设计

## 相关头文件

```
typedef struct{
    logic [63:0] pc;
    logic [63:0] inst;
    logic [63:0] rs1_id;
    logic [63:0] rs1_data;
    logic [63:0] rs2_id;
    logic [63:0] rs2_data;
    logic [63:0] alu;
    logic [63:0] mem_addr;
    logic [63:0] mem_we;
    logic [63:0] mem_wdata;
    logic [63:0] mem_rdata;
    logic [63:0] rd_we;
    logic [63:0] rd_id;
    logic [63:0] rd_data;
    logic [63:0] br_taken;
    logic [63:0] npc;
} CoreInfo;
```

## 源代码

```
`include "core_struct.vh"
module Core (
    input clk,
    input rst,
    Mem_ift.Master imem_ift,
    Mem_ift.Master dmem_ift,
    output cosim_valid,
    output CorePack::CoreInfo cosim_core_info
);
    import CorePack::*;

    // fill your code

    CorePack::inst_t inst;
    CorePack::data_t inst_data;
    CorePack::data_t modified_dmem_rdata;

    CorePack::data_t pc;
    CorePack::data_t next_pc;
    logic br_taken;

    logic we_reg;
```

```
logic we_mem;
logic re_mem;
logic npc_sel;
CorePack::imm_op_enum immgen_op;
CorePack::alu_op_enum alu_op;
CorePack::cmp_op_enum cmp_op;
CorePack::alu_asel_op_enum alu_asel;
CorePack::alu_bsel_op_enum alu_bsel;
CorePack::wb_sel_op_enum wb_sel;
CorePack::mem_op_enum mem_op;

CorePack::data_t a;
CorePack::data_t b;
CorePack::data_t alu_res;

CorePack::reg_ind_t rs1;
CorePack::reg_ind_t rs2;
CorePack::reg_ind_t rd;
CorePack::data_t read_data_1;
CorePack::data_t read_data_2;
CorePack::data_t wb_val;

logic [31:0] imm;

// IF

assign imem_ift.w_request_bits.wdata = '0;
assign imem_ift.w_request_bits.wmask = '0;
assign imem_ift.w_request_bits.waddr = '0;
assign imem_ift.w_request_valid = 0;
assign imem_ift.w_reply_ready = 0; //写通道常关

assign imem_ift.r_request_bits.raddr = pc;
assign imem_ift.r_request_valid = 1; //读请求通道打开

assign imem_ift.r_reply_ready = 1;
assign inst_data = imem_ift.r_reply_bits.rdata; //读响应通道打开
assign inst = pc[2] ? inst_data[63:32] : inst_data[31:0];

always@(posedge clk or posedge rst) begin
    if(rst) pc <= '0;
    else pc <= next_pc;
end

always_comb begin
    if(npc_sel | br_taken) next_pc = alu_res;
    else next_pc = pc + 4;
end

// ID
assign rs1 = inst[19:15];
assign rs2 = inst[24:20];
```

```
assign rd = inst[11:7];

controller Cont(
    .inst(inst),
    .we_reg(we_reg),
    .we_mem(we_mem),
    .re_mem(re_mem),
    .npc_sel(npc_sel),
    .immgen_op(immgen_op),
    .alu_op(alu_op),
    .cmp_op(cmp_op),
    .alu_asel(alu_asel),
    .alu_bsel(alu_bsel),
    .wb_sel(wb_sel),
    .mem_op(mem_op)
);

//EXE

RegFile reg_file(          // 读取rs1,rs2数据(ID) & 写回rd(EXE), 可以共享同一个物理寄存器文件
    .clk(clk),
    .rst(rst),
    .we(we_reg),
    .read_addr_1(rs1),
    .read_addr_2(rs2),
    .write_addr(rd),
    .write_data(wb_val),
    .read_data_1(read_data_1),
    .read_data_2(read_data_2)
);

ALU alu(
    .a(a),
    .b(b),
    .alu_op(alu_op),
    .res(alu_res)
);

Cmp cmp(
    .a(read_data_1),
    .b(read_data_2),
    .cmp_op(cmp_op),
    .cmp_res(br_taken)
);

always_comb begin
    case(immgen_op)
        I_IMM: imm = {{20{inst[31]}}, inst[31:20]};
        S_IMM: imm = {{20{inst[31]}}, inst[31:25], inst[11:7]};
        B_IMM: imm = {{19{inst[31]}}, inst[31], inst[7], inst[30:25],
inst[11:8], 1'b0};
        U_IMM: imm = {inst[31:12], 12'b0};
    endcase
end
```

```
UJ_IMM: imm = {{11{inst[31]}}, inst[31], inst[19:12], inst[20],
inst[30:21], 1'b0};
      default: imm = '0;
endcase
end

always_comb begin
  case(wb_sel)
    WB_SEL0: wb_val = 0;
    WB_SEL_ALU: wb_val = alu_res;
    WB_SEL_MEM: wb_val = modified_dmem_rdata;
    WB_SEL_PC: wb_val = pc + 4;
  endcase
end

always_comb begin
  case(alu_asel)
    ASEL_REG: a = read_data_1;
    ASEL_PC: a = pc;
    default: a = '0;
  endcase
  case(alu_bsel)
    BSEL_REG: b = read_data_2;
    BSEL_IMM: b = {{32{imm[31]}},imm};           //立即数当做有符号数
    default: b = '0;
  endcase
end
//MEM

//读(load)
assign dmem_ift.r_request_valid = re_mem;
assign dmem_ift.r_reply_ready = re_mem;
assign dmem_ift.r_request_bits.raddr = alu_res;

DataTrunc trunc(      // 读取的数据处理
  .dmem_rdata(dmem_ift.r_reply_bits.rdata),
  .mem_op(mem_op),
  .dmem_raddr(alu_res),
  .read_data(modified_dmem_rdata)
);

//写(store)
assign dmem_ift.w_request_valid = we_mem;
assign dmem_ift.w_reply_ready = we_mem;
assign dmem_ift.w_request_bits.waddr = alu_res;

DataPkg pkg(      // 写入的数据处理
  .mem_op(mem_op),
  .reg_data(read_data_2),
  .dmem_waddr(alu_res),
  .dmem_wdata(dmem_ift.w_request_bits.wdata)
);
```

```

MaskGen mask(
    .mem_op(mem_op),
    .dmem_waddr(alu_res),
    .dmem_wmask(dmem_ift.w_request_bits.wmask)
);

assign cosim_valid = 1'b1;

assign cosim_valid = 1'b1;
assign cosim_core_info.pc      = pc;
assign cosim_core_info.inst   = {32'b0,inst};
assign cosim_core_info.rs1_id  = {59'b0, rs1};
assign cosim_core_info.rs1_data = read_data_1;
assign cosim_core_info.rs2_id  = {59'b0, rs2};
assign cosim_core_info.rs2_data = read_data_2;
assign cosim_core_info.alu     = alu_res;
assign cosim_core_info.mem_addr = dmem_ift.r_request_bits.raddr;
assign cosim_core_info.mem_we   = {63'b0, dmem_ift.w_request_valid};
assign cosim_core_info.mem_wdata = dmem_ift.w_request_bits.wdata;
assign cosim_core_info.mem_rdata = dmem_ift.r_reply_bits.rdata;
assign cosim_core_info.rd_we   = {63'b0, we_reg};
assign cosim_core_info.rd_id   = {59'b0, rd};
assign cosim_core_info.rd_data  = wb_val;
assign cosim_core_info.br_taken = {63'b0, br_taken};
assign cosim_core_info.npc     = next_pc;

endmodule

```

## Part 2: 单周期 CPU 设计

### RV64I 指令格式

RV64I 是 RISC-V 的 64 位基础整数指令集，包含 6 种基本指令格式，所有指令长度均为 **32 位**。

| 格式 | 名称        | 主要用途                                             | 指令示例                                                                                                       | 字段结构 (位) |
|----|-----------|--------------------------------------------------|------------------------------------------------------------------------------------------------------------|----------|
| R  | Register  | 寄存器-寄存器操作<br>用于算术逻辑运算<br>(ADD/SUB/XOR等)          | add rd, [31:25] funct7   [24:20] rs2   [19:15] rs1<br>rs1, rs2   [14:12] funct3   [11:7] rd   [6:0] opcode |          |
| I  | Immediate | 加载指令 (LOAD) 、<br>立即数运算<br>(ADDI) 、跳转链接<br>(JALR) | addi rd, [31:20] imm[11:0]   [19:15] rs1   [14:12]<br>rs1, imm funct3   [11:7] rd   [6:0] opcode           |          |

| 格式 | 名称     | 主要用途                                            | 指令示例                 | 字段结构 (位)                                                                                                                           |
|----|--------|-------------------------------------------------|----------------------|------------------------------------------------------------------------------------------------------------------------------------|
| S  | Store  | 存储指令                                            | sw rs2,<br>imm(rs1)  | [31:25] imm[11:5]   [24:20] rs2   [19:15]<br>rs1   [14:12] funct3   [11:7] imm[4:0]  <br>[6:0] opcode                              |
| B  | Branch | 条件分支跳转                                          | beq rs1,<br>rs2, imm | [31] imm[12]   [30:25] imm[10:5]   [24:20]<br>rs2   [19:15] rs1   [14:12] funct3   [11:8]<br>imm[4:1]   [7] imm[11]   [6:0] opcode |
| U  | Upper  | 长立即数 (高20位)<br>用于 LUI (加载高位立即数)、AUIPC (PC 相对地址) | lui rd,<br>imm       | [31:12] imm[31:12]   [11:7] rd   [6:0]<br>opcode                                                                                   |
| J  | Jump   | 无条件长跳转<br>用于 JAL (跳转并链接)                        | jal rd,<br>imm       | [31] imm[20]   [30:21] imm[10:1]   [20]<br>imm[11]   [19:12] imm[19:12]   [11:7] rd  <br>[6:0] opcode                              |

## RV64I 立即数存储方式

| 指令类型 | 立即数位数 | 存储字段                            | 实际值计算                                           | 典型指令          |
|------|-------|---------------------------------|-------------------------------------------------|---------------|
| I    | 12    | [31:20] (连续)                    | imm[11:0]                                       | addi,<br>ld   |
| S    | 12    | [31:25] + [11:7] (分两段)          | {imm[11:5], imm[4:0]}                           | sd, sw        |
| B    | 12    | [31] + [30:25] + [11:8] + [7]   | {imm[12], imm[11], imm[10:5],<br>imm[4:1], 0}   | beq,<br>bne   |
| U    | 20    | [31:12] (高 20 位)                | imm[31:12] << 12                                | lui,<br>auipc |
| J    | 20    | [31] + [30:21] + [20] + [19:12] | {imm[20], imm[19:12], imm[11],<br>imm[10:1], 0} | jal           |

## RV64I 指令编码与控制信号表

## RV64I 指令编码表

| 指令          | Opcode (7-bit) | funct3 (3-bit) | funct7 (7-bit) |
|-------------|----------------|----------------|----------------|
| <b>加载指令</b> |                |                |                |
| lb          | 0000011        | 000            | -              |
| lh          | 0000011        | 001            | -              |

| 指令  | Opcode (7-bit) | funct3 (3-bit) | funct7 (7-bit) |
|-----|----------------|----------------|----------------|
| lw  | 0000011        | 010            | -              |
| ld  | 0000011        | 011            | -              |
| lbu | 0000011        | 100            | -              |
| lhu | 0000011        | 101            | -              |
| lwu | 0000011        | 110            | -              |

**存储指令**

|    |         |     |   |
|----|---------|-----|---|
| sb | 0100011 | 000 | - |
| sh | 0100011 | 001 | - |
| sw | 0100011 | 010 | - |
| sd | 0100011 | 011 | - |

**立即数运算**

|       |         |     |         |
|-------|---------|-----|---------|
| addi  | 0010011 | 000 | -       |
| slli  | 0010011 | 001 | 0000000 |
| srlti | 0010011 | 101 | 0000000 |
| srai  | 0010011 | 101 | 0100000 |
| andi  | 0010011 | 111 | -       |
| ori   | 0010011 | 110 | -       |
| xori  | 0010011 | 100 | -       |
| addiw | 0011011 | 000 | -       |
| slliw | 0011011 | 001 | 0000000 |
| srliw | 0011011 | 101 | 0000000 |
| sraiw | 0011011 | 101 | 0100000 |

**寄存器运算**

|      |         |     |         |
|------|---------|-----|---------|
| add  | 0110011 | 000 | 0000000 |
| sub  | 0110011 | 000 | 0100000 |
| sll  | 0110011 | 001 | 0000000 |
| slt  | 0110011 | 010 | 0000000 |
| sltu | 0110011 | 011 | 0000000 |
| xor  | 0110011 | 100 | 0000000 |
| srl  | 0110011 | 101 | 0000000 |

| 指令   | Opcode (7-bit) | funct3 (3-bit) | funct7 (7-bit) |
|------|----------------|----------------|----------------|
| sra  | 0110011        | 101            | 0100000        |
| or   | 0110011        | 110            | 0000000        |
| and  | 0110011        | 111            | 0000000        |
| addw | 0111011        | 000            | 0000000        |
| subw | 0111011        | 000            | 0100000        |
| sllw | 0111011        | 001            | 0000000        |
| srlw | 0111011        | 101            | 0000000        |
| sraw | 0111011        | 101            | 0100000        |

**分支指令**

|      |         |     |   |
|------|---------|-----|---|
| beq  | 1100011 | 000 | - |
| bne  | 1100011 | 001 | - |
| blt  | 1100011 | 100 | - |
| bge  | 1100011 | 101 | - |
| bltu | 1100011 | 110 | - |
| bgeu | 1100011 | 111 | - |

**其他指令**

|       |         |     |   |
|-------|---------|-----|---|
| lui   | 0110111 | -   | - |
| auipc | 0010111 | -   | - |
| jal   | 1101111 | -   | - |
| jalr  | 1100111 | 000 | - |

## 控制单元 Controller

### 输出信号

| 类型          | 名称        | 宽度 | 含义         |
|-------------|-----------|----|------------|
| logic       | we_reg    | 1  | 寄存器写使能     |
| logic       | we_mem    | 1  | 内存写使能      |
| logic       | re_mem    | 1  | 内存读使能      |
| logic       | npc_sel   | 1  | NEXT PC 选择 |
| imm_op_enum | immgen_op | 3  | 立即数选择      |

| 类型               | 名称       | 宽度 | 含义         |
|------------------|----------|----|------------|
| alu_op_enum      | alu_op   | 4  | 运算操作码      |
| cmp_op_enum      | cmp_op   | 3  | 分支计算操作码    |
| alu_asel_op_enum | alu_asel | 2  | ALU A 端口选择 |
| alu_bsel_op_enum | alu_bsel | 2  | ALU B 端口选择 |
| wb_sel_op_enum   | wb_sel   | 2  | 写回数据选择     |
| mem_op_enum      | mem_op   | 3  | 访存操作码      |

## 设计思路

| 指令                  | we_mem | re_mem | mem_op | npc_sel               | alu           | alu_res     | cmp  | wb      |
|---------------------|--------|--------|--------|-----------------------|---------------|-------------|------|---------|
| <b>LOAD</b>         |        |        |        | case(宽度&无符号)          |               | 访存地址        |      |         |
| lw<br>rd,imm(rs1)   |        | 1      |        |                       | add(rs1,imm)  | (读)         |      | 内存读的数据  |
| <b>STORE</b>        |        |        |        | case(宽度)              | add(rs1,imm)  | 访存地址        |      |         |
| sw<br>rs2,imm(rs1)  |        | 1      |        |                       |               | (写)         |      |         |
| <b>IMM</b>          |        |        |        |                       | case(rs1,imm) | 写回rd        |      | alu_res |
| addi<br>rd,rs1,imm  |        |        |        |                       |               |             |      |         |
| <b>IMMW</b>         |        |        |        |                       | case(rs1,imm) | 写回rd        |      | alu_res |
| addiw<br>rd,rs1,imm |        |        |        |                       |               |             |      |         |
| <b>REG</b>          |        |        |        |                       | case(rs1,rs2) | 写回rd        |      | alu_res |
| add<br>rd,rs1,rs2   |        |        |        |                       |               |             |      |         |
| <b>REGW</b>         |        |        |        |                       | case(rs1,rs2) | 写回rd        |      | alu_res |
| add<br>rd,rs1,rs2   |        |        |        |                       |               |             |      |         |
| <b>BRANCH</b>       |        |        |        | △这里赋0,在数据通路判断后再决定是否跳转 | add(PC,imm)   | 跳转地址 (如果跳转) | case |         |
| bne<br>rs1,rs2,imm  |        |        |        |                       |               |             |      |         |
| <b>JAL</b>          |        |        | 1      |                       | add(PC,imm)   | 跳转地址        |      | PC+4    |
| jar rd,imm          |        |        |        |                       |               |             |      |         |

| 指令                  | we_mem | re_mem | mem_op | npc_sel | alu          | alu_res | cmp | wb      |
|---------------------|--------|--------|--------|---------|--------------|---------|-----|---------|
| <b>JALR</b>         |        |        |        |         |              |         |     |         |
| jalr<br>rd,imm(rs1) |        |        |        | 1       | add(rs1,imm) | 跳转地址    |     | PC+4    |
| <b>LUI</b>          |        |        |        |         |              |         |     |         |
| lui rd,imm'         |        |        |        |         | add(0,imm)   | 写回rd    |     | alu_res |
| <b>AUIPC</b>        |        |        |        |         |              |         |     |         |
| auipc,rd,imm        |        |        |        |         | add(PC,imm)  | 写回rd    |     | alu_res |

## 源代码

```

`include "core_struct.vh"
module controller (
    input CorePack::inst_t inst,
    output we_reg,
    output we_mem,
    output re_mem,
    output npc_sel,
    output CorePack::imm_op_enum immgen_op,
    output CorePack::alu_op_enum alu_op,
    output CorePack::cmp_op_enum cmp_op,
    output CorePack::alu_asel_op_enum alu_asel,
    output CorePack::alu_bsel_op_enum alu_bsel,
    output CorePack::wb_sel_op_enum wb_sel,
    output CorePack::mem_op_enum mem_op
    // output ControllerPack::ControllerSignals ctrl_signals
);

import CorePack::*;
// import ControllerPack::*;

// fill your code

logic [6:0] opcode = inst[6:0];
logic [2:0] funct3 = inst[14:12];
logic [6:0] funct7 = inst[31:25];
logic load = opcode == LOAD_OPCODE;
logic store = opcode == STORE_OPCODE;
logic immcal = opcode == IMM_OPCODE;
logic regcal = opcode == REG_OPCODE;
logic branch = opcode == BRANCH_OPCODE;
logic jal = opcode == JAL_OPCODE;
logic jalr = opcode == JALR_OPCODE;
logic auipc = opcode == AUIPC_OPCODE;
logic lui = opcode == LUI_OPCODE;
logic immw = opcode == IMMW_OPCODE;
logic regw = opcode == REGW_OPCODE;

always_comb begin

```

```

we_reg = load|immcal|regcal|jal|jalr|auipc|immw|regw|lui;
we_mem = store;
re_mem = load;
npc_sel = jal|jalr;
immgen_op = IMM0;
alu_op = ALU_DEFAULT;
cmp_op = CMP_NO; //B指令类型, 仅B指令改变
alu_asel = ASEL0;
alu_bsel = BSEL0;
wb_sel = WB_SEL0; //写回寄存器的来源
mem_op = MEM_NO; //访问内存(读/写)操作码, 仅LOAD, STORE指令

```

改变

```

case(opcode)
    LOAD_OPCODE: begin
        immgen_op = I_IMM;
        alu_op = ALU_ADD;
        alu_asel = ASEL_REG;
        alu_bsel = BSEL_IMM;
        wb_sel = WB_SEL_MEM;
        case(func3)
            LB_FUNCT3: mem_op = MEM_B;
            LH_FUNCT3: mem_op = MEM_H;
            LW_FUNCT3: mem_op = MEM_W;
            LD_FUNCT3: mem_op = MEM_D;
            LBU_FUNCT3: mem_op = MEM_UB;
            LHU_FUNCT3: mem_op = MEM_UH;
            LWU_FUNCT3: mem_op = MEM_UW;
            default: mem_op = MEM_NO;
        endcase
    end
    STORE_OPCODE: begin
        immgen_op = S_IMM;
        alu_op = ALU_ADD;
        alu_asel = ASEL_REG;
        alu_bsel = BSEL_IMM;
        case(func3)
            SB_FUNCT3: mem_op = MEM_B;
            SH_FUNCT3: mem_op = MEM_H;
            SW_FUNCT3: mem_op = MEM_W;
            SD_FUNCT3: mem_op = MEM_D;
            default: mem_op = MEM_NO;
        endcase
    end
    IMM_OPCODE: begin
        immgen_op = I_IMM;
        alu_asel = ASEL_REG;
        alu_bsel = BSEL_IMM;
        wb_sel = WB_SEL_ALU;
        case(func3)
            ADD_FUNCT3: alu_op = ALU_ADD;
            SLL_FUNCT3: alu_op = ALU_SLL;
            SLT_FUNCT3: alu_op = ALU_SLT;
            SLTU_FUNCT3: alu_op = ALU_SLTU;
        endcase
    end
end

```

```
XOR_FUNCT3: alu_op = ALU_XOR;
SRL_FUNCT3: alu_op = (funct7 == 0) ? ALU_SRL : ALU_SRA;
OR_FUNCT3: alu_op = ALU_OR;
AND_FUNCT3: alu_op = ALU_AND;
default: alu_op = ALU_DEFAULT;
endcase
end
IMMW_OPCODE: begin
    immgen_op = I_IMM;
    alu_asel = ASEL_REG;
    alu_bsel = BSEL_IMM;
    wb_sel = WB_SEL_ALU;
    case(funct3)
        ADD_FUNCT3: alu_op = ALU_ADDW;
        SLL_FUNCT3: alu_op = ALU_SLLW;
        SRL_FUNCT3: alu_op = (funct7 == 0) ? ALU_SRLW : ALU_SRAW;
        default: alu_op = ALU_DEFAULT;
    endcase
end
REG_OPCODE: begin
    alu_asel = ASEL_REG;
    alu_bsel = BSEL_REG;
    wb_sel = WB_SEL_ALU;
    case(funct3)
        ADD_FUNCT3: alu_op = (funct7 == 0) ? ALU_ADD : ALU_SUB;
        SLL_FUNCT3: alu_op = ALU_SLL;
        SLT_FUNCT3: alu_op = ALU_SLT;
        SLTU_FUNCT3: alu_op = ALU_SLTU;
        XOR_FUNCT3: alu_op = ALU_XOR;
        SRL_FUNCT3: alu_op = (funct7 == 0) ? ALU_SRL : ALU_SRA;
        OR_FUNCT3: alu_op = ALU_OR;
        AND_FUNCT3: alu_op = ALU_AND;
        default: alu_op = ALU_DEFAULT;
    endcase
end
REGW_OPCODE: begin
    alu_asel = ASEL_REG;
    alu_bsel = BSEL_REG;
    wb_sel = WB_SEL_ALU;
    case(funct3)
        ADD_FUNCT3: alu_op = (funct7 == 0) ? ALU_ADDW : ALU_SUBW;
        SLL_FUNCT3: alu_op = ALU_SLLW;
        SRL_FUNCT3: alu_op = (funct7 == 0) ? ALU_SRLW : ALU_SRAW;
        default: alu_op = ALU_DEFAULT;
    endcase
end
BRANCH_OPCODE: begin
    immgen_op = B_IMM;
    alu_op = ALU_ADD;
    alu_asel = ASEL_PC;
    alu_bsel = BSEL_IMM;
    case(funct3)
        BEQ_FUNCT3: cmp_op = CMP_EQ;
        BNE_FUNCT3: cmp_op = CMP_NE;
```

```
    BLT_FUNCT3: cmp_op = CMP_LT;
    BGE_FUNCT3: cmp_op = CMP_GE;
    BLTU_FUNCT3: cmp_op = CMP_LTU;
    BGEU_FUNCT3: cmp_op = CMP_GEU;
    default: cmp_op = CMP_NO;
endcase
end

JAL_OPCODE: begin
    immgen_op = UJ_IMM;
    alu_op = ALU_ADD;
    alu_asel = ASEL_PC;
    alu_bsel = BSEL_IMM;
    wb_sel = WB_SEL_PC;
end

JALR_OPCODE: begin
    immgen_op = I_IMM;
    alu_op = ALU_ADD;
    alu_asel = ASEL_REG;
    alu_bsel = BSEL_IMM;
    wb_sel = WB_SEL_PC;
end

AUIPC_OPCODE: begin
    immgen_op = U_IMM;
    alu_op = ALU_ADD;
    alu_asel = ASEL_PC;
    alu_bsel = BSEL_IMM;
    wb_sel = WB_SEL_ALU;
end

LUI_OPCODE: begin
    immgen_op = U_IMM;
    alu_op = ALU_ADD;
    alu_asel = ASEL0;
    alu_bsel = BSEL_IMM;
    wb_sel = WB_SEL_ALU;
end

default: begin
    immgen_op = IMM0;
    alu_op = ALU_DEFAULT;
    cmp_op = CMP_NO;
    alu_asel = ASEL0;
    alu_bsel = BSEL0;
    wb_sel = WB_SEL0;
    mem_op = MEM_NO;
end
endcase

end
endmodule
```

## 仿真测试

## R Type

```

core 0: j 0x0000000000000002d0 (0x0000070b3) x1 0x0000000000000000
core 0: >>> test_56
core 0: 0x0000000000000002d4 (0x03800193) li      gp, 56
core 0: 3 0x0000000000000002d4 (0x03800193) x3 0x00000000000000038
core 0: 0x0000000000000002d8 (0x01100093) li      ra, 17
core 0: 3 0x0000000000000002d8 (0x01100093) x1 0x00000000000000011
core 0: 0x0000000000000002dc (0x02200113) li      sp, 34
core 0: 3 0x0000000000000002dc (0x02200113) x2 0x00000000000000022
core 0: 0x0000000000000002e0 (0x0020f033) and    zero, ra, sp
core 0: 3 0x0000000000000002e0 (0x0020f033)
core 0: 0x0000000000000002e4 (0x00301663) bne    zero, gp, pc + 12
core 0: 3 0x0000000000000002e4 (0x00301663)
core 0: >>> pass
core 0: 0x0000000000000002f0 (0x00000093) li      ra, 0
core 0: 3 0x0000000000000002f0 (0x00000093) x1 0x0000000000000000
core 0: 0x0000000000000002f4 (0xc001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000002f4
core 0:          tval 0x00000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3 0x0000000000000001
[error] PC SIM 0000000000000000, DUT 00000000000002f4
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x000000000000022 gp = 0x0000000000000001
tp = 0x0000000000000000 t0 = 0x0000000000000000 t1 = 0x0000000000000000 t2 = 0x0000000000000000
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000000000000 a5 = 0x0000000000000000
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```





## I Type

```

core 0: >>> test_60
core 0: 0x000000000000002cc (0x03c00193) li      gp, 60
core 0: 3 0x000000000000002cc (0x03c00193) x3  0x000000000000003c
core 0: 0x000000000000002d0 (0x00000097) auipc   ra, 0x0
core 0: 3 0x000000000000002d0 (0x00000097) x1  0x000000000000002d0
core 0: 0x000000000000002d4 (0x0580b083) ld      ra, 88(ra)
core 0: 3 0x000000000000002d4 (0x0580b083) x1  0x000000000000300 mem 0x00000000000000328
core 0: 0x000000000000002d8 (0xffd08093) addi    ra, ra, -3
core 0: 3 0x000000000000002d8 (0xffd08093) x1  0x000000000000002fd
core 0: 0x000000000000002dc (0x00b0b283) ld      t0, 11(ra)
core 0: 3 0x000000000000002dc (0x00b0b283) x5  0x000000000000ab mem 0x00000000000000308
core 0: 0x000000000000002e0 (0x00301663) bne     zero, gp, pc + 12
core 0: 3 0x000000000000002e0 (0x00301663)
core 0: >>> pass
core 0: 0x000000000000002ec (0x00000093) li      ra, 0
core 0: 3 0x000000000000002ec (0x00000093) x1  0x0000000000000000
core 0: 0x000000000000002f0 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x000000000000002f0
core 0:           tval 0x00000000c0001073
core 0: >>>
core 0: 0x000000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x000000000000000000 (0x00100193) x3  0x0000000000000001
[error] PC SIM 0000000000000000, DUT 000000000000002f0
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x0000000000000000 gp = 0x0000000000000001
tp = 0x0000000000000000 t0 = 0x00000000000000ab t1 = 0x0000000000000000 t2 = 0x0000000000000000
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x00000000000000cd a5 = 0x00000000000000cd
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```

## S Type

```
core 0: 3 0x0000000000000001cc (0x00729463)
core 0: 0x0000000000000001d0 (0x00301663) bne      zero, gp, pc + 12
core 0: 3 0x0000000000000001d0 (0x00301663)
core 0: >>>  pass
core 0: 0x0000000000000001dc (0x00000093) li      ra, 0
core 0: 3 0x0000000000000001dc (0x00000093) x1  0x0000000000000000
core 0: 0x0000000000000001e0 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000001e0
core 0:          tval 0x00000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3  0x0000000000000001
[error] PC SIM 0000000000000000, DUT 0000000000000001e0
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x000000000000bd gp = 0x0000000000000001
tp = 0x0000000000000238 t0 = 0x000000000000bd t1 = 0x0000000000000000 t2 = 0x000000000000bd
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x00000000000009a a5 = 0x0000000000000174
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed
```

## B Type

```

core 0: >>> test_44
core 0: 0x00000000000000510 (0x02c00193) li      gp, 44
core 0: 3 0x00000000000000510 (0x02c00193) x3 0x000000000000002c
core 0: 0x00000000000000514 (0xffff00093) li      ra, -1
core 0: 3 0x00000000000000514 (0xffff00093) x1 0xfffffffffffffff
core 0: 0x00000000000000518 (0x00000113) li      sp, 0
core 0: 3 0x00000000000000518 (0x00000113) x2 0x0000000000000000
core 0: 0x0000000000000051c (0x0020e463) bltu    ra, sp, pc + 8
core 0: 3 0x0000000000000051c (0x0020e463)
core 0: 0x00000000000000520 (0x00301463) bne     zero, gp, pc + 8
core 0: 3 0x00000000000000520 (0x00301463)
core 0: 0x00000000000000528 (0xfe20eee3) bltu    ra, sp, pc - 4
core 0: 3 0x00000000000000528 (0xfe20eee3)
core 0: 0x0000000000000052c (0x00301663) bne     zero, gp, pc + 12
core 0: 3 0x0000000000000052c (0x00301663)
core 0: >>> pass
core 0: 0x00000000000000538 (0x00000093) li      ra, 0
core 0: 3 0x00000000000000538 (0x00000093) x1 0x0000000000000000
core 0: 0x0000000000000053c (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000053c
core 0:           tval 0x00000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3 0x0000000000000001
[error] PC SIM 0000000000000000, DUT 0000000000000053c
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x0000000000000000 gp = 0x0000000000000001
tp = 0x0000000000000000 t0 = 0x0000000000000000 t1 = 0x0000000000000000 t2 = 0x0000000000000000
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000000000000 a5 = 0x0000000000000000
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```

## U Type

```

core 0: 0x0000000000000098 (0x00301663) bne     zero, gp, pc + 12
core 0: 3 0x0000000000000098 (0x00301663)
core 0: >>> pass
core 0: 0x00000000000000a4 (0x00000093) li      ra, 0
core 0: 3 0x00000000000000a4 (0x00000093) x1 0x0000000000000000
core 0: 0x00000000000000a8 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x00000000000000a8
core 0:           tval 0x00000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3 0x0000000000000001
[error] PC SIM 0000000000000000, DUT 00000000000000a8
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x0000000000000000 gp = 0x0000000000000001
tp = 0x0000000000000000 t0 = 0x0000000000000000 t1 = 0x0000000000000000 t2 = 0xfffffffffffffc18
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0xfffffffffffc18 a1 = 0x000000000000008c
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000000000000 a5 = 0x0000000000000000
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```

## J Type

```

core 0: 0x0000000000000000 (0x00301663) bne      zero, gp, pc + 12
core 0: 3 0x0000000000000000 (0x00301663)
core 0: >>> pass
core 0: 0x0000000000000000 (0x00000093) li      ra, 0
core 0: 3 0x0000000000000000 (0x00000093) x1  0x0000000000000000
core 0: 0x0000000000000000 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000000
core 0:           tval 0x0000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3  0x0000000000000001
[error] PC SIM 0000000000000000, DUT 0000000000000000
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x000000000000000c gp = 0x0000000000000001
tp = 0x000000000000000c t0 = 0x0000000000000000 t1 = 0x0000000000000000 t2 = 0x0000000000000003
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000000000000 a5 = 0x0000000000000000
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```

## remain

```

core 0: 3 0x0000000000000000 (0x00301663)
core 0: >>> pass
core 0: 0x0000000000000000fa4 (0x00000093) li      ra, 0
core 0: 3 0x0000000000000000fa4 (0x00000093) x1  0x0000000000000000
core 0: 0x0000000000000000fa8 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000000fa8
core 0:           tval 0x0000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3  0x0000000000000001
[error] PC SIM 0000000000000000, DUT 0000000000000000
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x0000000000000004 gp = 0x0000000000000001
tp = 0x000000000000000c t0 = 0x0000000000000000 t1 = 0x0000000000000000cf4 t2 = 0x00000000000000001234567
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0x0000000000000000 a1 = 0x0000000000000000
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000000000000 a5 = 0x0000000000000000cd4
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed

```

## full

```
core 0: 3 0x0000000000000009a0 (0x00301663)
core 0: >>> pass
core 0: 0x0000000000000009ac (0x00000093) li      ra, 0
core 0: 3 0x0000000000000009ac (0x00000093) x1  0x0000000000000000
core 0: 0x0000000000000009b0 (0xc0001073) unimp
core 0: exception trap_illegal_instruction, epc 0x0000000000000009b0
core 0:          tval 0x00000000c0001073
core 0: >>>
core 0: 0x0000000000000000 (0x00100193) li      gp, 1
core 0: 3 0x0000000000000000 (0x00100193) x3  0x0000000000000001
[error] PC SIM 0000000000000000, DUT 0000000000000000
[error] INSN SIM 00100193, DUT c0001073
ft0 = 0x0000000000000000 ft1 = 0x0000000000000000 ft2 = 0x0000000000000000 ft3 = 0x0000000000000000
ft4 = 0x0000000000000000 ft5 = 0x0000000000000000 ft6 = 0x0000000000000000 ft7 = 0x0000000000000000
fs0 = 0x0000000000000000 fs1 = 0x0000000000000000 fa0 = 0x0000000000000000 fa1 = 0x0000000000000000
fa2 = 0x0000000000000000 fa3 = 0x0000000000000000 fa4 = 0x0000000000000000 fa5 = 0x0000000000000000
fa6 = 0x0000000000000000 fa7 = 0x0000000000000000 fs2 = 0x0000000000000000 fs3 = 0x0000000000000000
fs4 = 0x0000000000000000 fs5 = 0x0000000000000000 fs6 = 0x0000000000000000 fs7 = 0x0000000000000000
fs8 = 0x0000000000000000 fs9 = 0x0000000000000000 fs10 = 0x0000000000000000 fs11 = 0x0000000000000000
ft8 = 0x0000000000000000 ft9 = 0x0000000000000000 ft10 = 0x0000000000000000 ft11 = 0x0000000000000000
x0 = 0x0000000000000000 ra = 0x0000000000000000 sp = 0x0000000000000004 gp = 0x0000000000000001
tp = 0x0000000000000002 t0 = 0x0000000000000002 t1 = 0xfffffffff00ff0ff t2 = 0x0000000023456780
s0 = 0x0000000000000000 s1 = 0x0000000000000000 a0 = 0xfffffffffffffd8f0 a1 = 0x000000000000072c
a2 = 0x0000000000000000 a3 = 0x0000000000000000 a4 = 0x0000000023456780 a5 = 0x000000000000053c
a6 = 0x0000000000000000 a7 = 0x0000000000000000 s2 = 0x0000000000000000 s3 = 0x0000000000000000
s4 = 0x0000000000000000 s5 = 0x0000000000000000 s6 = 0x0000000000000000 s7 = 0x0000000000000000
s8 = 0x0000000000000000 s9 = 0x0000000000000000 s10 = 0x0000000000000000 s11 = 0x0000000000000000
t3 = 0x0000000000000000 t4 = 0x0000000000000000 t5 = 0x0000000000000000 t6 = 0x0000000000000000
[CJ]          0 Commit Failed
```