

南开大学

计算机学院

体系结构仿真实验报告

# Lab1

冯思程 2112213

年级: 2021 级

专业:计算机科学与技术

指导教师:李雨森

# 摘要

本篇报告主要针对指令级 MIPS 模拟器的实现进行了探索,根据 MIPS 的指令集字段特征实现对 53 条 MIPS 指令进行执行,并根据测试文件对实现的 MIPS 模拟器进行测试,结果准确无误,成功实现了模拟器。

关键字: MIPS、模拟器、指令、测试文件

# 目录

| _           | <b>以验过程</b> | _        |        |        |    |    |   |  |  |      |  |  |  |  |  |  |  |
|-------------|-------------|----------|--------|--------|----|----|---|--|--|------|--|--|--|--|--|--|--|
| <u>(</u> →) | 实验》         | <b> </b> | Ī      |        |    |    |   |  |  | <br> |  |  |  |  |  |  |  |
| (二)         | sim.c       | 文件的:     | 编写 .   |        |    |    |   |  |  | <br> |  |  |  |  |  |  |  |
|             | 1.          | 划分字      | 段函数    |        |    |    |   |  |  | <br> |  |  |  |  |  |  |  |
|             | 2.          | 扩展函      |        |        |    |    |   |  |  |      |  |  |  |  |  |  |  |
|             | 3.          | process  | _instr | ıctior | 函数 | 效实 | 现 |  |  | <br> |  |  |  |  |  |  |  |
| (三)         | 测试          | 文件的编     | 写和生    | .成 .   |    |    |   |  |  | <br> |  |  |  |  |  |  |  |
|             | 1.          | 测试文      | 件编写    |        |    |    |   |  |  | <br> |  |  |  |  |  |  |  |
|             | 2.          | .s 文件    | 到.x 文  | 件的结    | 专换 |    |   |  |  | <br> |  |  |  |  |  |  |  |
| 四)          | 测试          |          |        |        |    |    |   |  |  | <br> |  |  |  |  |  |  |  |

## 一、 实验环境配置

根据实验内容、我首先对实验环境进行了配置、并对必备的一些工具进行安装和测试。

## (一) 实验环境与说明

本次实验我在 wsl 中完成, 其中具体的环境配置如下:

| Windows 版本 | wsl 版本 | vs code 版本 | 系统版本               | OS 类型      | qtspim 版本 |
|------------|--------|------------|--------------------|------------|-----------|
| windows11  | wsl2   | 1.82.0     | Ubuntu 22.04.2 LTS | Linux 64 位 | 9.1.24    |

表 1: 实验环境说明表

实验用到的代码可以见 GitHub 这个链接,工具目前主要用到了 make 命令,这个工具已经 在当前环境中配置好。

综上,感谢老师与助教的审查批阅与指正,辛苦!

# 二、 实验过程

## (一) 实验流程总览

首先在完成 sim.c 文件 (实现 MIPS 53 条指令) 的基础上,我利用在/src/路径下已经写好的 makefile,用命令1编译并链接生成一个目标文件 sim 。然后,我再利用/inputs/路径下的一些用于测试的.s 文件生成的.x 文件对 sim 进行测试,用到的命令是命令1。

make

src/sim inputs/addiu.x(以 addiu.x 文件为例)

补充说明: 这里从.s 测试文件到.x 文件 (16 进制机器码) 的转换是本来是通过 asm2hex 文件实现的, 但是由于这里 asm2hex 文件用到的 spim 没有给出,于是利用了 qtspim 进行.s 文件到.x 文件的转换。

## (二) sim.c 文件的编写

这里的 sim.c 文件中应该对全部的 53 条指令进行实现,这里的输入就是 instruction,即指令。我需要完成的是根据输入的 32 位指令的划分出字段,并根据字段判断指令的类型与具体操作,然后根据操作进行执行。最终我可以完成整个 MIPS 模拟器的实现。

### 1. 划分字段函数

MIPS 指令集包括三种类型的指令:

1. I型指令:与立即数操作相关的指令,这类指令通常有一个寄存器操作数和一个立即数(即固定的数值)。其字段划分如下:



图 1: I 型指令字段组成

2. J型指令:与跳转相关的指令,用于无条件跳转。它们指定一个跳转目标地址。其字段划分如下:



图 2: J型指令字段组成

