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