前段时间一直在为ADC0832的程序感到疑惑,从网上找了很多的代码,用Proteus仿真,最后都出现了一些奇怪的问题,有的根本没法读取数据,有的数据有错误。
当参考电压为5V时,如果把输入电压从0一直调到5V,读取的数据应该是从0到255,2.5V时应该是128。但是我发现一些源码在输入0~2.5V时读取出来的是0~255,到2.5V时读取的数据为0,从2.5到5V,读出的值又从0增加到255,始终不正确。今天下午特地查阅的ADC0832英文原版的DataSheet,又参考了一篇中文文档,终于写出了其完整的程序,并且先后读取了MSB FIRST DATA和LSB FIRST DATA,进行比较,如果两个数据相等,返回读取的数据,否则返回0,这样可以避免读取发生错误,更稳定可靠。并通过了Proteus仿真。
下图是ADC0832的时序图:
其中T-SetUp为250ns,由于使用的是51单片机,晶振11.0592MHz,机器周期比这个值大,可以不考虑,但为了防止出现异常,还是延时了两个机器周期。注意在第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
//把模拟电压值转换成8位二进制数并返回
unsigned char read0832()
{
unsigned char i, ch = 0, ch1 = 0;
CS_0832=0; // 片选,DO为高阻态
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口:应先写入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输出的第一位是在
// 同一个时钟下降沿之后,故此处先执行读取,后执行pulse
// LSB FIRST DATA
for(i = 0; i < 8; ++i) {
ch1 >>= 1;
if(DO_0832==1)
ch1 |= 0x80;
pulse0832();
}
CS_0832=1; // 取消片选,一个转换周期结束
return (ch==ch1) ? ch : 0; // 返回转换结果
}