linux函数调用汇编解析,Linux汇编---函数调用过程

或许习惯于用高级语言编程的大部分同学都会忽略了函数调用的具体过程是怎样的,如果想知道这个过程就不得不从汇编入手,但汇编语言又晦涩难懂。在这里谨以一个简单的例子说说我对函数调用过程的学习心得。

先上C语言写的代码:

1 #include2

3

4 unsigned int test(int a,intb)5 {6 intc,d;7 c =a;8 d =b;9 returnc;10 }11

12 intmain()13 {14 unsigned intr;15

16 r = test(1,2);17

18 return 0;19 }

很简单,就是在main()函数里调用test()函数。通过下面的命令编译:

gcc -g -o test test.c   //加-g选项是为了反编译时可以混合显示源码和汇编代码

再通过以下命令将test反编译:

objdump -d -S test

截取其中反编译后的一个片段,如下:

1 08048394 :2 #include3

4

5 unsigned int test(int a,intb)6 {7 8048394: 55 push %ebp8 8048395: 89 e5 mov %esp,%ebp9 8048397: 83 ec 10 sub $0x10,%esp10 intc,d;11 c =a;12 804839a: 8b 45 08 mov 0x8(%ebp),%eax13 804839d: 89 45 fc mov %eax,-0x4(%ebp)14 d =b;15 80483a0: 8b 45 0c mov 0xc(%ebp),%eax16 80483a3: 89 45 f8 mov %eax,-0x8(%ebp)17 returnc;18 80483a6: 8b 45 fc mov -0x4(%ebp),%eax19 }20 80483a9: c9 leave21 80483aa: c3 ret22

23 080483ab :24

25 intmain()26 {27 80483ab: 55 push %ebp28 80483ac: 89 e5 mov %esp,%ebp29 80483ae: 83 ec 18 sub $0x18,%esp30 unsigned intr;31

32 r = test(1,2);33 80483b1: c7 44 24 04 02 00 00 movl $0x2,0x4(%esp)34 80483b8: 00

35 80483b9: c7 04 24 01 00 00 00 movl $0x1,(%esp)36 80483c0: e8 cf ff ff ff call 8048394 37 80483c5: 89 45 fc mov %eax,-0x4(%ebp)38

39 return 0;40 80483c8: b8 00 00 00 00 mov $0x0,%eax41 }42 80483cd: c9 leave43 80483ce: c3 ret44 80483cf: 90 nop

可以很清楚地看到每一条c语句对应的汇编代码。

从第27行开始看起,将ebp寄存器的值压入堆栈;第28行,把esp寄存器的值赋给ebp寄存器;第29行,esp的值自减0x18。假设执行完第29行后堆栈的情况如图1所示。

a72d54773968840f185eb6ff79e68cf6.png

图1

第33行,将立即数0x2(test()的第2个实参)放到[esp+0x4]地址里;第35行,将立即数0x1(test()的第1个实参)放到[esp]地址里;第36行。调用test()函数,此时会将断点(返回地址)也压入堆栈,如图2所示。

b1fac0a19c92436c4ea6d83911c6af78.png

图2

接着从第7行开始执行,将ebp压栈;第8行,将esp的值赋给ebp;第9行,esp自减0x10,如图3所示。

a177213b0ce2d252e532bf1405748dda.png

图3

第12行,将[ebp+0x8]地址里的内容赋给eax,由图3可以发现,[ebp+0x8]刚好是0x1这个数所在的地址,即把0x1赋给eax寄存器;第13行,把eax的值放到[ebp-4]这个地址里;第15行,将[ebp+0xc]地址里的内容赋给eax,[ebp+0xc]刚好是0x2这个数所在的地址,即把0x2赋给eax;第16行,把eax的值放到[ebp-8]这个地址里。此时,如图4所示。

2e39e1fa4c1e2ea62a653fe5c68d363e.png

图4

第18行,通过eax寄存器保存函数的返回值。

总的来说,就是函数调用时,先将参数从右到左依次压入堆栈,然后再将断点、ebp寄存器的值压栈,从这里也可以知道为什么值传递不能改变实参原来的值。

28033169563c2f1adcee0cc8217ac198.png

图5

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值