我一直在阅读div
和mul
汇编操作,我决定用C写一个简单的程序来看看它们的实际操作:
#include <stdlib.h>
#include <stdio.h>
int main()
{
size_t i = 9;
size_t j = i / 5;
printf("%zu\n",j);
return 0;
}
然后生成汇编语言代码:
gcc -S division.c -O0 -masm=intel
但是查看生成的division.s
文件,它不包含任何div操作!相反,它使用位移位和神奇数字来施展某种黑魔法。下面是计算i/5
的代码片段:
mov rax, QWORD PTR [rbp-16] ; Move i (=9) to RAX
movabs rdx, -3689348814741910323 ; Move some magic number to RDX (?)
mul rdx ; Multiply 9 by magic number
mov rax, rdx ; Take only the upper 64 bits of the result
shr rax, 2 ; Shift these bits 2 places to the right (?)
mov QWORD PTR [rbp-8], rax ; Magically, RAX contains 9/5=1 now,
; so we can assign it to j
这是怎么回事?为什么GCC根本不使用div ?它是如何产生这个神奇的数字的,为什么一切都有效?