今天看汇编代码的时候无意看到了一个lea指令的巧妙用法,觉得很有意思,有必要记录一下:
先看下面的C代码:
现在用一条lea指令可以实现上面的功能(但有些许不同之处,上面的代码修改了eax的值,flags也改变了)
先看下面的C代码:
#include<stdio.h>
int b=1;
int main(){
printf("%d\n",b++);
}
实在没什么可说的,接着来看其汇编代码片段:
b:
.long 1
.LC0:
.string "%d\n"
main:
. ............
movl b, %eax
leal 1(%eax), %edx
movl %edx, b
subl $8, %esp
pushl %eax
pushl $.LC0
call printf
............
b++的实现其实就是下面三行汇编语句:
movl b, %eax
leal 1(%eax), %edx
movl %edx, b
但第二行的lea指令有点难以理解.后查阅资料得知,这是lea的一个比较tricky的用法,简单来说就是用来进行数学运算,而非求取地址.比如我们想实现把一个寄存器的值加1,然后送到另一个寄存器中:
addl $1,%eax
movl %eax,%edx
现在用一条lea指令可以实现上面的功能(但有些许不同之处,上面的代码修改了eax的值,flags也改变了)
leal 1(%eax),%edx
之前学习汇编的时候就产生过这样的疑问:在上面一条汇编指令中嵌入了数学运算,这还能在一条指令中完成?答案是可以的,原因在于1(%eax)是内存操作,占用的是cpu中
Address generation unit (AGU),并没有占用
arithmetic logic unit (ALU)的时钟周期.
参考资料:http://www.cnblogs.com/polymorphism/archive/2011/12/12/LEA_VS_MOV.html