Linux汇编GAS调用C语言函数实例

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语言编程,还专门研究过缓冲区问题。

    编译过程如下:

Linux汇编GAS调用C语言函数实例Linux汇编GAS调用C语言函数实例Linux汇编GAS调用C语言函数实例

    之后可以用GDB调试下。【此处没输出"output??",因为我没有清空缓冲区!】

    就这些。


    有问题欢迎讨论。 

将C语言梳理一下,分布在以下10个章节中:

  1. Linux-C成长之路(一):Linux下C编程概要 http://www.linuxidc.com/Linux/2014-05/101242.htm
  2. Linux-C成长之路(二):基本数据类型 http://www.linuxidc.com/Linux/2014-05/101242p2.htm
  3. Linux-C成长之路(三):基本IO函数操作 http://www.linuxidc.com/Linux/2014-05/101242p3.htm
  4. Linux-C成长之路(四):运算符 http://www.linuxidc.com/Linux/2014-05/101242p4.htm
  5. Linux-C成长之路(五):控制流 http://www.linuxidc.com/Linux/2014-05/101242p5.htm
  6. Linux-C成长之路(六):函数要义 http://www.linuxidc.com/Linux/2014-05/101242p6.htm
  7. Linux-C成长之路(七):数组与指针 http://www.linuxidc.com/Linux/2014-05/101242p7.htm
  8. Linux-C成长之路(八):存储类,动态内存 http://www.linuxidc.com/Linux/2014-05/101242p8.htm
  9. Linux-C成长之路(九):复合数据类型 http://www.linuxidc.com/Linux/2014-05/101242p9.htm
  10. Linux-C成长之路(十):其他高级议题

相关推荐