問題:
我怎樣寫一個C程序回應來自8051串行接口的字符?
解答:
函數_getkey和putchar使用片內串口來完成串行I/O。這些子程序包含在C51庫中,每個函數的源文件位于C51LIB目錄下。參見你的C51用戶手冊獲取更多關于這些子程序的相關信息。
下面的例程設置片內串口,設置波特率,并等待一個字符(使用_getkey函數)。當接收到一個字符后,它使用putchar函數輸出。
#i nclude "reg51.h"
#i nclude "stdio.h"
void main (void)
{
SCON = 0x50; /* SCON: 模式1,8位UART,使能rcvr */
TMOD |= 0x20; /* TMOD: 定時器1,模式2,8位重新裝入 */
TH1 = 0xf3; /* TH1: 2400波特的重裝值 */
TR1 = 1; /* TR1: 定時器1開始定時 */
TI = 1; /* TI: 置位TI,發(fā)送UART的第一個字符 */
while (1)
{
unsigned char aaa;
aaa = _getkey();
putchar(aaa);
}
}
問題:
我想在我的匯編程序中調用一個用C寫的函數。我該怎么做呢?
解答:
最簡單的方法是讓C編譯器為你生成正確的匯編代碼。
假設你有一個C函數叫做“foo”,它只有一個無符號字符型變量的參數,并返回一個無符號字符型數值。在一個新的C文件中寫一個啞函數,也叫做“foo”。示例如下:
#pragma src
extern unsigned char foo(unsigned char);
void dummy(void)
{
unsigned char x, y;
x = 1;
y = foo(x);
}
當文件被編譯時,#pragma SRC指示C編譯器生成匯編代碼。匯編文件的擴展名為“src”。
如果你查看src文件,你就會發(fā)現如何從匯編中調用函數“foo”。該文件顯示了用于傳遞函數變量所用的寄存器和內存地址,以及返回數值用的寄存器和內存地址。另外,它還告訴你正確的函數命名規(guī)則,這對于匯編與C接口是必要的。
然后,你就可以用src文件作為模板來寫自己的匯編調用代碼。注意,你必須在函數中包含EXTRN偽指令,即:
EXTRN CODE (_foo)
問題:
在C51編譯器手冊中,有一個匯編函數調用C函數的例子。那么有沒有C程序調用匯編子程序的例子?
解答:
在那本手冊中沒有,但很容易自己做一個。ASM子程序必須知道參數是怎樣傳遞的,返回值是多少,以及段的命名規(guī)則。遵照下面的步驟你就可以自己制作一個全程。
在C中寫一個簡單的函數,它傳遞參數和返回值的方法與你想用匯編實現的一樣。
使用SRC偽指令(#PRAGMA SRC放在文件頂部)使C編譯器生成一個.SRC文件,而不生成.OBJ文件。
編譯C文件。因為使用了SRC偽指令,所以生成一個.SRC文件。.SRC文件包含你所寫的C代碼生成的匯編代碼。
把.SRC文件重命名為.A51文件。
編輯.A51文件,在匯編函數體中插入你想執(zhí)行的匯編代碼。
例如,下面的代碼
unsigned char my_assembly_func {
unsigned int argument)
{
return (argument + 1); //插入啞元行以訪問所有的變量和intevals。
}
當編譯時生成SRC文件。
問題:
你們有關于C和匯編混合編程的例子嗎?
解答:
下面的例程說明如何在8051程序中混合使用C和匯編。
本例是一個MAIN C函數開始,它調用一個匯編子程序,然后調用一個C函數。
MAIN C模塊如下:
extern void a_func (void);
void main (void)
{
a_func ();
)
函數a_func是一個匯編子程序:
NAME A_FUNC
?PR?a_func?A_FUNC SEGMENT CODE
EXTRN CODE (c_func)
PUBLIC a_func
RSEG ?PR?a_func?A_FUNC
a_func:
USING 0
LCALL c_func
RET
END
注意該匯編程序調用C函數c_func:
{
}
匯編模塊的實際代碼使用SRC編譯指示指令和下面的C源文件生成:
extern void c_func (void);
void a_func (void)
{
c_func ();
}
你可以從Keil網站下載C2ASM2C.ZIP。





