7

在 intel 指令中,idiv(integer divsion) 表示有符号除法。
我得到了 的结果idiv,但我不太明白结果。

- 例子

0xffff0000 idiv 0xffff1100


- 我的错误预测
只要我知道,quotient应该是0,remainder应该是0xffff0000,因为......

0xffff0000 / 0xffff1100 = 0  
0xffff0000 % 0xffff1100 = 0xffff0000  


- 然而,结果是……
之前idiv

eax            0xffff0000            # dividend
esi            0xffff1100            # divisor
edx            0x0                     

idiv

eax            0xfffeedcc            # quotient
edx            0x7400   29696        # remainder


- 问题
结果是我无法预料的价值。
有人可以解释一下已签名的除法(idiv)吗?


- 附加。 这里有更多关于idiv.
idiv使用 eax 寄存器作为源寄存器。
作为执行的结果,商存储在 eax 中,余数存储在 edx 中。

4

1 回答 1

8

idiv将 edx:eax 除以显式源操作数。请参阅英特尔的说明手册条目

由于 edx 为 0,因此 edx:eax 为正数。您将 4294901760 除以 -61184,得到 -70196,余数为 29696。

请记住,被除数(EDX:EAX)和除数(在您的情况下为 ESI)都被解释为 2 的补码符号数,因此任何设置高位的位模式都是负数。

00000000ffff0000 = 4294901760
ffff1100 = -61184
fffeedcc = -70196
7400 = 29696

您应该在使用 idiv 之前使用将扩展 eax 签名到 edx cdq,而不是通过将 EDX 归零来进行零扩展。

但是,这仍然不会给出您期望的结果,因为 -65536 除以 -61184 等于 1,余数为 -4352。


(负除数和正除数会产生负余数:X86 IDIV 余数符号取决于 8/-3 和 -8/3 的除数符号?

于 2019-01-02T04:10:21.177 回答