Keil C51對(duì)同一端口的連續(xù)讀取方法
C語(yǔ)言是當(dāng)前舉世公認(rèn)的高效簡(jiǎn)潔而又非常貼近硬件的編程語(yǔ)言之一。將C語(yǔ)言向單片機(jī)MCS-51上的移植始于2O世紀(jì)8O年代的中后期,經(jīng)過(guò)近1O年的發(fā)展,C語(yǔ)言克服了產(chǎn)生代碼過(guò)長(zhǎng)、運(yùn)行速度較慢的缺點(diǎn),并且由于C語(yǔ)言在開(kāi)發(fā)速度、軟件質(zhì)量、結(jié)構(gòu)化、可維護(hù)性等方面有著匯編語(yǔ)言無(wú)法比擬的優(yōu)勢(shì),從而得到日益廣泛的應(yīng)用。Keil C51是德國(guó)Keil公司開(kāi)發(fā)的單片機(jī)C語(yǔ)言編譯系統(tǒng).該軟件功能完備,是目前國(guó)內(nèi)技術(shù)開(kāi)發(fā)人員使用最為廣泛的語(yǔ)言之一。
在實(shí)際工作中發(fā)現(xiàn),用C語(yǔ)言編寫(xiě)的對(duì)同一端口進(jìn)行連續(xù)讀取的程序,經(jīng)Keil C51編譯后執(zhí)行結(jié)果往往會(huì)出錯(cuò),現(xiàn)以8051單片機(jī)讀取12位A/D MAX197為例,如圖1所示。
圖1中,P1.1口用于讀取轉(zhuǎn)換完成時(shí)A/D發(fā)出的中斷信號(hào),P1.0對(duì)讀取高4位或低8位進(jìn)行選擇?,F(xiàn)假定A/D 的地址為8000H,啟動(dòng)CH0端口工作字為40H。為得到相應(yīng)的高、低位轉(zhuǎn)換數(shù)據(jù),用C語(yǔ)言編程如下。
#include
unsigned char xdata MAX197 _at_ 0x8000;
sbit MAXINT= P1^1;
sbit MAXHBEN= P1^0;
……
void main()
{unsigned char up4,down8;//設(shè)置接收數(shù)據(jù)的2個(gè)變量
……
MAX197= 0X40;//啟動(dòng)A/D CH0口進(jìn)行轉(zhuǎn)換
while(MAXINT) //等待轉(zhuǎn)換完成
{};
P1.0=0; //讀取低8位
down8=MAX197;
P1.0=1; //讀取高4位
up4=MAX197;
}
上述的程序并沒(méi)有如所希望的那樣分別得到高、低位數(shù)據(jù),實(shí)際上在down8和up4中得到的都是低8位的數(shù)據(jù)。下面是上段C語(yǔ)言經(jīng)編譯后的部分代碼。
41: //取低8位
42: MAXHBEN=0;
C:0x000C C290 CLR MAXHBEN(0x90.0)
43: down8=MAX197;
C:0x000E 908000 MOV DPTR,#MAX197(0x8000)
C:0x0011 E0 MOVX A,@DPTR
C:0x0012 F509 MOV 0x09,A
44: //取高4位
45: MAXHBEN=1
C:0x0014 D290 SETB MAXHBEN(0x90.0)
46: up4=MAX197;
47:
48:
C:0x0016 F5O8 MOV 0x08,A //0x08為up4
49: }
通過(guò)分析上面的程序會(huì)發(fā)現(xiàn),C編譯出來(lái)的程序并沒(méi)有在P1.0置為高電位后再去讀一次端口,而只是直接將上次讀來(lái)的結(jié)果直接送給高4位變量。如果先讀高位后讀低位,結(jié)果會(huì)得到兩個(gè)高4位數(shù)據(jù)。為證實(shí)這一點(diǎn),將4條連續(xù)重復(fù)讀取一個(gè)外部端口的C語(yǔ)言語(yǔ)句放在一起,編譯后發(fā)現(xiàn)只有第一條語(yǔ)句被編譯執(zhí)行。也就是說(shuō),Keil C51對(duì)于連續(xù)重復(fù)讀取同一個(gè)端口地址,在編譯時(shí)進(jìn)行了“特殊”處理,這一點(diǎn)是十分值得注意的。那么對(duì)于確實(shí)需要對(duì)同一端口進(jìn)行連續(xù)讀取的情況應(yīng)該如何處理呢?下面介紹兩種方法以供參考。
第一種方法:加延時(shí)。
延時(shí)不宜太長(zhǎng),特別是在對(duì)轉(zhuǎn)換速度要求較高時(shí)。首先寫(xiě)一個(gè)延時(shí)函數(shù):
void delay()
{unsigned char i;
for (i=0,i<=1;i++);
}
然后將延時(shí)程序放在上面兩次讀取的中間位置。
P1.0=0; //讀取低8位
down8=MAX197:
delay();
P1.0=1; //讀取高4位
up4=MAX197;
編譯后的結(jié)果如下:
49: //取低8位
50: MAXHBEN=0:
C:0x000C C29O CLR MAXHBEN(0x90.0)
51: down8=MAX197;
C:0x000E 908000 MOV DPTR,#MAX197(0x8000)
C:0x0011 E0 MOVX A,@DPTR
C:0x0012 F509 MOV 0x09,A
52: delay();
53: //取高4位
C:0x0014 120029 LCALL delay(C:0029)
54: MAXHBEN = 1;
C:0x0017 D290 SETB MAXHBEN(0x90.0)
55:up4=MAX197;
56:
57:
C:0x0019 E0 MOVX A,@DPTR
C:0x001A F508 MOV 0x08,A
58: }
可以看出,在將P1.0置高后,又對(duì)端口進(jìn)行了一次讀寫(xiě),程序正常并得到了高4位。
第二種方法:另設(shè)指針。
void main()
{unsigned char up4,down8; //設(shè)置接收數(shù)據(jù)的2個(gè)變量
unsinged char xdata *pt1;
pt1=0x8000;
……
MAX197=0X40; //啟動(dòng)A/D CH0口進(jìn)行轉(zhuǎn)換
while(MAXINT) //等待轉(zhuǎn)換完成
{};
P1.0=0; //讀取低8位
down8= MAX197:
P1.0=1; //讀取高4位
up4=*pt1:
……
編譯的結(jié)果如下:
42: //取低8位
43: MAXHBEN=0;
C:0x0010 C290 CLR MAXHBEN(0x90.0)
44: down8=MAX197;
C:0x0012 908000 MOV DPTR,#MAX197(0x8000)
C:0x0015 E0 M0VX A,@DPTR
C:0x0016 F509 MOV 0x09,A
45: MAXHBEN=1:
46: //取高4位
47:
C:0x0018 D290 SETB MAXHBEN(0x90.0)
48: up4=*pt1:
49:
50:
C:0x001A 8F82 MOV DPL(0x82),R7
C:0x001C 8E83 MOV DPH (0x83),R6
C:0x001E E0 MOVX A,@DPTR
C:0x001F F508 MOV 0x08,A
上述兩種方法都很好地解決了Keil C51中不能處理對(duì)一個(gè)端口進(jìn)行連續(xù)讀寫(xiě)的問(wèn)題,但如果對(duì)轉(zhuǎn)換速度要求特別高,建議最好使用第二種方法。