

### 3. 伪指令 → 基本指令

- (1) nop → addi x0, x0, 0 , 空指令
- (2) ret → jalr x0, x1, 0 , 返回指令
- (3) call offset → auipc x6, offset[31:12]  
jalr x1, x6, offset[11:0]
- (4) mv rd, rs → addi rd, rs, 0
- (5) rdcycle rd → csrrs rd, cycle, x0 固期微流水线报告
- (6) sext.w rd, rs → addiw rd, rs, 0 符号位扩展指令

### 1. (1) 有符号数加法溢出检测:

add t0, t1, t2  
slti t3, t2, 0  
slt t4, t0, t1  
bne t3, t4, overflow

### (2) 无符号数:

add t0, t1, t2  
blt t0, t1, overflow

(3) ARM汇编中可以通过检查程序状态寄存器 PSR 中的标志位来检测加法溢出.

例如, MOV R0, R2

ADDI R1, R0, #constant

EVS overflow 如果发生溢出, 则跳转到 overflow 标签处.

X86汇编, 可以通过检查程序状态寄存器 PSR 中的标志位来检测加法溢出.

如果发生了溢出, 则程序状态寄存器 "OF" 标志位将被设为 1, 否则将被清零.

8. (1) Op=DIVV 时 rd = 0xffffffffffff

Op=REMO 时 rd = X

Op=DIV 时 rd = 0x7fffffff

Op=REM 时 rd = X

'div' 和 'divu' 的结果是最大有符号整数和最大无符号整数，此时表示整数溢出的标准化行为。在 'rem' 和 'remu' 中，余数为被除数本身，这是为了使余数仍有意义，使除法操作对不同的被除数都有确定的行为，这可以确保不会出现未定义的行为，使代码更可靠。

### (2) NX - 非精确异常

NX=0 时 没有产生非精确异常

NX=1 时 产生非精确异常

### UF - 下溢异常

UF=0 时 没有下溢异常

UF=1 时 有下溢异常

### OF - 上溢异常

OF=0 时 没有产生上溢异常

OF=1 时 有上溢异常

### D2 - 除零异常

D2=0 时 没有除零异常

D2=1 时 有除零异常

### NV - 无效操作数异常

NV=0 时 没有产生无效操作数异常

NV=1 时 产生无效操作数异常

fflags 被置位不会使得处理器陷入系统调用。

(3) x86 架构中除以 0 会产生一个异常。x86 处理器使用称为“除法错误”的异常来处理该情况。

处理器程序可以检查错误码，并根据需要采取适当措施

12. (1) Linux kernel 应当处于 S-mode

(2) Boot ROM 应当处于 M-mode

(3) Boot Loader 应当处于 M-mode

(4) USB Driver 应当处于 S-mode

(5) Vim 应当处于 U-mode

13. start : addi sp, sp, -32  
 sd ra, 24(sp)  
 sd so, [6(sp)]  
 addi so, sp, 32  
 mv a3, zero # t3 = i  
 li a4, 100 # t4 = 100  
 lw  
 loop : sll a5, a3, 2 # a5 = i \* 4  
 add a6, a5, t0  
 lw a6, 0(a6) # a6 = AC[i]  
 add a7, a5, t1  
 lw a7, 0(a7) # a7 = BC[i]  
 mul a6, a7, t2 # AC[i] = BC[i] \* C  
 addi a8, a3, 1 # iff  
 add a7, a5, t0  
 sw a6, 0(a7)  
 bge a3, a4, end # i < 100  
 end : ld t0, 0(t0)  
 ld ra, 24(sp)  
 ld so, [6(sp)]  
 addi sp, sp, 32  
 ret

14: ~~如果 a,b 和 C 分别又 d0,d1,d2~~

则

|                                                              |               |                                                                       |
|--------------------------------------------------------------|---------------|-----------------------------------------------------------------------|
| <pre> if (a&gt;b){   c=a+b; } else {   c=a-b; }       </pre> | $\rightarrow$ | <pre> bgt d1, a0, If add d2, a1, a0   sub d2, a2, a0, d1       </pre> |
|--------------------------------------------------------------|---------------|-----------------------------------------------------------------------|

15.   
 sw t0, 0(cp)  
 addi t1, x0, 3  
 addi t2, t0, 4  
 sw t1, 0(t2)  
 muli t2, t1, 4  
 add t2, t2, t0  
 sw t1, 0(t2)

16: swap:

mv t2, t0  
 mv t0, t1  
 mv t1, t2  
 ret

17. 原 RISC-V 编译器实现:

```

int i, a0 = 0, a1 = 1;
for (i = 0; a0 < 30; i++) {      // 将 a1 左移了 30 位，即乘以 230.
  a1 = a1 * 2;
  i++;
}
    
```