# 计算机组成与设计 Lab1

## 10192100571 俞辰杰

## 福黄

## 大傻逼

## 实验目的:

在此实验中,运用C++来模拟RISCV的指令级别的单周期循环,使用一些最简单的指令集(见下表)来模拟一些简单的RISCV的操作。在模拟运行的过程中,涉及到ALU的计算,PC地址的跳转,寄存器的读取,立即数的计算。

通过各个部分独立的实现,组合的运行,加深对于计算机底层设备的运行逻辑和设计方式 能够在用户层面加深对于RISCV指令集在机器层面上的操作和理解。

通过编写机器码,能够理解对于RISCV指令集不同类型指令的组成形式和解释方法。

| Name | Format Type | Opcode (Binary) | Func 3(Binary) | Func 7(Binary) |
|------|-------------|-----------------|----------------|----------------|
| add  | R-Type      | 0110011         | 000            | 0000000        |
| sub  | R-Type      | 0110011         | 000            | 0100000        |
| addi | I-Type      | 0010011         | 000            |                |
| and  | R-Type      | 0110011         | 111            | 0000000        |
| or   | R-Type      | 0110011         | 110            | 0000000        |
| xor  | R-Type      | 0110011         | 100            | 0000000        |
| beq  | SB-Type     | 1100011         | 000            |                |
| jal  | UJ-Type     | 1101111         |                |                |
| ld   | I-Type      | 0000011         | 011            |                |
| sd   | S-Type      | 0100011         | 011            |                |

- 如果需要测试自己的代码需要自行更改imem.txt以及在代码中配置寄存器初始值以及数据内存存储值,代码内修改部分如下:
  - o 寄存器堆RF初始化寄存器初始值

```
1
   RF(){
2
       Registers.resize(32);
 3
       // 有32个寄存器
       Registers[0] = bitset<64>(0);
 4
       // 0号寄存器一直为全零
 5
 6
 7
       /*========================*/
 8
       Registers[10] = bitset<64>("1");
 9
       // x10 = 1
10
       Registers[13] = bitset<64>("110000");
11
       // x13 = &A[0] = 0x00000030
12
       Registers[14] = bitset<64>("10000");
13
       // x14 = &B[0] = 0x00000010
       Registers[28] = bitset<64>("11110");
14
       // x28 = i = 30
15
       Registers[29] = bitset<64>("1");
16
17
       // x29 = j = 1
18
       /*=========================*/
19
```

。 数据内存DMem初始化数据内存初始值

```
1
    DataMem(){
 2
        DMem.resize(MemSize);
 3
        ifstream dmem;
        string line;
 4
 5
        int i = 0;
        dmem.open("dmem.txt");
 6
 7
        if (dmem.is_open()){
 8
            while (getline(dmem, line)){
                DMem[i] = bitset<8>(line.substr(0, 8));
 9
10
            }
11
            /*===========*/
12
13
            DMem[16 + 0 * 8 + 7] = bitset < 8 > ("11111111");
            // DMem[16] = B[0] = 6
14
15
            DMem[16 + 1 * 8 + 6] = bitset < 8 > ("1");
            DMem[16 + 1 * 8 + 7] = bitset < 8 > ("00110111");
16
17
            // B[1] = 137
18
            DMem[48 + 0 * 8 + 7] = bitset < 8 > ("0");
19
20
            // DMem[55] = A[0] = 0
            DMem[48 + 27 * 8 + 6] = bitset < 8 > ("1");
21
            DMem[48 + 27 * 8 + 7] = bitset < 8 > ("00110111");
22
23
            // A[27] = 137
            DMem[48 + 28 * 8 + 7] = bitset < 8 > ("11100");
24
```

```
25
          // A[28] = 28
26
          DMem[48 + 29 * 8 + 7] = bitset < 8 > ("11101");
27
          // A[29] = 29
         /*===========*/
28
29
     }
30
      else
31
         cout << "Unable to open file";</pre>
32
       dmem.close();
33 }
```

• 如果不进行修改的话,可以直接运行RISC-V.cpp,运行代码如下CCM.txt所 示:

样例程序的C语言代码,RISCV汇编代码,以及对于的指令集的机器码存放于此

```
1 /*=========================*/
2 while(B[1] != A[i - j]){
3
   j += 1;
4 }
5 | A[j] = B[0];
7 /*=========================*/
8 while(true){
9
    if(B[1] == A[i - j]){
10
         break;
     }else{
11
12
          j += 1;
13
     }
14 }
15 A[j] = B[0];
```

此外, 寄存器初始化和数据内存在代码中进行设定

```
1  x10 : 1
2  x12 : addr_temp
3  x13 : A[]
4  x14 : B[]
5  x27 : data_temp2
6  x28 : i (i = 30)
7  x29 : j (j = 1)
8  x30 : data_temp
```

#### 对应的riscv指令及32位指令码

