十七(4)PCF8591应用程序

  PCF8591 的通信接口是 I2C,那么编程肯定是要符合这个协议的。单片机对 PCF8591 进行初始化,一共发送三个字节即可。第一个字节,和 EEPROM 类似,是器件地址字节,其中 7 位代表地址,1 位代表读写方向。地址高 4 位固定是 0b1001,低三位是 A2,A1,A0,这三位我们电路上都接了 GND,因此也就是 0b000,如图 17-5 所示。 

 

  发送到 PCF8591 的第二个字节将被存储在控制寄存器,用于控制 PCF8591 的功能。其中第 3 位和第 7 位是固定的 0,另外 6 位各自有各自的作用,如图 17-6 所示,我逐一介绍。 

 

  控制字节的第 6 位是 DA 使能位,这一位置 1 表示 DA 输出引脚使能,会产生模拟电压输出功能。第 4 位和第 5 位可以实现把 PCF8591 的 4 路模拟输入配置成单端模式和差分模式,单端模式和差分模式的区别,我们在 17.5 节有介绍,这里大家只需要知道这两位是配置 AD输入方式的控制位即可,如图 17-7 所示。 

  控制字节的第 2 位是自动增量控制位,自动增量的意思就是,比如我们一共有 4 个通道,当我们全部使用的时候,读完了通道 0,下一次再读,会自动进入通道 1 进行读取,不需要我们指定下一个通道,由于 A/D 每次读到的数据,都是上一次的转换结果,所以同学们在使用自动增量功能的时候,要特别注意,当前读到的是上一个通道的值。为了保持程序的通用性,我们的代码没有使用这个功能,直接做了一个通用的程序。 

  控制字节的第 0 位和第 1 位就是通道选择位了,00、01、10、11 代表了从 0 到 3 的一共4 个通道选择。 

  发送给 PCF8591 的第三个字节 D/A 数据寄存器,表示 D/A 模拟输出的电压值。D/A 模拟我们一会介绍,大家知道这个字节的作用即可。我们如果仅仅使用 A/D 功能的话,就可以不发送第三个字节。 

  下面我们用一个程序,把 AIN0、AIN1、AIN3 测到的电压值显示在液晶上,同时大家可以转动电位器,会发现 AIN0 的值发生变化。 

  /***************************Lcd1602.c 文件程序源代码*****************************/

  (此处省略,可参考之前章节的代码)

  /*****************************I2C.c 文件程序源代码*******************************/

  (此处省略,可参考之前章节的代码)

 
  1. /*****************************main.c 文件程序源代码******************************/  
  2. #include <reg52.h>  
  3. bit flag300ms = 1; //300ms 定时标志  
  4. unsigned char T0RH = 0; //T0 重载值的高字节  
  5. unsigned char T0RL = 0; //T0 重载值的低字节  
  6.   
  7. void ConfigTimer0(unsigned int ms);  
  8. unsigned char GetADCValue(unsigned char chn);  
  9. void ValueToString(unsigned char *str, unsigned char val);  
  10. extern void I2CStart();  
  11. extern void I2CStop();  
  12. extern unsigned char I2CReadACK();  
  13. extern unsigned char I2CReadNAK();  
  14. extern bit I2CWrite(unsigned char dat);  
  15. extern void InitLcd1602();  
  16. extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);  
  17.   
  18. void main(){  
  19.     unsigned char val;  
  20.     unsigned char str[10];  
  21.      
  22.     EA = 1; //开总中断  
  23.     ConfigTimer0(10); //配置 T0 定时 10ms  
  24.     InitLcd1602(); //初始化液晶  
  25.     LcdShowStr(0, 0, "AIN0 AIN1 AIN3"); //显示通道指示  
  26.      
  27.     while (1){  
  28.         if (flag300ms){  
  29.             flag300ms = 0; //显示通道 0 的电压  
  30.             val = GetADCValue(0); //获取 ADC 通道 0 的转换值  
  31.             ValueToString(str, val); //转为字符串格式的电压值  
  32.             LcdShowStr(0, 1, str); //显示到液晶上  
  33.             //显示通道 1 的电压  
  34.             val = GetADCValue(1);  
  35.             ValueToString(str, val);  
  36.             LcdShowStr(6, 1, str);  
  37.             //显示通道 3 的电压  
  38.             val = GetADCValue(3);  
  39.             ValueToString(str, val);  
  40.             LcdShowStr(12, 1, str);  
  41.         }  
  42.     }  
  43. }  
  44. /* 读取当前的 ADC 转换值,chn-ADC 通道号 0~3 */  
  45. unsigned char GetADCValue(unsigned char chn){  
  46.     unsigned char val;  
  47.     I2CStart();  
  48.     if (!I2CWrite(0x48<<1)){ //寻址 PCF8591,如未应答,则停止操作并返回 0  
  49.         I2CStop();  
  50.         return 0;  
  51.     }  
  52.     I2CWrite(0x40|chn); //写入控制字节,选择转换通道  
  53.     I2CStart();  
  54.     I2CWrite((0x48<<1)|0x01); //寻址 PCF8591,指定后续为读操作  
  55.     I2CReadACK(); //先空读一个字节,提供采样转换时间  
  56.     val = I2CReadNAK(); //读取刚刚转换完的值  
  57.     I2CStop();  
  58.     return val;  
  59. }  
  60. /* ADC 转换值转为实际电压值的字符串形式,str-字符串指针,val-AD 转换值 */  
  61. void ValueToString(unsigned char *str, unsigned char val){  
  62.     //电压值=转换结果*2.5V/255,式中的 25 隐含了一位十进制小数  
  63.     val = (val*25) / 255;  
  64.     str[0] = (val/10) + '0'//整数位字符  
  65.     str[1] = '.'//小数点  
  66.     str[2] = (val%10) + '0'//小数位字符  
  67.     str[3] = 'V'//电压单位  
  68.     str[4] = '\0'//结束符  
  69. }  
  70. /* 配置并启动 T0,ms-T0 定时时间 */  
  71. void ConfigTimer0(unsigned int ms){  
  72.     unsigned long tmp; //临时变量  
  73.     tmp = 11059200 / 12; //定时器计数频率  
  74.     tmp = (tmp * ms) / 1000; //计算所需的计数值  
  75.     tmp = 65536 - tmp; //计算定时器重载值  
  76.     tmp = tmp + 12; //补偿中断响应延时造成的误差  
  77.     T0RH = (unsigned char)(tmp>>8); //定时器重载值拆分为高低字节  
  78.     T0RL = (unsigned char)tmp;  
  79.     TMOD &= 0xF0; //清零 T0 的控制位  
  80.     TMOD |= 0x01; //配置 T0 为模式 1  
  81.     TH0 = T0RH; //加载 T0 重载值  
  82.     TL0 = T0RL;  
  83.     ET0 = 1; //使能 T0 中断  
  84.     TR0 = 1; //启动 T0  
  85. }  
  86. /* T0 中断服务函数,执行 300ms 定时 */  
  87. void InterruptTimer0() interrupt 1{  
  88.     static unsigned char tmr300ms = 0;  
  89.     TH0 = T0RH; //重新加载重载值  
  90.     TL0 = T0RL;  
  91.     tmr300ms++;  
  92.     if (tmr300ms >= 30){ //定时 300ms  
  93.         tmr300ms = 0;  
  94.         flag300ms = 1;  
  95.     }  
  96. }   

  细心阅读程序的同学会发现,程序在进行 A/D 读取数据的时候,共使用了两条程序去读了 2 个字节:I2CReadACK(); val = I2CReadNAK(); PCF8591 的转换时钟是 I2C 的 SCL,8 个SCL 周期完成一次转换,所以当前的转换结果总是在下一个字节的 8 个 SCL 上才能读出,因此我们这里第一条语句的作用是产生一个整体的 SCL 时钟提供给 PCF8591 进行 A/D 转换,第二次是读取当前的转换结果。如果我们只使用第二条语句的话,每次读到的都是上一次的转换结果。

永不止步步 发表于01-30 10:15 浏览65535次
分享到:

已有0条评论

暂时还没有回复哟,快来抢沙发吧

添加一条新评论

只有登录用户才能评论,请先登录注册哦!

话题作者

永不止步步
金币:67417个|学分:404041个
立即注册
畅学电子网,带你进入电子开发学习世界
专业电子工程技术学习交流社区,加入畅学一起充电加油吧!

x

畅学电子网订阅号