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