ADC0832程序完整版源碼+Proteus仿真
前段時(shí)間一直在為ADC0832的程序感到疑惑,從網(wǎng)上找了很多的代碼,用Proteus仿真,最后都出現(xiàn)了一些奇怪的問(wèn)題,有的根本沒(méi)法讀取數(shù)據(jù),有的數(shù)據(jù)有錯(cuò)誤。
當(dāng)參考電壓為5V時(shí),如果把輸入電壓從0一直調(diào)到5V,讀取的數(shù)據(jù)應(yīng)該是從0到255,2.5V時(shí)應(yīng)該是128。但是我發(fā)現(xiàn)一些源碼在輸入0~2.5V時(shí)讀取出來(lái)的是0~255,到2.5V時(shí)讀取的數(shù)據(jù)為0,從2.5到5V,讀出的值又從0增加到255,始終不正確。今天下午特地查閱的ADC0832英文原版的DataSheet,又參考了一篇中文文檔,終于寫(xiě)出了其完整的程序,并且先后讀取了MSB FIRST DATA和LSB FIRST DATA,進(jìn)行比較,如果兩個(gè)數(shù)據(jù)相等,返回讀取的數(shù)據(jù),否則返回0,這樣可以避免讀取發(fā)生錯(cuò)誤,更穩(wěn)定可靠。并通過(guò)了Proteus仿真。
下圖是ADC0832的時(shí)序圖:
其中T-SetUp為250ns,由于使用的是51單片機(jī),晶振11.0592MHz,機(jī)器周期比這個(gè)值大,可以不考慮,但為了防止出現(xiàn)異常,還是延時(shí)了兩個(gè)機(jī)器周期。注意在第11個(gè)時(shí)鐘下降沿之后,DO上的電平既是MSB FIRST輸出的最后一位,又是LSB FIRST輸出的第一位。以下是讀取ADC0832的代碼。
[cpp] view plain copysbit CS_0832 = P1^0;
sbit CLK_0832 = P1^1;
sbit DO_0832 = P1^2; // DI、DO不同時(shí)有效,可共用一個(gè)接口
sbit DI_0832 = P1^2;
extern void _nop_ ( void );
#define pulse0832() _nop_();_nop_();CLK_0832=1;_nop_();_nop_();CLK_0832=0
//把模擬電壓值轉(zhuǎn)換成8位二進(jìn)制數(shù)并返回
unsigned char read0832()
{
unsigned char i, ch = 0, ch1 = 0;
CS_0832=0; // 片選,DO為高阻態(tài)
DI_0832=1;
// 此處暫停T-SetUp: 250ns (由pulse0832完成)
pulse0832(); // 第一個(gè)脈沖,起始位,DI置高
DI_0832=1;
pulse0832(); // 第二個(gè)脈沖,DI=1表示雙通道單極性輸入
DI_0832=1;
pulse0832(); // 第三個(gè)脈沖,DI=1表示選擇通道1(CH2)
// 51單片機(jī)為準(zhǔn)雙向IO口:應(yīng)先寫(xiě)入1再讀取
DI_0832=1;
// MSB FIRST DATA
for(i = 0; i < 8; ++i) {
pulse0832();
ch <<= 1;
if(DO_0832==1)
ch |= 0x01;
}
// MSB FIRST輸出的最后一位與LSB FIRST輸出的第一位是在
// 同一個(gè)時(shí)鐘下降沿之后,故此處先執(zhí)行讀取,后執(zhí)行pulse
// LSB FIRST DATA
for(i = 0; i < 8; ++i) {
ch1 >>= 1;
if(DO_0832==1)
ch1 |= 0x80;
pulse0832();
}
CS_0832=1; // 取消片選,一個(gè)轉(zhuǎn)換周期結(jié)束
return (ch==ch1) ? ch : 0; // 返回轉(zhuǎn)換結(jié)果
}