C51編譯器-高級(jí)編程技巧(3)-C語(yǔ)言與匯編的接口
C程序與匯編的接口
Cx51程序可以方便地與8051匯編程序接口。A51匯編器是一個(gè)以O(shè)MF-51格式發(fā)射對(duì)象模塊的8051宏匯編器。通過(guò)觀察一些編程規(guī)則,你可以在C程序中調(diào)用匯編程序,反之亦然。在匯編模塊中聲明的公有變量在C程序中也可以使用。
在C程序中調(diào)用匯編程序有幾個(gè)方面的原因。一是可以使用已有的匯編程序,二是在希望運(yùn)行速度快的地方使用,三是在希望直接用匯編操作SFR或使用I/O存儲(chǔ)器映像的地方使用。
對(duì)于一個(gè)將在C程序中調(diào)用的匯編程序,它的參數(shù)傳遞規(guī)和值返回規(guī)則必須與C函數(shù)一致。就應(yīng)用上來(lái)說(shuō),它必須看起來(lái)像一個(gè)C函數(shù)。
Function Parameters函數(shù)參數(shù)
一般情況下,C函數(shù)可以通過(guò)寄存器傳遞三個(gè)參數(shù)。其他的參數(shù)通過(guò)固定的存儲(chǔ)器傳遞。當(dāng)然,可以使用NOREGPARAMS來(lái)禁止使用寄存器傳遞參數(shù)。如果指定了不允許使用寄存器傳遞參數(shù)或參數(shù)太多的情況下,使用固定的存儲(chǔ)區(qū)域傳遞參數(shù)。在寄存器中傳遞參數(shù)的函數(shù)在生成代碼時(shí),名字前加“_”,使用固定存儲(chǔ)區(qū)域傳遞參數(shù)的函數(shù)名字不做變化。
Parameter Passing in Registers
在寄存器中傳遞參數(shù)
C函數(shù)在寄存器中也可以在固定存儲(chǔ)域傳遞參數(shù)。在寄存器中最多可以傳遞三個(gè)參數(shù)。下表顯示了參數(shù)傳遞的規(guī)則:
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) 參數(shù)a通過(guò)R6,R7傳遞
func2 (int b,int c,int *d) 參數(shù)b通過(guò)R6、R7傳遞,參數(shù)c通過(guò)R4,R5傳遞,參數(shù)d通過(guò)R1、R2、R3傳遞
func3 (long e,long f) 參數(shù)e通過(guò)R4,R5,R6,R7傳遞,參數(shù)f通過(guò)固定存儲(chǔ)區(qū)域傳遞
func4 (float g,char h) 參數(shù)g通過(guò)R4,R5,R6、R7傳遞,參數(shù)h通過(guò)固定存儲(chǔ)區(qū)域傳遞
Parameter Passing in Fixed Memory Locations
通過(guò)固定存儲(chǔ)區(qū)域傳遞參數(shù)
通過(guò)固定存儲(chǔ)區(qū)域傳遞參數(shù)的程序?function_name?BYTE和?function_name?BIT命名并承載參數(shù)值并傳遞到函數(shù)function_name。位參數(shù)在調(diào)用函數(shù)前拷貝到?function_name?BIT段。所有其他的參數(shù)據(jù)被拷貝到?function_name?BYTE段。所有的參數(shù)都被指定了在這些段內(nèi)的存儲(chǔ)空間,即使用他們可能使用寄存器來(lái)傳遞參數(shù)。參數(shù)都按他們聲明的順序保存的相應(yīng)的段中。
根據(jù)不同的存儲(chǔ)模式,傳遞參數(shù)的固定存儲(chǔ)區(qū)域可能是內(nèi)部數(shù)據(jù)存儲(chǔ)器也可能是外部數(shù)據(jù)存儲(chǔ)器。小存儲(chǔ)模式使用內(nèi)部的數(shù)據(jù)存儲(chǔ)器做參數(shù)據(jù)傳遞區(qū)域效率是最高的。Compact和大存儲(chǔ)模式使用外部數(shù)據(jù)存儲(chǔ)器做為參數(shù)傳遞區(qū)。
Function Return Values函數(shù)返回值
函數(shù)的返回值總是使用CPU寄存器。下表列出了可能的返回值采用的寄存器
返回值類型 寄存器 描述
bit Carry Flag 單個(gè)位在進(jìn)位位里返回
char / unsigned char, R7 單個(gè)字節(jié)在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 存儲(chǔ)器類型在R3,高位在R2,低位R1
Using the SRC Directive使用指令SRC
可以用Cx51編譯器來(lái)生成一個(gè)匯編代碼程序,并且用這個(gè)程序來(lái)決定應(yīng)該使用的傳遞規(guī)則。指令SRC可以指定Cx51生成一個(gè)匯編程序而不是目標(biāo)程序。例如以下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
在這個(gè)例子中,函數(shù)名asmfunc1,在匯編程序中加了一個(gè)下劃線前綴,表明參數(shù)通過(guò)寄存器傳遞。參數(shù)據(jù)arg通過(guò)寄存器R6,R7傳遞
以下程序是同一個(gè)源程序編譯成的匯編程序,只是使用了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
注意在例子中函數(shù)名asmfunc1前面沒(méi)有加了下劃線前綴,而參數(shù)是是通過(guò)?asmfunc1?BYTE段遞的