Blum的书上只讲了C语言调用汇编,没讲汇编调用C语言。我自己尝试了下。
最终试验成功了,在此写出与大家分享。期间历经无数错误,无数异常,我不是醉了,而是跪了。。。好在最后好了。
程序实现一个换值功能,在main.s里定义a=10,b=20,然后调用C语言函数把a,b换值。
新建两个文件分别为main.s的汇编文件,还有pro.c的C语言函数文件。
main.s的代码如下:
.section .data
a:
.int 10
b:
.int 20
.section .text
.globl main
.type main,@function #别忘了这句,因为main汇编函数也是被crt0.s调用的,main本质上也是个函数
main:
movl $a,%eax
movl $b,%ebx
pushl %ebx
pushl %eax
call swapint #不要写成 _swapint
movl $1,%eax
movl $0,%ebx
int $0x80
pro.c的代码如下:
#include<stdio.h>
int swapint(int *a,int *b)
{
int c;
char *str="success!!";;
c=*a;
*a=*b;
*b=c;
puts(str); #用puts可以输出
puts("end!"); #用puts可以输出
printf("output??"); #用printf会造成此句无输出,原因:缓冲区没满 #用\n清空缓冲区即可造成输出。
return 0;
}
在汇编函数里先把a,b的地址压栈,注意按照C语言函数参数从右往左的顺序压栈。即先压栈b,后搞a。
之后直接调用即可,CPU会自动把返回地址压栈,然后控制权移交C语言函数,之后就是C语言函数自动取参数,你就不用管了。【其实C语言函数所做的就是8(%ebp)取出堆栈中压入的a的地址(我们压栈的是地址),然后再12(%ebp)取出堆栈中压入的b的地址,然后开始运算换值。。。不神秘】
讲完了,一开始被printf无输出纠结了一会儿,不过以前学过Linux下C语言编程,还专门研究过缓冲区问题。
编译过程如下:
之后可以用GDB调试下。【此处没输出"output??",因为我没有清空缓冲区!】
就这些。
有问题欢迎讨论。
本文出自 “mirage1993” 博客,请务必保留此出处http://mirage1993.blog.51cto.com/2709744/1560510
原文:http://mirage1993.blog.51cto.com/2709744/1560510