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(); //按键扫描 |
} |