基于I2C的温湿度采集

一、I2C 协议简介

I2C 通讯协议(Inter-Integrated Circuit)是由 Phiilps 公司开发的,由于它引脚少,硬件实现简单,可扩展性强,不需要 USART、CAN 等通讯协议的外部收发设备,现在被广泛地使用在系统内多个集成电路(IC)间的通讯。

二、I2C的分类

1、软件I2C

将芯片的两个GPIO引脚分别用作SCL及SDA,按照I2C的时序要求,直接控制引脚的输出信号(若是接收数据时则读取 SDA 电平),就可以实现I2C通讯。由于是直接控制GPIO引脚的高低电平产生通讯时序,需要由CPU控制每个时刻的引脚状态,所以称为“软件模拟协议”方式,即软件I2C方式。

2、硬件I2C

硬件I2C对应芯片上的I2C外设,具有相应的I2C驱动电路,其所使用的I2C管脚也是专用的,因而效率要远高于软件模拟的I2C,但是程序较为繁琐。硬件I2C是直接调用内部寄存器进行配置。
对于硬件I2C来说,它需要I2C片上外设专门负责实现I2C通讯协议,只要配置好该外设,它就会自动根据协议要求产生I2C的通讯信号,收发数据并缓存起来,CPU只要检测该外设的状态和访问数据寄存器,就能完成数据收发。这种由硬件外设处理I2C协议的方式减轻了CPU的工作负担,并且使软件开发更简单。

3、区别

1)硬件I2C的通信速度要比软件I2C的速度要快。因为硬件I2C是通过片上外设实现的通信,CPU只需要去读写寄存器数据就可以进行I2C通信,程序较为简单并且占用的资源少。而软件I2C则需要程序模拟I2C时序,程序较为复杂且占用资源较多。

2)对于硬件I2C来说,由于芯片I2C外设的IO口已经确定,无法随意更改其他IO口,因而可移植性较差;但是由于软件I2C是通过IO口模拟I2C通信时序实现的通信,因而可移植性比较好,在任何单片机上都可以使用,只需要修改一下通信时间以及配置好IO口就可以实现I2C通信。

3)根据代码量判断,模拟的代码量肯定比固件的要大。
硬件IIC用法比较复杂,模拟IIC的流程更清楚一些。
硬件IIC速度比模拟快,并且可以用DMA
模拟IIC可以在任何管脚上,而硬件只能在固定管脚上。

4)软件I2C是程序员使用程序控制SCL,SDA线输出高低电平,模拟I2C协议的时序。一般较硬件i2c稳定,但是程序较为繁琐,但不难。硬件I2C程序员只要调用I2C的控制函数即可,不用直接的去控制SCL,SDA高低电平的输出。但是有些单片机的硬件I2C不太稳定,调试问题较多。

三、基于I2C硬件协议的AHT20温湿度传感器的数据采集

1、硬件连接

将AHT20上的的SCL,GND,SDA,VCC分别对应接到stm32f103指南者模块的B6,GND,B7,3V3。
在这里插入图片描述

2、代码实现

参考链接: 参考代码.
主函数main.c如下:

#include "led.h"
#include "usart.h"
#include "temhum.h"

 int main(void)
 {	 	
    u32 CT_data[2]={0};
    volatile float  hum=0,tem=0;     
     
    delay_init();	    	 //ÑÓʱº¯Êý³õʼ»¯	  
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //ÉèÖÃNVICÖжϷÖ×é2:2λÇÀÕ¼ÓÅÏȼ¶£¬2λÏìÓ¦ÓÅÏȼ¶
    uart_init(115200);	 //´®¿Ú³õʼ»¯Îª115200
    LED_Init();			     //LED¶Ë¿Ú³õʼ»¯
    temphum_init();     //ATH20³õʼ»¯    
	
	while(1)
	{
        AHT20_Read_CTdata(CT_data);       

        hum = CT_data[0]*100*10/1024/1024;  //¼ÆËãµÃµ½Êª¶ÈÖµ£¨·Å´óÁË10±¶£©
        tem = CT_data[1]*200*10/1024/1024-500;//¼ÆËãµÃµ½Î¶ÈÖµ£¨·Å´óÁË10±¶£©

        printf("湿度:%.1f%%\r\n",(hum/10));
        printf("温度:%.1f度\r\n",(tem/10));
        printf("\r\n");
        
        //ÑÓʱ2s,LEDÉÁ˸Ìáʾ´®¿Ú·¢ËÍ״̬
        LED=0;
        delay_ms(1000);
        LED=1;
        delay_ms(1000);
	 }

}

部分子函数如下:

myiic.c:

#include "myiic.h"
#include "delay.h"
 
