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

Blum的书上只讲了C语言调用汇编,没讲汇编调用C语言。我自己尝试了下。最终试验成功了,在此写出与大家分享。期间历经无数错误,无数异常,好在最后成功了。

程序实现一个换值功能,在main.s里定义a=10,b=20,然后调用C语言函数把a,b换值。

新建两个文件分别为main.s和pro.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??" 是因为我没有清空缓冲区

这就是在GAS汇编中调用C语言函数。
有问题欢迎讨论。

Show Comments