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





