您的位置 首页 新品

arm汇编语言调用C函数之参数传递

对于ARM体系来说,不同语言撰写的函数之间相互调用(mixcalls)遵循的是ATPCS(ARM-ThumbProcedureCallStandard),ATPCS主要是…

关于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的部分有必要经过仓库进行传递。
咱们先讨论一下形参个数为4的状况.

实例1:
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,=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函数)
END
test_c_args.c
//——————————————————————————–
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);
}
main.c
//——————————————————————————–
int main()
{
test_asm_args();
for(;;);
}

程序从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应该怎样传递参数呢?
实例2:
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(;;);
}

这部分的代码和实例1的代码大部分是相同的,不同的当地是test_c_args的参数个数和test_asm_args的参数传递方法。
在test_asm_args中,参数1~参数4仍是经过R0~R3进行传递,而参数5~参数8则是经过把其压入仓库的方法进行传递,不过要注意这四个入栈参数的入栈次序,是以参数8->参数7->参数6->参数5的次序入栈的。
直到调用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 |
+———-+

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/xinpin/274893.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部