热门标签 | HotTags
当前位置:  开发笔记 > 后端 > 正文

Ubuntu调用函数分析

俗话说365行,隔行如隔山(恩,我说的)。不同行业有不同行业的规矩,而今天说的Ubuntu操作系统也如此,今天我就闪了一下腰。程序很简单,大概是这样的。#includeintmain(){char*msg="mnzn";printf("msgis%s",msg);return0;}很简单吧,可是任凭我在x86-ubuntu的64位机器上怎么写汇编就是不对,刚

俗话说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,程序暂时没出错,不过手头没有资料。 (作者:木有)


推荐阅读
author-avatar
Kermit68_629
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有