//³õʼ»¯IIC
void IIC_Init(void)
{					     
	GPIO_InitTypeDef GPIO_InitStructure;
	RCC_APB2PeriphClockCmd(	RCC_APB2Periph_GPIOB, ENABLE );	//ʹÄÜGPIOBʱÖÓ
	   
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;   //ÍÆÍìÊä³ö
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	GPIO_SetBits(GPIOB,GPIO_Pin_6|GPIO_Pin_7); 	//PB6,PB7 Êä³ö¸ß
}
//²úÉúIICÆðʼÐźÅ
void IIC_Start(void)
{
	SDA_OUT();     //sdaÏßÊä³ö
	IIC_SDA=1;	  	  
	IIC_SCL=1;
	delay_us(4);
 	IIC_SDA=0;//START:when CLK is high,DATA change form high to low 
	delay_us(4);
	IIC_SCL=0;//ǯסI2C×ÜÏߣ¬×¼±¸·¢ËÍ»ò½ÓÊÕÊý¾Ý 
}	  
//²úÉúIICÍ£Ö¹ÐźÅ
void IIC_Stop(void)
{
	SDA_OUT();//sdaÏßÊä³ö
	IIC_SCL=0;
	IIC_SDA=0;//STOP:when CLK is high DATA change form low to high
 	delay_us(4);
	IIC_SCL=1; 
	IIC_SDA=1;//·¢ËÍI2C×ÜÏß½áÊøÐźÅ
	delay_us(4);							   	
}
µÈ´ýÓ¦´ðÐźŵ½À´
·µ»ØÖµ£º1£¬½ÓÊÕÓ¦´ð³É¹¦
        0£¬½ÓÊÕÓ¦´ðʧ°Ü
u8 IIC_Wait_Ack(void)
{
	u8 ucErrTime=0;
	SDA_IN();      //SDAÉèÖÃΪÊäÈë  
	IIC_SDA=1;delay_us(1);	   
	IIC_SCL=1;delay_us(1);	 
	while(READ_SDA)
	{
		ucErrTime++;
		if(ucErrTime>250)
		{
			IIC_Stop();
			return 0;
		}
	}
	IIC_SCL=0;//ʱÖÓÊä³ö0 	   
	return 1; 
}    

//²úÉúACKÓ¦´ð
void IIC_Ack(void)
{
	IIC_SCL=0;
	SDA_OUT();
	IIC_SDA=0;
	delay_us(2);
	IIC_SCL=1;
	delay_us(2);
	IIC_SCL=0;
}
//²»²úÉúACKÓ¦´ð		    
void IIC_NAck(void)
{
	IIC_SCL=0;
	SDA_OUT();
	IIC_SDA=1;
	delay_us(2);
	IIC_SCL=1;
	delay_us(2);
	IIC_SCL=0;
}					 				     
//IIC·¢ËÍÒ»¸ö×Ö½Ú
//·µ»Ø´Ó»úÓÐÎÞÓ¦´ð
//1£¬ÓÐÓ¦´ð
//0£¬ÎÞÓ¦´ð			  
void IIC_Send_Byte(u8 txd)
{                        
    u8 t;   
	SDA_OUT(); 	    
    IIC_SCL=0;//À­µÍʱÖÓ¿ªÊ¼Êý¾Ý´«Êä
    for(t=0;t<8;t++)
    {              
        IIC_SDA=(txd&0x80)>>7;
        txd<<=1; 	  
		delay_us(2);   //¶ÔTEA5767ÕâÈý¸öÑÓʱ¶¼ÊDZØÐëµÄ
		IIC_SCL=1;
		delay_us(2); 
		IIC_SCL=0;	
		delay_us(2);
    }	 
} 	    
//¶Á1¸ö×Ö½Ú£¬ack=1ʱ£¬·¢ËÍACK£¬ack=0£¬·¢ËÍnACK   
u8 IIC_Read_Byte(unsigned char ack)
{
	unsigned char i,receive=0;
	SDA_IN();//SDAÉèÖÃΪÊäÈë
    for(i=0;i<8;i++ )
	{
        IIC_SCL=0; 
        delay_us(2);
		IIC_SCL=1;
        receive<<=1;
        if(READ_SDA)receive++;   
		delay_us(1); 
    }					 
    if (!ack)
        IIC_NAck();//·¢ËÍnACK,±íʾֹͣ½ÓÊÕ
    else
        IIC_Ack(); //·¢ËÍACK,±íʾ¼ÌÐø½ÓÊÕ   
    return receive;
}

temhum.c:

#include "temhum.h"

//¶ÁÈ¡AHT20µÄ״̬¼Ä´æÆ÷
u8 AHT20_Read_Status(void)
{
	u8 Byte_first,flag;	
	IIC_Start();
    IIC_Send_Byte(0x71);
	flag=IIC_Wait_Ack();
	Byte_first = IIC_Read_Byte(flag);
	IIC_NAck();
	IIC_Stop();

	return Byte_first;
}

//ÏòAHT20·¢ËÍACÃüÁî
void AHT20_SendAC(void) 
{
	IIC_Start();
	IIC_Send_Byte(0x70);    //Æô¶¯´«Êäºó·¢Ë͵Ä01110000 £¨×îºó1bit±íʾ¶Á/д 0--д,1--¶Á£©
	IIC_Wait_Ack();
	IIC_Send_Byte(0xac);//0xAC²É¼¯ÃüÁî ÃüÁî²ÎÊýÓÐÁ½¸ö×Ö½Ú£¬µÚÒ»¸ö×Ö½ÚΪ 0x33£¬µÚ¶þ¸ö×Ö½ÚΪ0x00¡£
	IIC_Wait_Ack();
	IIC_Send_Byte(0x33);
	IIC_Wait_Ack();
	IIC_Send_Byte(0x00);
	IIC_Wait_Ack();
	IIC_Stop();

}