3. R 型指令: 主要用于寄存器之间的操作的指令,它们通常有三个寄存器操作数:源寄存器 1、源寄存器 2 和目标寄存器。其字段划分如下:



图 3: R 型指令字段组成

根据上文不同类型指令的字段划分,我在 sim.c 文件中编写函数进行不同字段的获取,函数的输入都是 32 位指令,然后根据不同字段对应的不同位置,利用**位与**运算的特点进行字段划分,代码如下:

#### 获取字段函数

```
//获取instruction的指定字段, 其中inst代表32位指令

uint32_t get_op(uint32_t inst) { return inst >> 26; }

uint32_t get_rs(uint32_t inst) { return (inst >> 21) & 0x1f; }

uint32_t get_rt(uint32_t inst) { return (inst >> 16) & 0x1f; }

uint32_t get_rd(uint32_t inst) { return (inst >> 11) & 0x1f; }

uint32_t get_targetadr(uint32_t inst) { return inst & 0x3ffffff; }//获取跳转 目标地址-26位

uint32_t get_imm(uint32_t inst) { return inst & 0xffff; }//获取立即数字段-16位

uint32_t get_shamt(uint32_t inst) { return (inst >> 6) & 0x1f; }

uint32_t get_func(uint32_t inst) { return inst & 0x3f; }
```

#### 2. 扩展函数

在指令执行的时候,需要实现 16 位立即数符号扩展、字节符号扩展、半字符号扩展、32 位立即数零扩展、字节零扩展、半字零扩展,扩展的目标都是 32 位。其中扩展方法是利用 C 语言的指针和类型转换实现的,代码如下:

#### 扩展函数

```
// 将16位立即数符号扩展到32位
  uint32_t sign_ext(uint32_t imm) {
      int32_t signed_imm = *((int16_t*)&imm); // 将16位值视为有符号数
      uint32_t extended_imm = *((uint32_t*)&signed_imm); // 转换为32位数
      return extended imm;
  // 将字节(8位)符号扩展到32位
  uint32_t sign_ext_byte(uint8_t imm) {
      int32_t signed_imm = *((int8_t*)&imm); // 将8位值视为有符号数
      uint32_t extended_imm = *((uint32_t*)&signed_imm); // 转换为32位数
      return extended_imm;
  }
  // 将半字(16位)符号扩展到32位,实现与16位立即数的符号扩展相同
  uint32_t sign_ext_half(uint16_t imm) {
      int32_t signed_imm = *((int16_t*)&imm); // 将16位值视为有符号数
      uint32_t extended_imm = *((uint32_t*)&signed imm); // 转换为32位数
      return extended_imm;
19
  }
20
  //将32位立即数零扩展(实际上,它保持不变,只需要转一下类型就可以了),
  uint32_t zero_ext(uint32_t imm) {
      return imm; // 32位值不变
  }
  // 将字节(8位)零扩展到32位,实现类似zero_ext
  uint32_t zero_ext_byte(uint8_t imm) {
      return imm; // 在C中, 8位无符号值会自动零扩展到32位
  }
30
31
  // 将半字(16位)零扩展到32位,实现类似zero_ext
  uint32_t zero_ext_half(uint16_t imm) {
     return imm; // 在C中, 16位无符号值会自动零扩展到32位
  }
```

#### 3. process instruction 函数实现

这里利用 switch-case 结构根据不同字段值区分不同指令,并针对 53 条指令一一进行实现,展示代码如下 (适当注释,标明了每个 case 实现的指令)。

#### process instruction 函数

```
switch (op) {
case 0x0: {//R型指令
switch (func) {
case 0x0: {
// SLL逻辑左移
```

