C语言函数和汇编函数相互调用

[复制链接]
980|11
 楼主| elecintop 发表于 2015-5-18 22:51 | 显示全部楼层 |阅读模式
在C程序main函数中,接收用户输入任意个整数,然后在main中调用使用ARM汇编编写的函数(在该函数中完成对这些整数的排序功能),然后再在C程序main函数中输出这些排好顺序的整数。
  1. main.c

  2. #include <stdio.h>
  3. int main()

  4. {
  5. int i=0;
  6. int num=0;
  7. int *array=NULL;
  8. while(num <= 0) //输入数组中元素的个数

  9.     {
  10. printf("please enter the number of elements:\n");
  11. scanf("%d",&num);
  12. if(num > 0)

  13.         {
  14. break;
  15. }
  16. }
  17. if(NULL == (array = (int *)malloc(num*sizeof(int))))

  18.     {
  19. printf("malloc failed!\n");
  20. exit(-1);
  21. }
  22. printf("please enter the elements:\n");
  23. for(i = 0; i<num; i++) {
  24. printf("\n%d:\t", i);
  25. scanf("%d", array+i);
  26. }
  27. sort(array, num);//调用相应的汇编的函数,注意分析传参过程
  28.    printf("The Result is:\n");
  29. for(i = 0; i<num; i++) {
  30. printf("%d:\t%d\n", i, *(array+i));
  31. }
  32. return 0;
  33. }


 楼主| elecintop 发表于 2015-5-18 22:51 | 显示全部楼层
下面是相应的Sort.s:

.section .text;声明为代码段
.globl sort;声明全局变量
sort:  ;linux下需要加冒号
mov        r2, #0            
mov        r8, r0
mov        r9, r0
loop1:
sub        r1, r1, #1
cmp        r2, r1         
add        r1, r1, #1
    beq        end     
mov        r6, r2
add        r3, r2, #1         
loop2:
cmp        r3, r1
    beq        continue1
mov        r3, r3, lsl #2
add        r8, r8, r3
    ldr        r5, [r8]
mov        r6, r6, lsl #2
add        r9, r9, r6
    ldr        r4, [r9]
cmp        r4, r5
    bgt        exchange
continue2:
sub        r8, r8, r3
mov        r3, r3, lsr #2
sub        r9, r9, r6
mov        r6, r6, lsr #2
add        r3, r3,    #1
    b        loop2
exchange:
str        r4, [r8]
str        r5, [r9]
        b        continue2
continue1:
add        r2, r2, #1
    b        loop1

end:
 楼主| elecintop 发表于 2015-5-18 22:51 | 显示全部楼层
注意:通过APCS传过来的两个变量,保存在r0和r1,分别代表是数组的首地址和元素的个数
使用Arm交叉编译通过
 楼主| elecintop 发表于 2015-5-18 22:52 | 显示全部楼层
对于ARM体系来说,不同语言撰写的函数之间相互调用(mix calls)遵循的是 ATPCS(ARM-Thumb Procedure Call Standard),ATPCS主要是定义了函数呼叫时参数的传递规则以及如何从函数返回,关于ATPCS的详细内容可以查看ADS1.2 Online Books ——Developer Guide的2.1节。这篇文档要讲的是汇编代码中对C函数调用时如何进行参数的传递以及如何从C函数正确返回。
   不同于x86的参数传递规则,ATPCS建议函数的形参不超过4个,如果形参个数少于或等于4,则形参由R0,R1,R2,R3四个寄存器进行传递;若形参个数大于4,大于4的部分必须通过堆栈进行传递。
 楼主| elecintop 发表于 2015-5-18 22:52 | 显示全部楼层
我们先讨论一下形参个数为4的情况.