//³õʼ»¯AHT20
void AHT20_Init(void)   
{	
	IIC_Init();
	IIC_Start();
	IIC_Send_Byte(0x70);
	IIC_Wait_Ack();
	IIC_Send_Byte(0xa8);//0xA8½øÈëNOR¹¤×÷ģʽ
	IIC_Wait_Ack();
	IIC_Send_Byte(0x00);
	IIC_Wait_Ack();
	IIC_Send_Byte(0x00);
	IIC_Wait_Ack();
	IIC_Stop();

	delay_ms(10);//ÑÓʱ10ms×óÓÒ

	IIC_Start();
	IIC_Send_Byte(0x70);
	IIC_Wait_Ack();
	IIC_Send_Byte(0xbe);//0xBE³õʼ»¯ÃüÁAHT20µÄ³õʼ»¯ÃüÁîÊÇ0xBE,   AHT10µÄ³õʼ»¯ÃüÁîÊÇ0xE1
	IIC_Wait_Ack();
	IIC_Send_Byte(0x08);//Ïà¹Ø¼Ä´æÆ÷bit[3]ÖÃ1£¬ÎªÐ£×¼Êä³ö
	IIC_Wait_Ack();
	IIC_Send_Byte(0x00);
	IIC_Wait_Ack();
	IIC_Stop();
	delay_ms(10);//ÑÓʱ10ms×óÓÒ
}

void temphum_init()
{
	delay_ms(40);//¸ÕÉϵ磬ÑÓʱ40ms²Å¿ÉÒÔ¶Áȡ״̬
	
    //Ê×ÏÈ·¢0x71¶Áȡ״̬×Öbit[3]£¬Èç¹û=1,ΪУ׼Êä³ö£¬ÎÞÐë³õʼ»¯!!!Õý³£Çé¿ö϶Á»ØÀ´µÄ״̬ÊÇ0x1C»òÕßÊÇ0x18,¶Á»ØÀ´ÊÇ0x80±íʾæ״̬;
	if(!((AHT20_Read_Status()&0x08)==0x08))
	{
        AHT20_Init(); //³õʼ»¯AHT20 
	}
}

//ûÓÐCRCУÑ飬ֱ½Ó¶ÁÈ¡AHT20µÄζȺÍʪ¶ÈÊý¾Ý    
void AHT20_Read_CTdata(u32 *ct) 
{
	volatile u8 Byte_1th=0,Byte_2th=0,Byte_3th=0;
    volatile u8 Byte_4th=0,Byte_5th=0,Byte_6th=0;
	u32 RetuData = 0;
	u16 cnt = 0,flag;
	AHT20_SendAC();//ÏòAHT20·¢ËÍACÃüÁî
	delay_ms(80);	//´óÔ¼ÑÓʱ80ms
    
	while(((AHT20_Read_Status()&0x80)==0x80))//Ö±µ½×´Ì¬bit[7]Ϊ0£¬±íʾΪ¿ÕÏÐ״̬£¬ÈôΪ1£¬±íʾæ״̬
	{
		delay_ms(1);
		if(cnt++>=100) break;
	}
    
	IIC_Start();
	IIC_Send_Byte(0x71);
	flag=IIC_Wait_Ack();
	Byte_1th = IIC_Read_Byte(flag);//״̬×Ö
	Byte_2th = IIC_Read_Byte(flag);//ʪ¶È,·¢ËÍACK(¼ÌÐø·¢ËÍ)
	Byte_3th = IIC_Read_Byte(flag);//ʪ¶È
	Byte_4th = IIC_Read_Byte(flag);//ʪ¶È/ζÈ
	Byte_5th = IIC_Read_Byte(flag);//ζÈ
	Byte_6th = IIC_Read_Byte(!flag);//ζÈ,·¢ËÍNACK(Í£Ö¹·¢ËÍ)  
	IIC_Stop();
    
    //±£´æµÃµ½µÄÊý¾Ýµ½RetuDataÖÐ
	RetuData = (RetuData|Byte_2th)<<8;  
	RetuData = (RetuData|Byte_3th)<<8;
	RetuData = (RetuData|Byte_4th);
	RetuData =RetuData >>4;
	ct[0] = RetuData;//ʪ¶È
    
	RetuData = 0;
	RetuData = (RetuData|Byte_4th)<<8;
	RetuData = (RetuData|Byte_5th)<<8;
	RetuData = (RetuData|Byte_6th);
	RetuData = RetuData&0x0fffff;
	ct[1] =RetuData; //ζÈ
}

编译生成hex文件,然后把产生的hex文件烧录到stm32中,打开串口调试助手,即可看到实验结果。

3、实验结果

在这里插入图片描述
对着芯片哈气,发现温度和湿度同时增高,说明实验成功。

四、总结

这次的实验用了一个新的硬件模块,做出来之后,感觉很有趣。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值