```
NEXT STATE.REGS[rd] = CURRENT STATE.REGS[rt] << shamt;</pre>
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
    break;
}
case 0x2: {
    // SRL逻辑右移
   NEXT_STATE.REGS[rd] = CURRENT_STATE.REGS[rt] >> shamt;
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
    break;
}
case 0x3: {
    // SRA算术右移
    int32\_t val = *((int32\_t*)\&CURRENT\_STATE.REGS[rt]);
    val = val >> shamt;
   NEXT\_STATE.REGS[rd] = val;
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
   break;
}
case 0x4: {
    // SLLV变量逻辑左移
    uint32_t shamt = CURRENT_STATE.REGS[rs] & 0x1f;
    NEXT_STATE.REGS[rd] = CURRENT_STATE.REGS[rt] << shamt;
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
    break;
}
case 0x6: {
    // SRLV变量逻辑右移
    uint32_t shamt = CURRENT_STATE.REGS[rs] & 0x1f;
   NEXT\_STATE.REGS[rd] = CURRENT\_STATE.REGS[rt] >> shamt;
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
    break;
}
case 0x7: {
    // SRAV变量算术右移
    int32\_t val = *((int32\_t*)\&CURRENT\_STATE.REGS[rt]);
    uint32_t shamt = CURRENT_STATE.REGS[rs] & 0x1f;
    val = val >> shamt;
   NEXT STATE.REGS[rd] = val;
   NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
   break;
}
case 0x8: {
    // JR无条件跳转
   NEXT\_STATE.PC = CURRENT\_STATE.REGS[rs];
    break;
}
case 0x9: {
    // JALR跳转并链接
```

```
NEXT STATE.REGS[rd] = CURRENT STATE.PC + 4;
                       NEXT\_STATE.PC = CURRENT\_STATE.REGS[rs];
                       break;
                   }
                   case 0xc: {
                       // SYSCALL系统调用,这里是按照实验手册的限制性定义进行编
                        if (CURRENT_STATE.REGS[2] = 0x0a) {//v0寄存器对应reg[2]
                           RUN\_BIT = FALSE;
                       } else {
                           NEXT STATE.PC = CURRENT STATE.PC + 4;
                       break;
                   }
                   case 0x10: {
                       // MFHI从HI寄存器移动值
                       NEXT\_STATE.REGS[rd] = CURRENT\_STATE.HI;
                       NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                       break;
                   }
                   case 0x11: {
                       // MTHI将值移动到 HI 寄存器
                       NEXT_STATE.HI = CURRENT_STATE.REGS[rs];
                       NEXT STATE.PC = CURRENT STATE.PC + 4;
                       break;
                   }
                   case 0x12: {
                       // MFLO从 LO 寄存器移动值
                       NEXT_STATE.REGS[rd] = CURRENT_STATE.LO;
                       NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                       break;
                   }
                   case 0x13: {
                       // MILO 将值移动到 LO 寄存器
                       NEXT_STATE.LO = CURRENT_STATE.REGS[rs];
                       NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                       break;
                   }
                   case 0x18: {
                       // MULT乘法
                       int64_t lhs = *((int32_t*)\&CURRENT_STATE.REGS[rs]);
                       int64\_t rhs = *((int32\_t*)\&CURRENT\_STATE.REGS[rt]);
                       int64_t product = lhs * rhs;
                       uint64_t uint_product = *((uint32_t*)&product);
                       NEXT\_STATE.HI =
                           (uint32_t)((uint\_product >> 32) \& 0xffffffff);
                       NEXT_STATE.LO = (uint32_t)(uint_product & 0xfffffffff);
                       NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
100
```

二、 实验过程 体系结构仿真实验报告

```
break:
                    }
                    case 0x19: {
                        // MULTH无符号乘法
                         uint64_t lhs = CURRENT_STATE.REGS[rs];
                        uint64_t rhs = CURRENT_STATE.REGS[rt];
                        uint64_t product = lhs * rhs;
                        NEXT_STATE.HI = (uint32_t)((product >> 32) \& 0xfffffffff);
                        NEXT_STATE.LO = (uint32_t)(product & 0xffffffff);
                        NEXT STATE.PC = CURRENT STATE.PC + 4;
                        break;
                    }
                    case 0x1a: {
114
                        // DIV 除法
                        int32\_t lhs = *((int32_t*)&CURRENT_STATE.REGS[rs]);
                        int32\_t rhs = *((int32\_t*)\&CURRENT\_STATE.REGS[rt]);
                        NEXT\_STATE.LO = lhs / rhs;
                        NEXT\_STATE.HI = lhs \% rhs;
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                    }
                    case 0x1b: {
                        // DIVU无符号除法
124
                         uint32_t lhs = CURRENT_STATE.REGS[rs];
                        uint32_t rhs = CURRENT_STATE.REGS[rt];
                        NEXT\_STATE.LO = lhs / rhs;
                        NEXT_STATE.HI = lhs % rhs;
128
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                    }
                    case 0x20: {
                        // ADD加法
                        NEXT\_STATE.REGS[rd] =
                            CURRENT_STATE.REGS[rs] + CURRENT_STATE.REGS[rt];
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
137
                    }
                    case 0x21: {
                        // ADDU无符号加法
140
                        NEXT\_STATE.REGS[rd] =
                            CURRENT_STATE.REGS[rs] + CURRENT_STATE.REGS[rt];
142
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                    case 0x22: {
                         // SUB减法
                        NEXT\_STATE.REGS[rd] =
148
```

