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