

3/21 第二章

3. (1) addi x0, x0, 0

(2) jalr x0, 0(x1)

(3) jal ra, offset

(4) add rd, rs, x0

(5) csrr rd, cycle

(6) addiw rd, rs, 0

7. (1) slt t3, t0, t1

slt t4, t0, t2

(2) add t0, t1, t2

bltu t0, t1, overflow

(3) x86 指令集通过标志寄存器中的进位标志 CF(Carry Flag) 和溢出标志 OF(Overflow Flag) 来检测加法溢出。加法指令执行时, CF 标志将设置为 1 如果有进位, OF 标志将设置为 1 如果加法结果无法用相应的数据类型表示。

ARM 指令集也用类似标志来检测加法溢出, 对于 32 位数, CPSR 进位标志 C 和溢出标志 V 分别用于检测加法的进位和溢出。对 64 位数, 这些标志存储在 64 位程序状态寄存器的对应位中。

8. (1)  $O_p = DIVU: rd = 0xffffffffffff$

~~O<sub>p</sub> = REMU: rd = X~~

~~O<sub>p</sub> = DIV: rd = 0xffffffffffff~~

~~O<sub>p</sub> = REM: rd = X~~

(2) NX - 非精确异常: NX=1, 产生非精确异常; VF - 下溢异常: VF=1, 产生下溢异常;

OF - 上溢异常: OF=1, 产生上溢异常; DZ - 除 0 异常: DZ=1, 产生除 0 异常;

NV - 无效操作数异常: NV=1, 产生无效操作数异常.

fflags 被置位不会使处理器陷入异常。

(3) 在 x86 全集架构中，处理器会设置相应的异常标志位，即 #DE，并且通过调用中断向量 0 来触发异常处理程序。

在 ARM 架构中，除以 0 异常会通过协处理器 CP15 来设置相应的异常标志位，并通过中断向量 14 来挂到异常处理程序中执行相应操作。

## 12. (1) S (2) M (3) M 与 S (4) S (5) U

Bootloader 在 M 模式下执行一些底层操作，加载操作系统内核时，要切换到 S 模式。

### 13. Loop:

vecMul:

li a2, 0

j loop

Loop:

b1t a2, t0, +99, a2, exit

add a2, a2, t1

sll a2, t0, a2, 2

sll t3, a2, 2

add t3, t3, t1

lw t2, 0(t0)

lw t3, 0(t3)

mul t0, t2, t1

addi a2, a2, 1

j loop

exit:  
mv a0, 0(t0).  
ret

14. b1t a, if  
sw a2, a0, a1

if:

add a2, a0, a1

15. sw t0, a(t0)

li 3

addi t0, t0, 4

sw t1, 0(t0)

addi t0, t0, 8

sw t1, 0(t0)

16. Swap:

mv t2, t0

add t0, t1, x0

add t1, t2, x0

17. 令 a0=0, a1=1, a2=30.

如果 a0 与 a2 相等，则转至 done，执行 exit 命令。

如果 a0 与 a2 不相等，a1 左移一位，即乘以 2，并再加 1，一直如此循环，直到 a0 增加至 30 与 a2 相等跳出循环。 $a0 = 0 \rightarrow 1 \rightarrow 3 \rightarrow 7 \rightarrow 15 \rightarrow 31 \rightarrow 63 \rightarrow 127 \rightarrow 255 \rightarrow 511 \rightarrow 1023 \rightarrow 2047 \rightarrow 4095 \rightarrow 8191 \rightarrow 16383 \rightarrow 32767 \rightarrow 65535$ ，即循环 30 次。