```
CURRENT STATE.REGS[rs] - CURRENT STATE.REGS[rt];
149
                        NEXT STATE.PC = CURRENT STATE.PC + 4;
                        break;
                     }
                     case 0x23: {
                         // SUBU无符号减法
                        NEXT\_STATE.REGS[rd] =
                             CURRENT\_STATE.REGS[rs] - CURRENT\_STATE.REGS[rt];
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
158
                     }
                     case 0x24: {
                        // AND位与
                        NEXT STATE.REGS[rd] =
                            CURRENT_STATE.REGS[rs] & CURRENT_STATE.REGS[rt];
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
164
                        break;
                     }
                     case 0x25: {
                         // OR 位或
                        NEXT_STATE.REGS[rd]
                            CURRENT_STATE.REGS[rs] | CURRENT_STATE.REGS[rt];
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                     }
                     case 0x26: {
174
                        // XOR位异或
                        NEXT STATE.REGS[rd] =
                             CURRENT_STATE.REGS[rs] ^ CURRENT_STATE.REGS[rt];
177
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                     }
                     case 0x27: {
                         // NOR位非或
                        NEXT\_STATE.REGS[rd] =
                             ~(CURRENT_STATE.REGS[rs] | CURRENT_STATE.REGS[rt]);
184
                        NEXT STATE.PC = CURRENT STATE.PC + 4;
185
                        break;
186
                    }
187
                     case 0x2a: {
                         // SLT设置小于
                         int32\_t lhs = *((int32_t*)&CURRENT_STATE.REGS[rs]);
190
                        int32\_t rhs = *((int32\_t*)\&CURRENT\_STATE.REGS[rt]);
                        NEXT\_STATE.REGS[rd] = (lhs < rhs) ? 1 : 0;
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                     }
                     case 0x2b: {
196
```

```
// SLTU无符号设置小于
                        NEXT\_STATE.REGS[rd] =
                            CURRENT\_STATE.REGS[rs] < CURRENT\_STATE.REGS[rt] ? 1 :
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                        break;
                    }
                    default: {
                         printf("Unknown instruction: □0x%x\n", inst);
204
                        break;
205
206
207
                break;
208
            }
209
            case 0x8: {
                // ADDI立即数加法
211
                NEXT_STATE.REGS[rt] = CURRENT_STATE.REGS[rs] + sign_ext(imm);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
213
                break;
            }
            case 0x9: {
                // ADDIU无符号立即数加法
217
                NEXT_STATE.REGS[rt] = CURRENT_STATE.REGS[rs] + sign_ext(imm);
218
                NEXT STATE.PC = CURRENT STATE.PC + 4;
219
                break;
220
            }
221
            case 0xc: {
222
                // ANDI立即数位与
                NEXT_STATE.REGS[rt] = CURRENT_STATE.REGS[rs] & zero_ext(imm);
224
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                break;
            }
            case 0xd: {
                // ORI立即数位或
                NEXT_STATE.REGS[rt] = CURRENT_STATE.REGS[rs] | zero_ext(imm);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
231
                break;
233
            case 0xe: {
                // XORI立即数位异或
235
                NEXT_STATE.REGS[rt] = CURRENT_STATE.REGS[rs] ^ zero_ext(imm);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                break;
            case 0x4: {
                // BEQ相等跳转
                uint32\_t offset = sign\_ext(imm) << 2;
243
```

