D/A 是和 A/D 刚好反方向的,一个 8 位的 D/A,从 0~255,代表了 0~2.55V 的话,那么我们用单片机给第三个字节发送 100,D/A 引脚就会输出一个 1V 的电压,发送 200 就输出一个 2V 的电压,很简单,我们用一个简单的程序实现出来,并且通过上、下按键可以增大或减小输出幅度值,每次增加或减小 0.1V。如果有万用表的话,可以直接测试一下板子上AOUT 点的输出电压,观察它的变化。由于 PCF8591 的 DA 输出偏置误差最大是 50mv(由数据手册提供),所以我们用万用表测到的电压值和理论值之间的误差就应该在 50mV 以内。

/*****************************I2C.c 文件程序源代码*******************************/
(此处省略,可参考之前章节的代码)
/***************************keyboard.c 文件程序源代码****************************/
(此处省略,可参考之前章节的代码)
/*****************************main.c 文件程序源代码******************************/

  • #include <reg52.h>
  • unsigned char T0RH = 0; //T0 重载值的高字节
  • unsigned char T0RL = 0; //T0 重载值的低字节
  • void ConfigTimer0(unsigned int ms);
  • extern void KeyScan();
  • extern void KeyDriver();
  • extern void I2CStart();
  • extern void I2CStop();
  • extern bit I2CWrite(unsigned char dat);

  • void main(){
  •     EA = 1; //开总中断
  •     ConfigTimer0(1); //配置 T0 定时 1ms

  •     while (1){
  •         KeyDriver(); //调用按键驱动
  •     }
  • }
  • /* 设置 DAC 输出值,val-设定值 */
  • void SetDACOut(unsigned char val){
  •     I2CStart();
  •     if (!I2CWrite(0x48<<1)){ //寻址 PCF8591,如未应答,则停止操作并返回
  •         I2CStop();
  •         return;
  •     }
  •     I2CWrite(0x40); //写入控制字节
  •     I2CWrite(val); //写入 DA 值
  •     I2CStop();
  • }
  • /* 按键动作函数,根据键码执行相应的操作,keycode-按键键码 */
  • void KeyAction(unsigned char keycode){
  •     static unsigned char volt = 0; //输出电压值,隐含了一位十进制小数位
  •     if (keycode == 0x26){ //向上键,增加 0.1V 电压值
  •         if (volt < 25){
  •             volt++;
  •             SetDACOut(volt*255/25); //转换为 AD 输出值
  •         }
  •     }else if (keycode == 0x28){ //向下键,减小 0.1V 电压值
  •         if (volt > 0){
  •             volt--;
  •             SetDACOut(volt*255/25); //转换为 AD 输出值
  •         }
  •     }
  • }
  • /* 配置并启动 T0,ms-T0 定时时间 */
  • void ConfigTimer0(unsigned int ms){
  •     unsigned long tmp; //临时变量
  •     tmp = 11059200 / 12; //定时器计数频率
  •     tmp = (tmp * ms) / 1000; //计算所需的计数值
  •     tmp = 65536 - tmp; //计算定时器重载值
  •     tmp = tmp + 28; //补偿中断响应延时造成的误差
  •     T0RH = (unsigned char)(tmp>>8); //定时器重载值拆分为高低字节
  •     T0RL = (unsigned char)tmp;
  •     TMOD &= 0xF0; //清零 T0 的控制位
  •     TMOD |= 0x01; //配置 T0 为模式 1
  •     TH0 = T0RH; //加载 T0 重载值
  •     TL0 = T0RL;
  •     ET0 = 1; //使能 T0 中断
  •     TR0 = 1; //启动 T0
  • }
  • /* T0 中断服务函数,执行按键扫描 */
  • void InterruptTimer0() interrupt 1{
  •     TH0 = T0RH; //重新加载重载值
  •     TL0 = T0RL;
  •     KeyScan(); //按键扫描
  • }