ARM 汇编与C调用的若干问题(一般函数调用情况)

时间:2024-11-03 22:05:50

ARM 汇编与C之间的函数调用需要符合ATPCS,建议函数的形参不超过4个,如果形参个数少于或等于4,则形参由R0,R1,R2,R3四个寄存器进行传递;若形参个数大于4,大于4的部分必须通过堆栈进行传递。

R0 用来存放函数的第一个参数,R1用来存放第二个参数,R2用来存放第三个参数,R3用来存放第四个参数。其中R0还用来返回函数的调用结果,对应C函数里面的return value语句中的value 存放在R0中。

ARM堆栈的是满栈FULL STACK,SP指针指向的位置是存放有效数据的地方,若压栈新的数据,必须先改变SP,再向SP里面压入数据。下面结合博客http://blog.sina.com.cn/s/blog_3e5694650100tsbf.html,的内容进行分析。

情景(一)函数形参的个数<= 4

test_asm_args.asm
 
            IMPORT test_c_args ;声明test_c_args函数
            AREA TEST_ASM, CODE, READONLY
            EXPORT test_asm_args
            test_asm_args
            STR lr, [sp, #-4]!

;保存当前LR.栈是满递减栈FD,首先调整SP指针,然后压入LR地址
            ldr r0,=0x10 ;参数 1
            ldr r1,=0x20 ;参数 2
            ldr r2,=0x30 ;参数 3
            ldr r3,=0x40 ;参数 4
            bl test_c_args ;调用C函数
            LDR pc, [sp], #4 ;将LR装进PC(返回main函数) ,PC = LR,SP = SP+4,恢复原来的栈。
                END
 
           void test_c_args(int a,int b,int c,int d)
           {
              printk("test_c_args:\n");
              printk("%0x %0x %0x %0x\n",a,b,c,d);
           }

int main()
           {
              test_asm_args();
              for(;;);
           }

情景二:函数的参数是8个

test_asm_args.asm
//--------------------------------------------------------------------------------

IMPORT test_c_args ;声明test_c_args函数
        AREA TEST_ASM, CODE, READONLY
        EXPORT test_asm_args
test_asm_args
       STR lr, [sp, #-4]! ;保存当前lr
       ldr r0,=0x1 ;参数 1
       ldr r1,=0x2 ;参数 2
       ldr r2,=0x3 ;参数 3
       ldr r3,=0x4 ;参数 4
       ldr r4,=0x8
       str r4,[sp,#-4]! ;参数 8 入栈
       ldr r4,=0x7
       str r4,[sp,#-4]! ;参数 7 入栈
       ldr r4,=0x6
       str r4,[sp,#-4]! ;参数 6 入栈
       ldr r4,=0x5
       str r4,[sp,#-4]! ;参数 5 入栈
       bl test_c_args_lots
       ADD sp, sp, #4     ;清除栈中参数 5,本语句执行完后sp指向参数6 
       ADD sp, sp, #4     ;清除栈中参数 6,本语句执行完后sp指向参数7
       ADD sp, sp, #4     ;清除栈中参数 7,本语句执行完后sp指向参数8
       ADD sp, sp, #4     ;清除栈中参数 8,本语句执行完后sp指向 lr
       LDR pc, [sp],#4    ;将lr装进pc(返回main函数) 
        END
test_c_args.c
//--------------------------------------------------------------------------------
void test_c_args(int a,int b,int c,int d,int e,int f,int g,int h)
{
       printk("test_c_args_lots:\n");
       printk("%0x %0x %0x %0x %0x %0x %0x %0x\n",
              a,b,c,d,e,f,g,h);
}
main.c
//--------------------------------------------------------------------------------
int main()
{
     test_asm_args();
     for(;;);
}