```
1
    Loop:
            sub x30, x28, x29
                                                    // compute i-j
2
    00
            0100000 11101 11100 000 11110 0110011
3
            add x12, x30, x30
                                                    // multiply by 8
4
    04
            0000000 11110 11110 000 01100 0110011
5
            add x12, x12, x12
6
    08
            0000000 01100 01100 000 01100 0110011
7
            add x12, x12, x12
            0000000 01100 01100 000 01100 0110011
8
    12
9
            add x12, x12, x13
            0000000 01101 01100 000 01100 0110011
10
    16
            1d \times 30, 0(\times 12)
11
                                                    // x30 load A[i-j]
            00000000000 01100 011 11110 0000011
12
    20
            1d x27, 8(x14)
13
                                                    // x31 load B[1]
            00000001000 01110 011 11011 0000011
    24
14
15
            beq x27, x30, Exit
                                   (40 - 28 = 12 = 000000000110[0])
            0 000000 11110 11011 000 0110 0 1100111
16
    28
17
            add x29, x29, x10
                                                    // j += 1
    32
            0000000 01010 11101 000 11101 0110011
18
19
            jal x0, Loop
                                    20
    36
            1 1111101110 1 11111111 00000 1101111
            1d \times 30, 0(\times 14)
21
   Exit:
                                                    // x30 = B[0]
            00000000000 01110 011 11110 0000011
22
    40
                                                    // j * 8
            add x12, x29, x29
23
24
   44
            0000000 11101 11101 000 01100 0110011
25
            add x12, x12, x12
            0000000 01100 01100 000 01100 0110011
26
    48
27
            add x12, x12, x12
            0000000 01100 01100 000 01100 0110011
28
    52
29
            add x12, x12, x13
            0000000 01101 01100 000 01100 0110011
30
    56
31
            sd x30, 0(x12)
                                                    // A[j] = x30
32
    60
            0000000 11110 01100 011 00000 0100011
33
            1111111 11111 11111 111 11111 1111111
34
    64
```

```
1
    01000001110111100000111100110011
    00000001111011110000011000110011
 2
 3
    0000000110001100000011000110011
4
    0000000110001100000011000110011
    0000000110101100000011000110011
 6
    00000000000011000111111100000011
 7
    0000000100001110011110110000011
8
    00000001111011011000011001100111
9
    0000000101011101000111010110011
    111111011101111111111000001101111
10
11
    00000000000011100111111100000011
    00000001110111101000011000110011
12
13
    0000000110001100000011000110011
    0000000110001100000011000110011
14
15
    0000000110101100000011000110011
16
    00000001111001100011000000100011
```

## 对于给定的测试例子,在RFresult.txt中,可以看到寄存器的最终结果:

```
1
2
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
```

## 在dmemresult.txt中,可以跟踪数据内存的数据:

按照预期 A[j] 和 B[0] 应该有相同的数据,A[i-j] = A[27] 和 B[1] 应该有相同的结果根据上方寄存器的结果,我们反向追踪数据内存的位置,其中:

- A[j] 应该在 0x1001000 + 0x111=79 的内存位置
- B[0] 应该在 0x10000 + 0x111 = 23 的内存位置



A[j] 和 B[0] 有相同的结果

- A[27] 应该在 0x110000 + 0x11011000 + 0x111 = 0x100001111 = 271 的内存位置
- B[1] 应该在 0x11000 + 0x111 = 0x11111 = 31 的内存位置

| 265 | 00000000 | 25 | 00000000 |
|-----|----------|----|----------|
| 266 | 00000000 | 26 | 00000000 |
| 267 | 00000000 | 27 | 00000000 |
| 268 | 00000000 | 28 | 00000000 |
| 269 | 00000000 | 29 | 00000000 |
| 270 | 00000000 | 30 | 00000000 |
| 271 | 00000001 | 31 | 00000001 |
| 272 | 00110111 | 32 | 00110111 |

A[27] 和 B[1] 有相同的结果

在实验过程中,我们跟踪输出了每条指令所**读取的寄存器内容、立即数数字,执行的操作类型,PC** 的目标地址,读写操作的目标地址。

由于执行条数过多,故不全部展示,如果修改代码,可以通过此方法在运行过程中及时观察运行过程是否符合逻辑,下图各种类型操作分别展示一次。

Instruction:01000001110111100000111100110011 Regsier1: 28 Register2: 29 Regsier target: 30 Immediate Number: SUBU result: Regsier1: 28 Register2: 29 Regsier target: 30 PC point to 4 Finish A Instruction

#### 图1 数据计算R-type指令结果

Instruction:00000000000001100011111100000011 Regsier1: 12 Register2: 0 Regsier target: 30 Immediate Number: sw/lw address: Load start at 264 MemoryAccess: Regsier1: 12 Register2: 0 Regsier target: 30 PC point to 24 Finish A Instruction

图2数据内存读写sd/ld指令结果

Instruction:00000001111011011000011001100111 Regsier1: 27 Register2: 30 Immediate Number: Branch/Jump Regsier1: 27 Register2: 30 Success Branch addressExtend: 000000000000000000000000000001100 PC point to 40 Finish A Instruction

#### 图3 分支跳转beq指令结果

Instruction: 111111011101111111111000001101111 Regsier1: 0 Register2: 0 Regsier target: 0 Immediate Number: store (PC + 4): Regsier1: 0 Register2: 0 Regsier target: 0 Success Jump addressExtend: 111111111111111111111111111011100 PC point to 0 Finish A Instruction

图4直接跳转jal指令结果