C51編程經(jīng)驗三則
在單片機的開(kāi)發(fā)應用中,已逐漸開(kāi)始引入高級語(yǔ)言,C語(yǔ)言就是其中的一種。用慣了匯編的人,總覺(jué)得高級語(yǔ)言“可控性”不好,不如匯編那樣隨心所欲。以下是筆者在C51編程中的幾點(diǎn)經(jīng)驗,希望對初學(xué)C51者有所幫助。
一、C51熱啟動(dòng)代碼的編制
工業(yè)控制計算機,往往設有看門(mén)狗電路,看門(mén)狗動(dòng)作,計算機復位,這就是熱啟動(dòng)。熱啟動(dòng)時(shí),一般不允許程序從頭開(kāi)始,因為這將使測量或計算值復位,導致系統工作異常。故程序必須判斷是熱啟動(dòng)還是冷啟動(dòng)。常用的方法是:設定某內存單位為標志位(如0x7f位和0x7e位),啟動(dòng)時(shí)首先讀該內存單元的內容,如果它等于一個(gè)特定的值(例如兩個(gè)內存單元的都是0xaa),就認為是熱啟動(dòng),否則就是冷啟動(dòng),程序執行初始化部分,并將0xaa賦予這兩個(gè)內存單元。
根據以上的設計思路,編程時(shí),設置一個(gè)指針,指向特定的內存單元如0x7f,然后在程序中根據特定內存單元的值判斷冷/熱啟動(dòng),程序如下:
void main()
{
if((*HotPoint==0xaa)&&(*(--Hot
Point)==0xaa))
else
{ HotPoint=0x7e; /*冷啟動(dòng)的處理
}
/*正常工作代碼*/
}
實(shí)際調試中發(fā)現,無(wú)論是熱啟動(dòng)還是冷啟動(dòng),開(kāi)機后所有內存單元的值都被復位為0,實(shí)現不了熱啟動(dòng)的要求。這是為什么呢?原來(lái),用C語(yǔ)言編程時(shí),開(kāi)機時(shí)執行的代碼并非是從main()函數的第一語(yǔ)句開(kāi)始的,在main()函數的第一語(yǔ)句執行前要先執行一段‘起始代碼’。正是這段代碼執行了內存清零的工作。C編譯程序提供了這段起始代碼的源程序,名為CSTARTUPA51,打開(kāi)這個(gè)文件,可以看到如下代碼:
IDATALEN EQU 8011 the length of IDATA memory m bytes
STARTUP1:
IF IDATALEN<>0
IDATALOOP:
ENDIF
可見(jiàn),在執行到判斷是否熱啟動(dòng)的代碼之前,起始代碼已將所有內存單元清零。如何解決這個(gè)問(wèn)題呢?好在起始代碼是可以更改的,方法是:修改startup.a51源文件,然后用編譯程序所附帶的a51.exe程序對startup.a51編譯,得到startup.obj文件,用這段代碼代替原來(lái)的起始代碼。具體步驟是(設C源程序名為HOTSTARTC):
1修改startup.a51源文件(這個(gè)文件在C51LIB目錄下)。
2執行如下命令:
A51 startup.a51得到startup.obj文件。將此文件拷入HOTSTARTC所在目錄。
3將編好的C源程序用C51EXE編譯好,得到目標文件HOTSTARTOBJ。
4用L51 HOTSTART,STARTUPOBJ命令連接,得到絕對目標文件HOTSTART。
5用OHS51 HOTSTART得到HOTSTARTHEX文件,即可完成啟動(dòng)代碼的修改。
對于startup.a51的修改,根據自己的需要進(jìn)行,如將IDATALEN EQU 80H中的80H改為70H,就可以使6F到7F的16字節內存不被清零。
二、直接調用EPROM中已固化的程序
筆者用的仿真機,由6位數碼管顯示,在DE00H處存放顯示子程序,只要將顯示的數存入顯示緩沖區,然后調用顯示子程序就可以了,匯編指令為:
LCALL 0DE00H
在用C語(yǔ)言編程時(shí),如何實(shí)現這一功能呢?C語(yǔ)言中有指向函數的指針這一概念,可以用來(lái)實(shí)現用函數指針調用函數。指向函數的指針變量的定義格式為:
類(lèi)型標識符(*指針變量名)();
在定義好指針后就可以給指針變量賦值,使其指向某個(gè)函數的開(kāi)始地址,然后用(*指針變量名)()即可調用這個(gè)函數。程序如下例:
void main(void)
{
void (*DispBuffer)();/*定義指向函數指針*/
DispBuffer=0xde00;
for(;;)
{ Key();
}
}
三、將浮點(diǎn)數轉化為字符數組
筆者在編制應用程序時(shí)有這樣的要求:將運算的結果(浮點(diǎn)數)存入E2PROM中。我們知道,浮點(diǎn)數在C語(yǔ)言中是以IEEE格式存儲的,一個(gè)浮點(diǎn)數占四個(gè)字節。例如浮點(diǎn)數34526存為160、26、10、66四個(gè)數。要將該浮點(diǎn)數存入E2PROM,實(shí)際上就是要存這四個(gè)數。如何在程序中得到一個(gè)浮點(diǎn)數的組成數呢?
浮點(diǎn)數在存儲時(shí),是存儲在連續的字節中的,只要設法找到存儲位置,就可以得到這些數了??梢远x一個(gè)void指針,將此指針指向需要存儲的浮點(diǎn)數,然后再將此指針強制轉化為char型。這樣,利用指針就可以得到組成該浮點(diǎn)數的各個(gè)字節的值了。具體程序如下:
#define uchar unsigned char
#define uint unsigned int
void FtoC(void)
{
void *pf;
px=x;
pf=&a;/*void型指針指向浮點(diǎn)數首地址*/
a=34.526;
for(I=0;I<4;I++)
}
}
如果已將數存入E2PROM,要將其取出合并,方法也是一樣,可參考下面的程序。
#define uchar unsigned char
#define uint unsigned int
void CtoF(void)
{ float a;
uchar I,*px
uchar x[4]-{56,180,150,73};
void *pf;
px=x;
pf=&a;
for(I=o;I<4;I++)
{?。?(char *)pf+I)=*(px+I)
}
}
以上程序所用C語(yǔ)言為FRANKLIN C51 VER 32。
評論