前段时间一直在为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的代码。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 sbit 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; // 返回转换结果 }
Proteus仿真文件以及英文DataSheet、中文介绍请点击此处下载。
http://download.csdn.net/detail/jzj1993/5141464