实例1:
  1. test_asm_args.asm
  2. //--------------------------------------------------------------------------------
  3.         IMPORT test_c_args ;声明test_c_args函数
  4.         AREA TEST_ASM, CODE, READONLY
  5.         EXPORT test_asm_args
  6. test_asm_args
  7.         STR lr, [sp, #-4]! ;保存当前lr
  8.         ldr r0,=0x10       ;参数 1
  9.         ldr r1,=0x20       ;参数 2
  10.         ldr r2,=0x30       ;参数 3
  11.         ldr r3,=0x40       ;参数 4
  12.         bl test_c_args     ;调用C函数
  13.         LDR pc, [sp], #4   ;将lr装进pc(返回main函数)
  14.         END
  15. test_c_args.c
  16. //--------------------------------------------------------------------------------
  17. void test_c_args(int a,int b,int c,int d)
  18. {
  19.         printk("test_c_args:\n");
  20.         printk("%0x %0x %0x %0x\n",a,b,c,d);
  21. }
  22. main.c
  23. //--------------------------------------------------------------------------------
  24. int main()
  25. {
  26.      test_asm_args();
  27.      for(;;);
  28. }
 楼主| elecintop 发表于 2015-5-18 22:53 | 显示全部楼层
程序从main函数开始执行,main调用了test_asm_args,test_asm_args调用了test_c_args,最后从test_asm_args返回main。代码分别使用了汇编和C定义了两个函数,test_asm_args 和 test_c_args,test_asm_args调用了test_c_args,其参数的传递方式就是向R0~R3分别写入参数值,之后使用bl语句对test_c_args进行调用。其中值得注意的地方是用红色标记的语句,test_asm_args在调用test_c_args之前必须把当前的 lr入栈,调用完test_c_args之后再把刚才保存在栈中的lr写回pc,这样才能返回到main函数中。
   如果test_c_args的参数是8个呢?这种情况test_asm_args应该怎样传递参数呢?
 楼主| elecintop 发表于 2015-5-18 22:54 | 显示全部楼层
实例2:
  1. test_asm_args.asm
  2. //--------------------------------------------------------------------------------
  3.         IMPORT test_c_args ;声明test_c_args函数
  4.         AREA TEST_ASM, CODE, READONLY
  5.         EXPORT test_asm_args
  6. test_asm_args
  7.        STR lr, [sp, #-4]! ;保存当前lr
  8.        ldr r0,=0x1 ;参数 1
  9.        ldr r1,=0x2 ;参数 2
  10.        ldr r2,=0x3 ;参数 3
  11.        ldr r3,=0x4 ;参数 4
  12.        ldr r4,=0x8
  13.        str r4,[sp,#-4]! ;参数 8 入栈
  14.        ldr r4,=0x7
  15.        str r4,[sp,#-4]! ;参数 7 入栈
  16.        ldr r4,=0x6
  17.        str r4,[sp,#-4]! ;参数 6 入栈
  18.        ldr r4,=0x5
  19.        str r4,[sp,#-4]! ;参数 5 入栈
  20.        bl test_c_args_lots
  21.        ADD sp, sp, #4     ;清除栈中参数 5,本语句执行完后sp指向 参数6
  22.        ADD sp, sp, #4     ;清除栈中参数 6,本语句执行完后sp指向 参数7
  23.        ADD sp, sp, #4     ;清除栈中参数 7,本语句执行完后sp指向 参数8
  24.        ADD sp, sp, #4     ;清除栈中参数 8,本语句执行完后sp指向 lr
  25.        LDR pc, [sp],#4    ;将lr装进pc(返回main函数)
  26.        END
  27. test_c_args.c
  28. //--------------------------------------------------------------------------------
  29. void test_c_args(int a,int b,int c,int d,int e,int f,int g,int h)
  30. {
  31.        printk("test_c_args_lots:\n");
  32.        printk("%0x %0x %0x %0x %0x %0x %0x %0x\n",
  33.               a,b,c,d,e,f,g,h);
  34. }
  35. main.c
  36. //--------------------------------------------------------------------------------
  37. int main()
  38. {
  39.      test_asm_args();
  40.      for(;;);
  41. }
 楼主| elecintop 发表于 2015-5-18 22:55 | 显示全部楼层
这部分的代码和实例1的代码大部分是相同的,不同的地方是test_c_args的参数个数和test_asm_args的参数传递方式。
在test_asm_args中,参数1~参数4还是通过R0~R3进行传递,而参数5~参数8则是通过把其压入堆栈的方式进行传递,不过要注意这四个入栈参数的入栈顺序,是以参数8->参数7->参数6->参数5的顺序入栈的。
 楼主| elecintop 发表于 2015-5-18 22:55 | 显示全部楼层
直到调用test_c_args之前,堆栈内容如下:
sp->+----------+
        |  参数5  |
       +----------+
        |  参数6  |
       +----------+
        |  参数7  |
       +----------+
        |  参数8  |
       +----------+
        |   lr   |
       +----------+
test_c_args执行返回后,则设置sp,对之前入栈的参数进行清除,最后将lr装入pc返回main函数,在执行 LDR pc, [sp],#4 指令之前堆栈内容如下:
       +----------+
        |  参数5  |
       +----------+
        |  参数6  |
       +----------+
        |  参数7  |
       +----------+
        |  参数8  |
sp->+----------+
        |   lr   |
       +----------+
旧影子 发表于 2015-6-16 15:27 | 显示全部楼层
嵌入汇编,已经很少用了。在时序很严格的场合,还能见到,一般就是nop延时还用下。
白丁野老 发表于 2015-6-17 07:28 | 显示全部楼层
楼主,可以具体讲讲比如M3是如何内嵌入汇编的么,需要用到这个。
大叔乔东 发表于 2015-6-24 15:31 | 显示全部楼层
看了半天,糊涂了,还是基础不够哇。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

176

主题

1329

帖子

3

粉丝
快速回复 在线客服 返回列表 返回顶部