1. 应遵守的规则

  在使用C语言时,要用到和汇编语言的混合编程。若汇编代码较为简洁,则可使用直接内嵌汇编的方法;否则要将汇编程序以文件的形式加入到项目中,按照ATPCS(ARM/Thumb过程调用标准,ARM/Thumb Procedure Call Standard)的规定与C程序相互调用与访问。

  在C程序和ARM汇编程序之间相互调用时,必须遵守ATPCS规则。ATPCS规定了一些子程序间调用的基本规则,哪寄存器的使用规则,堆栈的使用规则和参数的传递规则等。

2. ATPCS规则

2.1 寄存器的使用规则

  寄存器r0~r3:子程序之间通过r0~r3来传递参数,当参数个数多于4个时,使用堆栈来传递参数。此时r0~r3可记作A1~A4。
  寄存器r4~r11:在子程序中使用r4~r11保存局部变量。因此当进行子程序调用时要注意对这些寄存器的保存和恢复。此时r4~r11可记作V1~V8。
  寄存器r12:用于保存堆栈指针SP,当子程序返回时使用该寄存器出栈,记作IP。
  寄存器r13:堆栈指针,记作SP。
  寄存器r14:链接寄存器,记作LR。用于保存子程序的返回地址。
  寄存器r15:程序计数器,记作PC。

2.2 堆栈的使用规则

  ATPCS规定堆栈采用满递减类型(FD, Full Descending),即堆栈通过减小存储器地址而向下增长,堆栈指针指向内含有效数据项的最低地址。

2.3 参数的传递规则

  整数参数:前4个使用r0~r3传递,其他参数使用堆栈传递;浮点参数:使用编号最小且能够满足需要的一组连续的FP寄存器传递参数。

  子程序的返回结果为一个32位整数时,通过r0返回;返回结果为一个64位整数时,通过r0和r1返回;依此类推。结果为浮点数时,通过浮点运算部件的寄存器F0、D0或者S0返回。

3. 汇编程序调用C程序

  汇编程序的书写要遵循ATPCS规则,以保证程序调用时参数正确传递。

在汇编程序中调用C程序的方法为:首先,在汇编程序中使用IMPORT伪指令声明将要调用的C语言函数;然后,通过BL指令来调用C函数。

  例如:在一个C源文件中定义了如下求和函数:

int add(int x,int y)
{
    return(x+y);
}
调用add()函数的汇编程序结构如下:
IMPORT add ;声明要调用的C函数
……
MOV r0,1
MOV r1,2
BL add ;调用C函数add
……

当进行函数调用时,使用r0和r1实现参数传递,返回结果由r0带回。函数调用结束后,r0的值变成3。

4. C程序调用汇编程序

C程序调用汇编程序时,汇编程序的书写也要遵循ATPCS规则,以保证程序调用时参数正确传递。

在C程序中调用汇编子程序的方法为:首先,在汇编程序中使用EXPORT伪指令声明被调用的子程序,表示该子程序将在其他文件中被调用;然后在C程序中使用extern关键字声明要调用的汇编子程序为外部函数。

例如:在一个汇编源文件中定义了如下求和函数:

EXPORT add ;声明add子程序将被外部函数调用
……
add ;求和子程序add
ADD r0,r0,r1
MOV pc,lr
……

在一个C程序的main()函数中对add汇编子程序进行了调用:

extern int add (int x,int y); //声明add为外部函数

void main(){
    int a=1,b=2,c;
    c=add(a,b); //调用add子程序
    ...
}

当main()函数调用add汇编子程序时,变量a、b的值会给了r0和r1,返回结果由r0带回,并赋值给变量c。函数调用结束后,变量c的值变成3。

5. C程序中内嵌汇编语句

在C语言中内嵌汇编语句可以实现一些高级语言不能实现或者不容易实现的功能。

对于时间紧迫的功能也可以通过在C语言中内嵌汇编语句来实现。

内嵌的汇编器支持大部分ARM指令和Thumb指令,但是不支持诸如直接修改PC实现跳转的底层功能,也不能直接引用C语言中的变量。