C51编译器-高级编程技巧(3)-C语言与汇编的接口
扫描二维码
随时随地手机看文章
C程序与汇编的接口
Cx51程序可以方便地与8051汇编程序接口。A51汇编器是一个以OMF-51格式发射对象模块的8051宏汇编器。通过观察一些编程规则,你可以在C程序中调用汇编程序,反之亦然。在汇编模块中声明的公有变量在C程序中也可以使用。
在C程序中调用汇编程序有几个方面的原因。一是可以使用已有的汇编程序,二是在希望运行速度快的地方使用,三是在希望直接用汇编操作SFR或使用I/O存储器映像的地方使用。
对于一个将在C程序中调用的汇编程序,它的参数传递规和值返回规则必须与C函数一致。就应用上来说,它必须看起来像一个C函数。
Function Parameters函数参数
一般情况下,C函数可以通过寄存器传递三个参数。其他的参数通过固定的存储器传递。当然,可以使用NOREGPARAMS来禁止使用寄存器传递参数。如果指定了不允许使用寄存器传递参数或参数太多的情况下,使用固定的存储区域传递参数。在寄存器中传递参数的函数在生成代码时,名字前加“_”,使用固定存储区域传递参数的函数名字不做变化。
Parameter Passing in Registers
在寄存器中传递参数
C函数在寄存器中也可以在固定存储域传递参数。在寄存器中最多可以传递三个参数。下表显示了参数传递的规则:
Arg Number char, 1-byte ptr int, 2-byte ptr long, float generic ptr
1 R7 R6 & R7 R4—R7 R1—R3
(MSB in R6, (Mem type in R3,
LSB in R7) MSB in R2
LSB in R1)
2 R5 R4 & R5 R4—R7 R1—R3
(MSB in R4, (Mem type in R3
LSB in R5) MSB in R2
LSB in R1)
3 R3 R2 & R3 R1—R3
(MSB in R2, (Mem type in R3
LSB in R3) MSB in R2,
LSB in R1)
例示
Declaration Description
func1 (int a) 参数a通过R6,R7传递
func2 (int b,int c,int *d) 参数b通过R6、R7传递,参数c通过R4,R5传递,参数d通过R1、R2、R3传递
func3 (long e,long f) 参数e通过R4,R5,R6,R7传递,参数f通过固定存储区域传递
func4 (float g,char h) 参数g通过R4,R5,R6、R7传递,参数h通过固定存储区域传递
Parameter Passing in Fixed Memory Locations
通过固定存储区域传递参数
通过固定存储区域传递参数的程序?function_name?BYTE和?function_name?BIT命名并承载参数值并传递到函数function_name。位参数在调用函数前拷贝到?function_name?BIT段。所有其他的参数据被拷贝到?function_name?BYTE段。所有的参数都被指定了在这些段内的存储空间,即使用他们可能使用寄存器来传递参数。参数都按他们声明的顺序保存的相应的段中。
根据不同的存储模式,传递参数的固定存储区域可能是内部数据存储器也可能是外部数据存储器。小存储模式使用内部的数据存储器做参数据传递区域效率是最高的。Compact和大存储模式使用外部数据存储器做为参数传递区。
Function Return Values函数返回值
函数的返回值总是使用CPU寄存器。下表列出了可能的返回值采用的寄存器
返回值类型 寄存器 描述
bit Carry Flag 单个位在进位位里返回
char / unsigned char, R7 单个字节在R7里返回
1-byte pointer
int / unsigned int, R6 & R7 高位在R6,低位在R7
2-byte ptr
long / unsigned long R4-R7 高位在R4,低位在R7
float R4-R7 32位IEEE格式
generic pointer R1-R3 存储器类型在R3,高位在R2,低位R1
Using the SRC Directive使用指令SRC
可以用Cx51编译器来生成一个汇编代码程序,并且用这个程序来决定应该使用的传递规则。指令SRC可以指定Cx51生成一个汇编程序而不是目标程序。例如以下C程序:
#pragma SRC
#pragma SMALL
unsigned int asmfunc1 (
unsigned int arg)
{
return (1 + arg);
}
生成以下的汇编程序:
; ASM1.SRC generated from: ASM1.C
NAME ASM1
?PR?_asmfunc1?ASM1 SEGMENT CODE
PUBLIC _asmfunc1
; #pragma SRC
; #pragma SMALL
;
; unsigned int asmfunc1 (
RSEG ?PR?_asmfunc1?ASM1
USING 0
_asmfunc1:
;---- Variable 'arg?00' assigned to Register 'R6/R7' ----
; SOURCE LINE # 4
; SOURCE LINE # 6
; return (1 + arg);
; SOURCE LINE # 7
MOV A,R7
ADD A,#01H
MOV R7,A
CLR A
ADDC A,R6
MOV R6,A
; }
; SOURCE LINE # 8
?C0001:
RET
; END OF _asmfunc1
END
在这个例子中,函数名asmfunc1,在汇编程序中加了一个下划线前缀,表明参数通过寄存器传递。参数据arg通过寄存器R6,R7传递
以下程序是同一个源程序编译成的汇编程序,只是使用了NOREGPARMS指令
; ASM2.SRC generated from: ASM2.C
NAME ASM2
?PR?asmfunc1?ASM2 SEGMENT CODE
?DT?asmfunc1?ASM2 SEGMENT DATA
PUBLIC ?asmfunc1?BYTE
PUBLIC asmfunc1
RSEG ?DT?asmfunc1?ASM2
?asmfunc1?BYTE:
arg?00: DS 2
; #pragma SRC
; #pragma SMALL
; #pragma NOREGPARMS
;
; unsigned int asmfunc1 (
RSEG ?PR?asmfunc1?ASM2
USING 0
asmfunc1:
; SOURCE LINE # 5
; SOURCE LINE # 7
; return (1 + arg);
; SOURCE LINE # 8
MOV A,arg?00+01H
ADD A,#01H
MOV R7,A
CLR A
ADDC A,arg?00
MOV R6,A
; }
; SOURCE LINE # 9
?C0001:
RET
; END OF asmfunc1
END
注意在例子中函数名asmfunc1前面没有加了下划线前缀,而参数是是通过?asmfunc1?BYTE段递的