汇编知识备忘(1)

常用的汇编指令(对寄存器直接操作)
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

你可能感兴趣的:(c基础知识)