Ubuntu调用函数分析


俗话说365行,隔行如隔山(恩,我说的)。不同行业有不同行业的规矩,而今天说的Ubuntu操作系统也如此,今天我就闪了一下腰。程序很简单,大概是这样的。

#include
int main()
{
char *msg = "mnzn";
printf("msg is %s", msg);
return 0;
}

很简单吧,可是任凭我在x86-ubuntu的64位机器上怎么写汇编就是不对,刚才突然开窍,反汇编看了看,心里的谜团全部解释清楚了。
 
1 |#include
2 |
3 |int main()
4 |{
5 | int a = 10;
6 | char b = 'a';
7 |
8 | printf("the number is %d, char is %c\n", a, b);
9 | return 0;
10 |}


接下来来看看反汇编的结果吧

File Edit Options Buffers Tools Help
1 |.section .rodata
2 |.LC0:
3 | .string "the number is %d, char is %c\n"
4 |
5 |.text
6 |.globl _start
7 |_start:
8 | # init stack frame
9 | pushq %rbp
10 | movq %rsp, %rbp
11 |
12 | # get local var
13 | subq $16, %rsp
14 |
15 | # int a = 10
16 | movl $10, -8(%rbp)
17 |
18 | # char b = 'a'
19 | movb $97, -1(%rbp)
20 |
21 | # call printf
22 | movsbl -1(%rbp),%edx
23 | movl -8(%rbp), %esi
24 | movl $.LC0, %edi
25 | movl $0, %eax
26 | call printf
27 |
28 | # call exit
29 | movl $0, %edi
30 | xor %eax, %eax
31 | call exit

----31 |-cc-:---F1 print.s ?? 1?? 24 02:40 (Assembler)--L31--C17--Bot
End of buffer


结果来自我们可爱的 emacs

分析下:
第9,10行开始初始化栈帧
13行为局部变量开辟空间
16,19为局部变量赋值

今天的主角就是22行开始的函数调用,在这里我犯了太主观的错误,竟然一只认为我用的操作系统也是用堆栈传递参数,结果不管怎么写都是段错误,唉,吸取教训了,看看吧,如果有三个参数,第一个参数用edi,第二个参数用esi, 第三个参数用edx, 而且要把eax清0, 应该是作为返回值吧。

同样的调用 exit的时候 我仿照了上面的方法,没有出现错误。另,就算不把eax清0,程序暂时没出错,不过手头没有资料。 (作者:木有)

相关内容