当需要C/C++与汇编混合编程时,可以有以下两种处理策略:
用C/C++程序嵌入汇编程序中可以实现一些高级语言没有的功能,提高程序执行效率。armcc编译器的内嵌汇编器支持ARM指令集,tcc编译器的内嵌汇编器支持Thumb指令集。
在ARM的C语言程序中可以使用关键字__asm来加入一段汇编语言的程序,格式如下:
__asm
{
指令 [;指令] /* comments */
...
指令
}
void enable_IRQ(void) //使能中断程序
{
int tmp; //定义临时变量,后面使用
__asm //内嵌汇编程序的关键词
{
MRS tmp, CPSR //把状态寄存器加载给tmp
BIC tmp, tmp, #80 //将IRQ控制位清0
MSR CPSR_c, tmp //加载程序状态寄存器
}
}
void disable_IRQ(void) //禁止中断程序
{
int tmp; //定义临时变量,后面使用
__asm //内嵌汇编程序的关键词
{
MRS tmp, CPSR //把状态寄存器加载给tmp
ORR tmp, tmp, #80 //将IRQ控制位置1
MSR CPSR_c, tmp //加载程序状态寄存器
}
}
__asm
{
MOV R0, x //把x的值给R0
ADD y, R0, x/y //计算x/y时R0的值会被修改
}
int var;
__asm
{
MOV var, x //把x的值给R0
ADD y, var, x/y //计算x/y时R0的值会被修改
}
对于在内嵌汇编语言程序中用到的寄存器,编译器在编译时会自动保存和恢复这些寄存器,用户不用保存和恢复这些寄存器。除了CPSR和SPSR寄存器外,其他物理寄存器在读之前必须先赋值,否则编译器会报错。
int fun (int x)
{
__asm
{
STMFD SP!, {R0} //保存R0,先读后写,汇编出错
ADD R0, x, #1
EOR x, R0, x
LDMFD SP!, {R0} //多余的
}
return x;
}
在C/C++程序中声明的全局变量可以被汇编程序通过地址间接访问。具体访问方法/步骤如下:
1) 在C/C++程序中声明全局变量。
2) 在汇编程序中使用IMPORT/EXTERN伪指令声明引用该全局变量。
3) 使用LDR伪指令读取该全局变量的内存地址。
4) 根据该数据的类型,使用相应的LDR指令读取该全局变量;使用相应的STR指令存储该全局变量的值。对于不同类型的变量,需要采用不同选项的LDR和STR指令,如下表所示。
C/C++语言中的变量类型 |
带后缀的LDR和STR指令 |
描述 |
unsigned char |
LDRB/STRB |
无符号字符型 |
unsigned short |
LDRH/STRH |
无符号短整型 |
unsigned int |
LDR/STR |
无符号整型 |
char |
LDRSB/STRSB |
字符型(8位) |
short |
LDRSH/STRSH |
短整型(16位) |
读取C的一个全局变量,并进行修改,然后保存新的值到全局变量中:
AREA Example4, CODE, READONLY
EXPORT AsmAdd
IMPORT g_cVal @声明外部变量g_cVal,在C中定义的全局变量
AsmAdd
LDR R1, =g_cVal @装载变量地址
LDR R0, [R1] @从地址中读取数据到R0
ADD R0, R0, #1 @加1操作
STR R0, [R1] @保存变量值
MOV PC, LR @程序返回
END
假设在汇编程序中定义了一块内存区域,并保存一串字符,汇编代码如下:
EXPORT Message @声明全局标号
Message DCB "HELLO$" @定义了5个有效字符,$为结束符
extern char* Message;
int MessageLength()
{
int Length = 0;
char *pMessage; //定义字符指针变量
pMessage = Message; //指针指向Message 内存块的首地址
/*while循环,统计字符串的长度*/
while(*pMessage != '$') //$为字符串的结束符
{
Length++;
pMessage++;
}
return(Length); //返回字符串的长度
}
ATPCS基本规则见ATPCS。
汇编程序的设置要遵循ATPCS规则,保证程序调用时参数的正确传递,在这种情况下,C程序可以调用汇编子函数。C程序调用汇编程序的方法如下:
1) 汇编程序中使用EXPORT伪指令声明本子程序可外部使用,使其他程序可调用该子程序。
2) 在C语言程序中使用extern关键字声明外部函数(声明要调用的汇编子程序),才可调用此汇编的子程序。
#include
extern void strcopy(char *d, const char *s); //声明外部函数,即要调用的汇编子程序
int main(void)
{
const char *srcstr = "First ource"; //定义字符串常量
char dststr[] = "Second string-destination"; //定义字符串变量
printf("Before copying: \n");
printf("src=%s, dst=%s\n", srcstr, dststr); //显示源字符串和目标字符串的内容
strcopy(dststr, srcstr); //调用汇编子程序R0=dststr, R1=srcstr
printf("After copying: \n");
printf("src=%s, dst=%s\n", srcstr, dststr); //显示复制后的结果
return(0);
}
AREA Example, CODE, READONLY @声明代码段Example
EXPORT strcopy @声明strcopy,以便外部函数调用
strcopy @ R0为目标字符串的地址, R1为源字符串的地址
LDRB R2, [R1], #1 @读取字节数据,源地址加1
STRB R2, [R0], #1 @保存读取的1字节数据,目标地址加1
CMP R2, #0 @判断字符是否复制完毕
BNE strcopy @没有复制完,继续循环复制
MOV PC, LR
汇编程序设置要遵循APTCS规则,保证程序调用时参数的正确传递。汇编程序调用C程序的方法如下:
1) 在汇编程序中使用IMPORT伪指令声明将要调用的C程序函数。
2) 在调用C程序时,要正确设置入口参数,然后使用BL指令调用。
int sum(int a, int b, int c, int d, int e)
{
return(a+b+c+d+e); //返回5个变量的和
}
AREA Example, CODE, READONLY
IMPORT sum @ 声明外部标号sum,即C函数sum()
EXPORT CALLSUM
CALLSUM
STMFD SP!, {LR} @LR寄存器入栈
MOV R0, #1 @设置sum函数入口参数,R0为参数a
MOV R1, #2 @R1为参数b
MOV R2, #3 @R2为参数c
MOV R3, #5 @参数 e=5,保存到堆栈中
STR R3, {SP, #-4}!
MOV R3, #4 @R3为参数d, d=4
BL sum @调用C程序中的sum函数,结果放在R0中
ADD SP, SP, #4 @调整堆栈指针
LDMFD SP, {PC} @程序返回
END