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