在C/C++与汇编言语混合编程的情况下,一般咱们都会挑选C/C++来完结所等待的大部分功用,关于少量和硬件相关度高(例如操作某些CPU寄存器)以及对运算的实时性要求高(例如高速、多点的FFT)的功用才运用汇编来完结,这就使得大多数情况下,C/C++与汇编的交互都是从C/C++代码调用汇编代码中的函数与变量,所以在此咱们就来看一下这种调用的规矩。
1.从C/C++中调用汇编代码中的函数
假如一个在汇编代码中界说的函数需求在C/C++中被调用,那么这个汇编函数相关于C/C++代码来说,相当于一个外部的函数,所以需求运用extern C关键字进行特别声明,使得编译器和链接器能够知道这个函数并不存在于当时的C/C++代码中。
注:
(1)假如C/C++中的函数需求在汇编代码中被调用,则在C/C++代码中,相同需求运用extern C关键字进行特别声明,这也是extern关键字的多用途地点。
(2)在C++程序中,extern C声明用来告知编译器运用C言语的命名规矩,而不是运用C++中的函数命名转化(Name Mangling)在链接时对函数名进行修正,否则就找不到对应的汇编函数了(mangle在英语中是“乱砍”的意思,能够据此幻想一下它的作用,不知道最初创造C++言语的人是怎样想到的。。。)。
举例说明一个汇编函数asmfunc是如安在C++中的main函数中被调用的:
C/C++代码:
extern C{
extern int asmfunc(int a); /* 声明外部的汇编函数*/
int gvar = 0; /*界说大局变量*/
}
void main()
{
int i = 5;
i = asmfunc(i); /*调用汇编函数 */
}
汇编代码中的汇编函数界说:
.global _gvar
.global _asmfunc
_asmfunc:
MOVZ DP,#_gvar
ADDB AL,#5
MOV @_gvar,AL
LRETR
当链接器从符号表中解析到.global _asmfunc这条句子的时分,它就能够把汇编代码中的asmfunc函数与C/C++中调用的汇编函数给相关上了。
2.运用内联函数法调用汇编函数
这种办法一般用于引证单条的汇编句子,例如:
asm(;*** this is an assembly language comment);
上面比如并没有影响任何的变量,它的作用只是在C/C++代码编译成汇编代码之后,在相对应的方位刺进了一端汇编代码的注释,对调试特别有协助。
当然,咱们也能够刺进特定的汇编函数进完结特定的功用。例如,在DSP的编程中,咱们常常运用的EALLOW和EDIS句子其实便是这种办法的典型比如,只不过为了书写的简洁,咱们在头文件中进行了简略的转化:
#define EALLOW asm( EALLOW)
#define EDIS asm( EDIS)
运用这样的内联函数调用办法,有必要紧记以下五点:
(1) 该办法有可能会损坏代码的优化作用。
(2) 不要内嵌汇编中的跳转或许符号(label)等指令或许伪指令,它会寄存器的值,形成不行意料的成果。
(3) 不要在内嵌的汇编句子中改动C/C++变量的值,由于有可能会发生意料之外的成果。
(4) 不要在内嵌的汇编句子中运用汇编言语的指示性指令(directives)。
(5) 防止在C代码中运用内嵌汇编句子界说汇编的宏,一起运用-debug:dwarf (即-g)选项来编译,由于二者是不兼容的。
3.从C/C++中调用汇编代码中的变量或许常量
为了调试等功用的便利,有时分咱们需求直接在C/C++代码中运用汇编代码中的变量值或许状况等。依据汇编变量/常量的类型,详细的调用的办法也不相同。
3.1 调用汇编中的大局变量
从C/C++中调用汇编中的大局变量的办法与调用汇编函数的办法相似,都是比较直观的:
(1) 在汇编中运用.bss或许.usect指令界说变量
(2) 在汇编中运用.def或许.global指令把变量声明为大局变量
(3) 在汇编中运用特色的链接命名规矩
(4) 在C/C++中,用extern声明在汇编中现已界说的变量,然后就依照一般变量的运用规矩进行调用即可。
例如,在汇编代码中界说大局变量var:
.bss _var,1 ; 界说变量
.global _var ; 声明为大局类型
在C代码中调用该变量:
extern int var; /* 声明var为外部变量 */
var = 1; /* 运用汇编变量 */
3.2 调用汇编中的常量
变量constant与常量的一个明显区别是,编译器编译发生的符号表中会包括变量的地址,所以在对变量进行引证时,编译器能够直接从符号表中找到对应的地址;可是对汇编常量而言,符号表中保存的是它的值,而不是它的地址,所以假如在C/C++中直接运用汇编常量的姓名,需求运用取地址符才干得到正确的值:用C/C++言语编程的话天然不会生疏,即假如x是汇编代码中的常量,需求在C/C++中运用x对其进行调用才干得到正确的成果;调用的规矩与变量是相同的。
例如,在汇编代码中界说常量table_size:
_table_size .set 10000 ; 界说常量table_size=10000
.global _table_size ; 声明为大局类型
在C代码中调用该变量:
extern int table_size; /*声明外部引证,而且运用链接命名规矩*/
#define TABLE_SIZE ((int) (table_size))
. /* 用来引证汇编常量,用#define来防止每次书写 */
.
.
for (i=0; i
4. 在汇编代码中同享C/C++的头文件
既然是C/C++与汇编代码的交互,那么就要既有“来”,又有“往”,咱们能够经过在汇编代码中运用.cdecls指令声明某些变量,然后告诉编译器把C/C++头文件中的这些变量转化为汇编代码能够运用的信息。其调用格局为:
.cdecls [options ,] filename [, filename2 [,…]]
或许
.cdecls [options]
%{
/*———————————————————————————*/
/* C/C++ code – Typically a list of #includes and a few defines */
/*———————————————————————————*/
%}
例如,在C/C++头文件myheader.h中界说
#define WANT_ID 10
#define NAME John\n
extern int a_variable;
extern float cvt_integer(int src);
struct myCstruct { int member_a; float member_b; };
enum status_enum { OK = 1, FAILED = 256, RUNNING = 0 };
然后在汇编代码中运用.cdecls就能够引证这头文件了:
.cdecls C,LIST,myheader.h
size: .int $sizeof(myCstruct)
aoffset: .int myCstruct.member_a
boffset: .int myCstruct.member_b
okvalue: .int status_enum.OK
failval: .int status_enum.FAILED
.if $$defined(WANT_ID)
id .cstring NAME
.endif
究竟专门运用汇编代码进行DSP编程的份额不高,在此就不对汇编编程的细节进行剖析了,需求更详细信息的读者可参阅《TMS320C28x Assembly Language Tools User’s Guide》。