```
244
                if (CURRENT_STATE.REGS[rs] == CURRENT_STATE.REGS[rt]) {
                    NEXT\_STATE.PC = CURRENT\_STATE.PC + offset + 4;
246
                } else {
                    NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                break;
            case 0x1: {
                uint32_t offset = sign_ext(imm) << 2;</pre>
253
254
                switch (rt) {
255
                    case 0x0: {
                         // BLTZ小于0分支
257
                         if ((CURRENT_STATE.REGS[rs] & 0x80000000) != 0) {
258
                             NEXT\_STATE.PC = CURRENT\_STATE.PC + offset + 4;
                         } else {
                             NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                         }
                         break;
                     }
                     case 0x10: {
265
                         // BLTZAL小于0跳转并链接
266
                         NEXT STATE.REGS[31] = CURRENT STATE.PC + 4;
267
                         if ((CURRENT_STATE.REGS[rs] & 0x80000000) != 0) {
                             NEXT STATE.PC = CURRENT STATE.PC + offset + 4;
269
                         } else {
270
                             NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
272
                         }
                         break;
                     case 0x1: {
                         // BGEZ大于等于0分支
                         if ((CURRENT\_STATE.REGS[rs] \& 0x80000000) == 0) {
                             NEXT\_STATE.PC = CURRENT\_STATE.PC + offset + 4;
                         } else {
279
                             NEXT STATE.PC = CURRENT STATE.PC + 4;
280
281
                         break;
282
                     }
283
                     case 0x11: {
284
                         // BGEZAL大于等于0跳转并链接
285
                         NEXT\_STATE.REGS[31] = CURRENT\_STATE.PC + 4;
                         if ((CURRENT_STATE.REGS[rs] & 0x80000000) == 0) {
                             NEXT\_STATE.PC = CURRENT\_STATE.PC + offset + 4;
                         } else {
                             NEXT STATE.PC = CURRENT STATE.PC + 4;
                         }
291
```

```
break;
                       }
                  }
294
                  break;
295
             case 0x5: {
                  // BNE不相等分支
                  uint32\_t offset = sign\_ext(imm) << 2;
300
301
                  printf("BNE: offset: \( \)\%d, \( \)rs: \( \)\%d, \( \)rt: \( \)\%d\n", offset, rs, rt);
302
303
                  printf("rs:\_0x%08x\n", CURRENT_STATE.REGS[rs]);
304
                  printf("rt:\u0x%08x\n", CURRENT_STATE.REGS[rt]);
305
                  if (CURRENT_STATE.REGS[rs] != CURRENT_STATE.REGS[rt]) {
307
                       \label{eq:next_state.pc} \mbox{NEXT\_STATE.PC} = \mbox{CURRENT\_STATE.PC} + \mbox{offset} + 4;
                  } else {
309
                       NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                  }
                  break;
313
             case 0x6: {
314
                  // BLEZ小于等于0分支
                  uint32_t offset = sign_ext(imm) << 2;
317
318
                  if (rt == 0) {
                       if ((CURRENT_STATE.REGS[rs] & 0x80000000) != 0 ||
                           CURRENT\_STATE.REGS[rs] == 0)  {
                           NEXT\_STATE.PC = CURRENT\_STATE.PC + offset + 4;
                       } else {
                           NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                  } else {
                       // Illegal instruction
                       printf("Illegal_rt_in_BLEZ.\n");
328
                  break;
             case 0x7: {
                  // BGTZ大于0分支
                  uint32_t offset = sign_ext(imm) << 2;</pre>
                  printf("BGTZ: \_offset: \_0x\%08x, \_rs: \_\%d, \_rt: \_\%d, \_pc: \_0x\%08x \n",
                       offset,
                          rs, rt, CURRENT_STATE.PC);
338
```

```
if (rt == 0) {
                    if ((CURRENT_STATE.REGS[rs] & 0x80000000) == 0 &&
340
                        CURRENT_STATE.REGS[rs] != 0) {
341
                        NEXT_STATE.PC = CURRENT_STATE.PC + offset + (uint32_t)4;
                        printf("PC:\u0x%08x\n", NEXT_STATE.PC);
                    } else {
                        NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                } else {
                    // Illegal instruction
348
                    printf("Illegal, rt, in, BGTZ. \n");
                break;
            }
            case 0x2: {
                // J无条件跳转
                NEXT_STATE.PC = (CURRENT_STATE.PC & 0xf0000000) | (targetadr <<
                break;
            }
            case 0x3: {
                // JAL跳转并链接
                NEXT\_STATE.REGS[31] = CURRENT\_STATE.PC + 4;
                NEXT_STATE.PC = (CURRENT_STATE.PC & 0xf0000000) | (targetadr <<
361
                    2);
                break;
362
363
            case 0xf: {
                // LUI将立即数加载到寄存器高16位
365
                if (rs == 0) {
                    NEXT_STATE.REGS[rt] = imm << 16;
                    NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                } else {
                    // Illegal instruction
371
                break;
374
            case 0x20: {
                // LB从内存加载字节
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
378
                uint8_t byte = mem_read_32(addr) & 0xff;
                NEXT_STATE.REGS[rt] = sign_ext_byte(byte);
                NEXT STATE.PC = CURRENT STATE.PC + 4;
                break;
384
```

