常用的汇编指令(对寄存器直接操作)
mov add sub pop push
常用的汇编指令(辅助)(对寄存器间接操作)
call jump ret
根据硬件位数的不同,指令和寄存器的名称要做调整:
16位(defualt) 32位指令后添加l,如movl addl subl popl pushl等。
16位(default),32位寄存器,前面添加e,如eax,ebp,esp等。
寻址:
mov $4 %eax
$4为立即数寻址,%eax为寄存器寻址。eax=4
mov $4 (%eax)
(%eax)为寄存器间接寻址,*eax=4
mov $4 8(%eax)
注意:栈的地址为从高到低,而存储数据时,数据都是从低到高。所以每次手动存储数据时,都要进行类似如下的操作:
subl $4,%esp
movl $data,%esp
注意:在被调用函数取参数时 movl 8(%ebp), %eax,是因为调用函数中用了call,被调用函数中使用了
pushl。例如 main函数 movl $10 (%esp)后调用f,f函数一看是就调用了 pushl %ebp,f函数要取mian传入的参数,就必须上回溯8.
mov add sub pop push
常用的汇编指令(辅助)(对寄存器间接操作)
call jump ret
根据硬件位数的不同,指令和寄存器的名称要做调整:
16位(defualt) 32位指令后添加l,如movl addl subl popl pushl等。
16位(default),32位寄存器,前面添加e,如eax,ebp,esp等。
寻址:
mov $4 %eax
$4为立即数寻址,%eax为寄存器寻址。eax=4
mov $4 (%eax)
(%eax)为寄存器间接寻址,*eax=4
mov $4 8(%eax)
表示相对寻址,*(eax+8)=4
int g(int x)
{
return x+5;
}
int f(int x)
{
return g(x);
}
int main(void)
{
return f(10)+1;
}
g:
pushl %ebp
movl %esp, %ebp
movl 8(%ebp), %eax
addl $5, %eax
popl %ebp
ret
f:
pushl %ebp
movl %esp, %ebp
subl $4, %esp
movl 8(%ebp), %eax
movl %eax, (%esp)
call g
leave
ret
main:
pushl %ebp
movl %esp, %ebp
subl $4, %esp
movl $10, (%esp)
call f
addl $1, %eax
leave
ret
注意:隐性更改esp寄存器的指令为:pushl,popl,call他们都会让esp的地址-4
注意:栈的地址为从高到低,而存储数据时,数据都是从低到高。所以每次手动存储数据时,都要进行类似如下的操作:
subl $4,%esp
movl $data,%esp
注意:在被调用函数取参数时 movl 8(%ebp), %eax,是因为调用函数中用了call,被调用函数中使用了
pushl。例如 main函数 movl $10 (%esp)后调用f,f函数一看是就调用了 pushl %ebp,f函数要取mian传入的参数,就必须上回溯8.
一张有用的图片:
参考资料:https://segmentfault.com/a/1190000002575242