pcf8591芯片电路:
在板子上的位置:
打开蓝桥杯竞赛官方提供的芯片手册可以找到以上两个图, 从第三个图我们可以知道芯片的地址前面4位是固定的1001即为9,后面分别是A2,A1,A0,从上面第一个图我们可以看到A2,A1,A0是接地的,所以为000,最后一位R/W是读写控制位,当我们给它发0的时候是写,发1的时候是读。
所以我们可以由上面知道:当我们需要通过单片机给pcf8591写的时候指令为0x90;需要读取pcf8591上的数据的时候指令为0x91。
单片机和pcf8591(A/D,D/A转换芯片)进行通信时,采取的是双总线iic通信。由上面第一个图我们可以知道,pcf8591芯片上的数据总线SDA与单片机的引脚P2^1连接,时钟总线SCL与单片机的引脚P2^0相连接。
由于比赛所用单片机无iic总线硬件设施,我们需要用软件来模拟iic通讯,当然在比赛的时候提供的资料包括了iic总线的底层驱动程序,我们只需要把它加进来调用已经封装好的函数就行了。
下面是官方提供的iic.c文件:
/*
程序说明: IIC总线驱动程序
软件环境: Keil uVision 4.10
硬件环境: CT107单片机综合实训平台 8051,12MHz
日 期: 2011-8-9
*/
#include "reg52.h"
#include "intrins.h"
#define somenop {_nop_();_nop_();_nop_();_nop_();_nop_();}
#define SlaveAddrW 0xA0
#define SlaveAddrR 0xA1
//总线引脚定义
sbit SDA = P2^1; /* 数据线 */
sbit SCL = P2^0; /* 时钟线 */
//总线启动条件
void IIC_Start(void)
{
SDA = 1;
SCL = 1;
somenop;
SDA = 0;
somenop;
SCL = 0;
}
//总线停止条件
void IIC_Stop(void)
{
SDA = 0;
SCL = 1;
somenop;
SDA = 1;
}
//应答位控制
void IIC_Ack(bit ackbit)
{
if(ackbit)
{
SDA = 0;
}
else
{
SDA = 1;
}
somenop;
SCL = 1;
somenop;
SCL = 0;
SDA = 1;
somenop;
}
//等待应答
bit IIC_WaitAck(void)
{
SDA = 1;
somenop;
SCL = 1;
somenop;
if(SDA)
{
SCL = 0;
IIC_Stop();
return 0;
}
else
{
SCL = 0;
return 1;
}
}
//通过I2C总线发送数据
void IIC_SendByte(unsigned char byt)
{
unsigned char i;
for(i=0;i<8;i++)
{
if(byt&0x80)
{
SDA = 1;
}
else
{
SDA = 0;
}
somenop;
SCL = 1;
byt <<= 1;
somenop;
SCL = 0;
}
}
//从I2C总线上接收数据
unsigned char IIC_RecByte(void)
{
unsigned char da;
unsigned char i;
for(i=0;i<8;i++)
{
SCL = 1;
somenop;
da <<= 1;
if(SDA)
da |= 0x01;
SCL = 0;
somenop;
}
return da;
}
下面是官方提供的iic.h文件,比赛时最好检查一下iic.h里面包含完iic.c里面的函数没。
#ifndef _IIC_H
#define _IIC_H
//函数声明
void IIC_Start(void);
void IIC_Stop(void);
void IIC_Ack(bit ackbit);
void IIC_SendByte(unsigned char byt);
bit IIC_WaitAck(void);
unsigned char IIC_RecByte(void);
#endif
pcf8591A/D转换时:可以将连接在芯片上的AIN0,AIN1,AIN3,AIN2通道等的模拟量转换为数字量。
而且PCF8591芯片是一个8位的,通道上的电压取值范围是0-5v,相当于0-255一共256个刻度把5v平均分成了255份,那么1v电压对应的数字量就是51。所以我们在进行ad转换的时候不能简单的把读取到的量显示在数码管显示上,比如现在某个通道是3.1v的,那么它通过pcf8591芯片读取转换之后,我们实际上读取到的是3.1*51=158.1,但是如果题目要求显示电压值的话,我们就需要再把读取出来的数字量给处理一下,比如除以一个51.0。
下面是adc转换的核心代码:
void read_pcf8591(addr)
{
IIC_Start(); //总线起始信号
IIC_SendByte(0x90); //字节写地址
IIC_WaitAck(); //等待应答
IIC_SendByte(addr); //发送控制字
IIC_WaitAck();
IIC_Start();
IIC_SendByte(0x91); //字节读地址
IIC_WaitAck();
dat=IIC_RecByte();
IIC_WaitAck(); //不要乱去省略等待指令,单片机要扯拐
IIC_Ack(0);
IIC_Stop();
dat_v=(float)dat/51.0*100;//处理读到的AIN3的量
从机等待应答部分的代码(即留下空格的部分)可省略。
PCF8591芯片D/A转换时(即把数字量转换为模拟量)
//******D/A转换 写进去的数字 函数 iic通讯******
void write_dac(uint dat)
{
IIC_Start();
IIC_SendByte(0x90); //pcf8591写操作地址
IIC_SendByte(0x40); //选择dac输出通道,第6位为1打开D/A
IIC_SendByte(dat); //选择要给pcf8591芯片的数字量
IIC_Ack(0); //iic总线产生非应答
IIC_Stop(); //停止通讯
}