```
385
            case 0x24: {
386
                // LBU从内存加载无符号字节
387
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
                uint8_t byte = mem_read_32(addr) & 0xff;
                NEXT_STATE.REGS[rt] = zero_ext_byte(byte);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
394
                break;
            case 0x21: {
                // LH从内存加载半字
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
400
                uint16_t half = mem_read_32(addr) & 0xffff;
402
                NEXT_STATE.REGS[rt] = sign_ext_half(half);
                NEXT STATE.PC = CURRENT STATE.PC + 4;
                break;
406
407
            case 0x25: {
408
                // LHU从内存加载无符
409
410
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
411
412
                uint16_t half = mem_read_32(addr) & 0xffff;
413
                NEXT_STATE.REGS[rt] = zero_ext_half(half);
415
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                break;
            case 0x23: {
419
                // LW从内存加载字
421
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
422
423
                NEXT_STATE.REGS[rt] = mem_read_32(addr);
424
                NEXT STATE.PC = CURRENT STATE.PC + 4;
425
                break;
426
            }
            case 0x28: {
                // SB将字节存储到内存
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
432
```

二、 实验过程 体系结构仿真实验报告

```
uint32 t val = (mem read 32(addr) & 0xfffffff00)
433
                                 (CURRENT STATE.REGS[rt] & 0xff);
434
435
                mem_write_32(addr, val);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
                break;
            }
            case 0x29: {
                // SH将半字存储到内存
                uint32 t addr = sign ext(imm) + CURRENT STATE.REGS[rs];
443
444
                uint32\_t val = (mem\_read\_32(addr) & 0xffff0000)
445
                                 (CURRENT STATE.REGS[rt] & 0xffff);
446
                mem_write_32(addr, val);
                NEXT\_STATE.PC = CURRENT\_STATE.PC + 4;
448
                break;
            }
450
            case 0x2b: {
                // SW 将字存储到内存
                uint32_t addr = sign_ext(imm) + CURRENT_STATE.REGS[rs];
454
                mem_write_32(addr, CURRENT_STATE.REGS[rt]);
455
                NEXT STATE.PC = CURRENT STATE.PC + 4;
456
                break;
457
            }
458
            default: {
459
                printf("wrong_inst! at: u0x%08x\n", inst);
                break;
461
462
        }
```

### (三) 测试文件的编写和生成

#### 1. 测试文件编写

在/inputs/路径下已经存在多个用于测试的.s 文件, 但是根据要求我还要多编写几个测试文件以确保 sim.c 文件的正确性。下面展示两个额外编写的测试文件 (.s 文件格式)

addtest1.s

```
.text
test1:
                 addiu $a0, $zero, 20
                                                   \# \$a0 = 20
                 addiu $a1, $a0, 30
                                                   \# \$a1 = \$a0 + 30 = 50
                 addiu $a2, $a1, 40
                                                   \# \$a2 = \$a1 + 40 = 90
                 addu
                         $a3, $a2, $a0
                                                   \# \$a3 = \$a2 + \$a0 = 110
                         $t0, $a3, $a1
                                                   # $t0 = $a3 | $a1
                 or
                         $t1, $t0, 255
                                                   \# \$t1 = \$t0 \& 255
                 andi
```

#### addtest2.s

```
.text
test2:
                                          \# $t0 = 500
        addiu
                  $t0, $zero, 500
                  $t1, $t0, $t0
                                          \# \$t1 = \$t0 + \$t0 = 1000
        addu
                  $t2, $t1, $t0
                                          \# \$t2 = \$t1 \mid \$t0 = 1500
        or
                 $t3, $t2, 1
                                          \# \$t3 = \$t2 << 1 = 3000
        sll
                  $t4, $t3, $t1
                                          \# \$t4 = \$t3 - \$t1 = 2000
        subu
                                          \# \$t5 = \$t4 ^ \$t2
                  $t5, $t4, $t2
        xor
                                          # $t6 = $t5 ^ 100
                  $t6, $t5, 100
        xori
                                          \# \$t7 = \$t6 >> 2
                  $t7, $t6, 2
        srl
                                          \# $t8 = $t7 >> 1 (arithmetic shift)
                  $t8, $t7, 1
        sra
                  $t9, $t8, $t6
                                          \# \$t9 = \$t8 \& \$t6
        and
                                          \# \$s0 = 50 << 16
                  \$s0, 50
        lui
                  $v0, $zero, 0x5
                                          \# \$v0 = 5
        addiu
         syscall
```

