一、原理图
上图可知该芯片使用的是iic时序,而不是51单片机的xpt2046时序,iic我们都很熟悉了吧
并且大赛还提供了我们iic底层驱动代码
左上角有AIN0-4四个转换输入通道,和AOUT一个输出通道,由控制字节选择
地址字节:0x90 | (读写位)
8 | 7 | 6 5 | 4 | 3 | 21 |
---|---|---|---|---|---|
固定为零 | 如果给1就开始DA输出 | 选择单端还是多端通道,我们一般使用单端就可以了,给00 | 固定为0 | 是否开启自动增量,即是否地址自增,我们不需要,就给0 | 通道选择 |
通道选择 | |||||
00-外部输入 | 01-光敏电阻RD1 | 10-接了一个仪表放 大器 | 11-电位器Rb2 |
所以若是使用AD输入就是0x01-光敏转换 0x03-电位器转换
DA输出就是0x41 0x43 等等
如图,寄存器是8位的,所以数据是从0-255的数字量,我们要把它转化成0-5V就是直接可以
DATA*5.0/255
二、上代码
1.底层驱动iic
#include "iic.h"
sbit SCL = P2^0;
sbit SDA = P2^1;
#define DELAY_TIME 5
//I2C总线内部延时函数
void IIC_Delay(unsigned char i)
{
do{_nop_();}
while(i--);
}
//I2C总线启动信号
void IIC_Start(void)
{
SDA = 1;
SCL = 1;
IIC_Delay(DELAY_TIME);
SDA = 0;
IIC_Delay(DELAY_TIME);
SCL = 0;
}
//I2C总线停止信号
void IIC_Stop(void)
{
SDA = 0;
SCL = 1;
IIC_Delay(DELAY_TIME);
SDA = 1;
IIC_Delay(DELAY_TIME);
}
//发送应答或非应答信号
void IIC_SendAck(bit ackbit)
{
SCL = 0;
SDA = ackbit;
IIC_Delay(DELAY_TIME);
SCL = 1;
IIC_Delay(DELAY_TIME);
SCL = 0;
SDA = 1;
IIC_Delay(DELAY_TIME);
}
//等待应答
bit IIC_WaitAck(void)
{
bit ackbit;
SCL = 1;
IIC_Delay(DELAY_TIME);
ackbit = SDA;
SCL = 0;
IIC_Delay(DELAY_TIME);
return ackbit;
}
//I2C总线发送一个字节数据
void IIC_SendByte(unsigned char byt)
{
unsigned char i;
for(i=0; i<8; i++)
{
SCL = 0;
IIC_Delay(DELAY_TIME);
if(byt & 0x80) SDA = 1;
else SDA = 0;
IIC_Delay(DELAY_TIME);
SCL = 1;
byt <<= 1;
IIC_Delay(DELAY_TIME);
}
SCL = 0;
}
//I2C总线接收一个字节数据
unsigned char IIC_RecByte(void)
{
unsigned char i, da;
for(i=0; i<8; i++)
{
SCL = 1;
IIC_Delay(DELAY_TIME);
da <<= 1;
if(SDA) da |= 1;
SCL = 0;
IIC_Delay(DELAY_TIME);
}
return da;
}
2.AD/DA转换
数据帧和AT24C02数据帧一样
void PCF_DA(float Data,u8 Control_Byte)
{
IIC_Start();
IIC_SendByte(0x90);//地址
IIC_WaitAck();
IIC_SendByte(Control_Byte);//控制字节
IIC_WaitAck();
IIC_SendByte(Data);
IIC_WaitAck();
IIC_Stop();
}
u8 PCF_AD(u8 Control_Byte)
{
u8 Data;
IIC_Start();
IIC_SendByte(0x90);
IIC_WaitAck();
IIC_SendByte(Control_Byte);
IIC_WaitAck();
// 8 7 65 4 3 21
// 固定为0 1DA/0AD 单端模式给00 固定为0 不自动增量给0 通道选择
//00-外部输入 01-RD1光敏电阻 02-仪表放大器 03-Rb2电位器
IIC_Start();
IIC_SendByte(0x91);
IIC_WaitAck();
Data = IIC_RecByte();
IIC_SendAck(1);
IIC_Stop();
return Data;
}
3.使用
u16 RD1 = 0;
float temp=0;
void main()
{
ALL_INIT();
while(1)
{
PCF_PROC();
}
}
void PCF_PROC()
{
temp=PCF_AD(0x01)*5.0/255;//先转换成0-5v的范围
RD1 = temp*100;//*100扩大范围,目的是取出小数
seg_set(16,16,16,16,16,RD1/100+32,RD1/10%10,RD1%10);
}