蓝桥杯速成5-AD/DA模数转换

本文介绍了使用IIC总线进行通信的原理,包括时序、地址选择和数据传输,并给出了底层驱动代码示例。同时详细解释了如何通过控制字节实现AD(模拟到数字)和DA(数字到模拟)转换,以及在实际应用中的数据处理过程。
摘要由CSDN通过智能技术生成

一、原理图

上图可知该芯片使用的是iic时序,而不是51单片机的xpt2046时序,iic我们都很熟悉了吧

并且大赛还提供了我们iic底层驱动代码

左上角有AIN0-4四个转换输入通道,和AOUT一个输出通道,由控制字节选择

地址字节:0x90 | (读写位)

876 54321
固定为零如果给1就开始DA输出选择单端还是多端通道,我们一般使用单端就可以了,给00固定为0是否开启自动增量,即是否地址自增,我们不需要,就给0通道选择
                                          通道选择        
00-外部输入01-光敏电阻RD110-接了一个仪表放  大器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);
}

  • 13
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值