#### 2. .s 文件到.x 文件的转换

这里由于原来的 asm2hex 可执行文件不可用,这里我下载了 9.1.24 版本的 qtspim 进行转换,在链接 安装.deb 文件,然后在 wsl 中进行安装配置。

安装成功后,在命令行输入 qtspim 命令进入其图形化界面,然后通过 load file 的选项加载一个.s 文件,下面以 addiu.s 文件为例进行操作,加载文件后结果如下:



图 4: qtspim 结果

观察发现, .s 文件中的 asm 汇编代码, 正好可以与 qtspim 中的 16 进制机器码一一对应, 如上图中的红色箭头, 然后复制 16 进制机器码并放入一个 addiu.x 文件, 完成转换。其他的.s 文件用同样的操作进行转换。

展示 addiu.x 文件的代码如下:

#### addiu.x

```
1 2402000a

2 24080005

3 2509012c

4 240a01f4

5 254b0022

6 256b002d

7 0000000c
```

## (四) 测试

现在我已经完成了 sim.c 文件的编写和.x 测试文件的生成, 现在只需要进行测试并检验 MIPS 模拟器是否可以正确执行指令。

首先通过 make 命令编译链接生成 sim 可执行文件, 结果如下:

```
fsc@FSC:~/lab-simulation$ cd src
fsc@FSC:~/lab-simulation/src$ ls
Makefile shell.c shell.h sim.c
fsc@FSC:~/lab-simulation/src$ make
gcc -g -02 shell.c sim.c -o sim
fsc@FSC:~/lab-simulation/src$ ls
Makefile shell.c shell.h sim sim.c
fsc@FSC:~/lab-simulation/src$ [
```

图 5: sim 可执行文件生成

然后通过 cd 命令回到 src 上级路径中,实行命令1进行测试,这里以 addiu.x 文件为例进行测试并说明,命令执行结果如下,成功进行内置的 shell 界面:

src/sim inputs/addiu.x(addiu.x 为例)

```
• fsc@FSC:~/lab-simulation/src$ cd ..
• fsc@FSC:~/lab-simulation$ src/sim inputs/addiu.x
MIPS Simulator

Read 7 words from program into memory.

MIPS-SIM>
```

图 6: addiu.x 测试

运行 shell 中已经内置好的命令 go 和 rdump 进行检验,结果如下:

图 7: 检验结果

展示出 addiu.s 文件的代码,如下:

#### addiu.s

```
.text
___start: addiu $v0, $zero, 10
addiu $t0, $zero, 5
addiu $t1, $t0, 300
addiu $t2, $zero, 500
addiu $t3, $t2, 34
addiu $t3, $t3, 45
syscall
```

根据代码进行分析, v0 寄存器值为 10, t0 寄存器值为 5, t1 寄存器值为 305, t2 寄存器值 为 500, t3 寄存器值为 579。

观察寄存器值, R2 寄存器是 0x0000 000a, R8 寄存器值是 0x0000 0005, R9 寄存器值是 0x0000 0131, R10 寄存器值是 0x0000 01f4, R11 寄存器值是 0x0000 0243。

v0 寄存器对应 R2 寄存器, t0 寄存器对应 R8 寄存器, t1 寄存器对应 R9 寄存器, t2 寄存器 对应 R10 寄存器, t3 寄存器对应 R11 寄存器, 经过进制转换后检验发现, 程序正确执行, MIPS 模拟器成功执行这个程序。

**遍历测试** 对其他的用于测试的.x 文件一一进行测试,发现均可以正确执行,证明了 MIPS 模拟器的准确性。

# 三、总结

## (一) 工作与结果

在这次实验中,我主要根据 MIPS 指令的字段特性进行 sim.c 文件的编写,并额外编写了几个测试文件,然后用生成的.x 文件进行 MIPS 模拟器的测试,最后测试结果均正确,证明了我成功的实现了一个 MIPS 模拟器。



# 参考文献

- [1] README.pdf
- [2] lab s1.pdf
- [3] MIPSISA.pdf

