STM32(I2C)接口采集AHT20温湿度

一、I2C集成电路总线协议

I²C(Inter-Integrated Circuit)字面上的意思是集成电路之间,它其实是I²C Bus简称,所以中文应该叫集成电路总线,它是一种串行通信总线,使用多主从架构,由飞利浦公司在1980年代为了让主板、嵌入式系统或手机用以连接低速周边设备而发展。I²C的正确读法为“I平方C”(“I-squared-C”)。

它支持设备之间的短距离通信,经常用于处理器和一些外围设备之间的接口通信。I2C总线的标准通信速率是100Kbps,快速模式是400Kbps,高速模式支持3.4Mbps。I2C总线支持多设备的通信,而且各个设备之间的SCL和SDA线都是线与关系。I2C总线上扩展的器件的数量主要由电容负载来决定,其负载能力为400pF。I2C总线具有极低的电流消耗。

1、I2C物理层

I²C只使用两条双向漏极开路(Open Drain)线,其中一条线为传输数据的串行资料线(SDA),另一条线是启动或停止传输以及发送时钟序列的串行时脉(SCL)线,这两条线上都有上拉电阻[2]。I²C允许相当大的工作电压范围,但典型的电压准位为+3.3V或+5v。
Alt

2、I2C协议层

I2C总线上的每一个设备都可以作为主设备或者从设备,而且每一个设备都会对应一个唯一的设备地址。通常的我们将CPU模块作为主设备,而挂接在总线上的其他设备作为从设备。I2C总线上的主设备与从设备之间以8字节为单位进行双向数据传输,并且每个单位后还须跟着一位ACK位。其中数据在SCL处于低电平时被放到SDA数据线上,在SCL处于高电平时进行数据的采样。下图是I2C总线的数据传输协议时序图。
Alt

Alt

由图可知,I2C总线的传输时序包括:开始条件、地址帧、数据帧、停止条件、重复开始条件。

开始条件:标识传输正式开始,当SCL处于高电平时,SDA由高电平变为低电平。这样所有Slave设备都会知道传输已经开始。

地址帧:地址帧总是在一次通信的最开始出现,通常包括7位的设备地址(MSB)和最后1位的读写控制位(1表示读,0表示写)。接下来是1位的NACK/ACK,当这8位地址发送完成后,Slave设备获得SDA的控制权,此时Slave设备应该在第9个时钟脉冲之前回复一个ACK(将SDA拉低)以表示数据接收正常,否则表示数据接受失败,控制权交由Master设备处理。

数据帧:在地址帧发送之后,就可以开始传送数据了。每个数据帧8位,数据帧的数量可以是任意的,直到产生停止条件。每一个8位数据传输完成之后,接收方就需要回复一个ACK/NACK。
停止条件:当所有数据都发送完成时,当SCL处于高电平时,SDA由低电平变为高电平。除了开始条件和停止条件,在正常的数据传输过程中,当SCL处于高电平时,SDA上的值不能变化,否则会意外产生停止条件。
重复开始条件:有时Master设备需要在一次通信中进行多次消息交换(例如切换读写操作等),并且不希望其他Master设备干扰,这时可以使用重复开始条件。再一次通信中,Master设备可以产生多次开始条件来完成多次信息交换,最后在产生一个停止条件结束整个通信过程。

请注意,这里引用的比特率为主节点和从节点之间没有时钟延长或其他硬件开销的传输比特率。协议开销包括一个字节从地址(或许还有从设备内部寄存器地址)以及每个字节的ACK / NACK比特。因此这意味着,用户数据的实际传输速率要低于峰值比特率。例如,如果与从设备以低效的每次仅一个字节数据进行传输,数据率将比峰值比特率少于一半(因为其余的时钟需要发送一个字节地址和ACK比特)。

一开始,主节点处于主节点发送模式,发送起始比特(START),跟着发送希望与之通信的从节点的7比特地址,最后再发送1位读写比特,该比特表示主节点想要与从节点进行读(1)还是写(0)操作。
如果从节点在总线上,它将以ACK字符应答(低态触发)该地址。主节点收到应答后,根据它发送的读写比特,处于发送模式或者接收模式,从节点则处于对应的相反模式(接收或发送)。
地址和数据首先发送最高有效位。 起始比特在SCL为高时,由SDA上准位从高变低表示;停止比特在SCL为高时,由SDA上准位从低变高表示。其他SDA上的准位变化在SCL为低时发生。
如果主节点想要向从节点写数据,它将发送一个字节,然后从节点以ACK比特应答,如此重复。此时,主节点处于主节点发送模式,从节点处于从节点接收模式。
如果主节点想要读取从节点数据,它将不断接收从节点发送的一个个字节,在收到每个字节后发送ACK进行应答,除了接收到的最后一个字节。此时,主节点处于主节点接收模式,从节点处于从节点发送模式。
此后,主节点不是发送停止比特终止传输,就是发送另一个起始比特以发起另一次传输(即“组合消息”)

3、I2C的两种方式——硬件I2C和软件I2C

①硬件I2C
直接利用 STM32 芯片中的硬件 I²C 外设。
硬件I2C的使用
只要配置好对应的寄存器,外设就会产生标准串口协议的时序。在初始化好 I2C 外设后,只需要把某寄存器位置 1,此时外设就会控制对应的 SCL 及 SDA 线自动产生 I2C 起始信号,不需要内核直接控制引脚的电平。

②软件I2C
直接使用 CPU 内核按照 I2C 协议的要求控制 GPIO 输出高低电平,从而模拟I2C。
软件I2C的使用
需要在控制产生 I2C 的起始信号时,控制作为 SCL 线的 GPIO 引脚输出高电平,然后控制作为 SDA 线的 GPIO 引脚在此期间完成由高电平至低电平的切换,最后再控制SCL 线切换为低电平,这样就输出了一个标准的 I2C 起始信号。

③两者的差别
硬件 I2C 直接使用外设来控制引脚,可以减轻 CPU 的负担。不过使用硬件I2C 时必须使用某些固定的引脚作为 SCL 和 SDA,软件模拟 I2C 则可以使用任意 GPIO 引脚,相对比较灵活。对于硬件I2C用法比较复杂,软件I2C的流程更清楚一些。如果要详细了解I2C的协议,使用软件I2C可能更好的理解这个过程。在使用I2C过程,硬件I2C可能通信更加快,更加稳定。

二、AHT20的信息采集

1、AHT20芯片信息

下载官方产品介绍文档: http://www.aosong.com/class-36.html

Alt
Alt

2、STM32F103连接AHT20

Alt

3、传感器读取流程代码实现步骤

main.c

#include "delay.h"
#include "usart.h"
#include "bsp_i2c.h"


int main(void)
{	
	delay_init();     //?óê±oˉêy3?ê??ˉ	  
	uart_init(115200);	 //′??ú3?ê??ˉ?a115200
	IIC_Init();
		while(1)
	{
		printf("温度湿度显示");
		read_AHT20_once();
		delay_ms(2000);
  }
}

(1)、上电后等待40ms,读取温湿度值之前, 首
先要看状态字的校准使能位Bit[3]是否为 1(通
过发送0x71可以获取一个字节的状态字),如果
不为1,要发送0xBE命令(初始化),此命令参数
有两个字节, 第一个字节为0x08,第二个字节
为0x00,然后等待10ms。

void  read_AHT20_once(void)
{
	delay_ms(10);

	reset_AHT20();//重置AHT20芯片
	delay_ms(10);

	init_AHT20();//初始化AHT20芯片
	delay_ms(10);

	startMeasure_AHT20(); //开始测试AHT20芯片
	delay_ms(80);

	read_AHT20();// 读取数据
	delay_ms(5);
}
void  reset_AHT20(void)//重置AHT20芯片
{

	I2C_Start();

	I2C_WriteByte(0x70);
	ack_status = Receive_ACK();
	if(ack_status) printf("1");
	else printf("1-n-");
	I2C_WriteByte(0xBA);
	ack_status = Receive_ACK();
		if(ack_status) printf("2");
	else printf("2-n-");
	I2C_Stop();

	/*
	AHT20_OutData[0] = 0;
	AHT20_OutData[1] = 0;
	AHT20_OutData[2] = 0;
	AHT20_OutData[3] = 0;
	*/
}
void  init_AHT20(void)//初始化AHT20
{
	I2C_Start();

	I2C_WriteByte(0x70);
	ack_status = Receive_ACK();
	if(ack_status) printf("3");
	else printf("3-n-");	
	I2C_WriteByte(0xE1);
	ack_status = Receive_ACK();
	if(ack_status) printf("4");
	else printf("4-n-");
	I2C_WriteByte(0x08);
	ack_status = Receive_ACK();
	if(ack_status) printf("5");
	else printf("5-n-");
	I2C_WriteByte(0x00);
	ack_status = Receive_ACK();
	if(ack_status) printf("6");
	else printf("6-n-");
	I2C_Stop();
}
void  startMeasure_AHT20(void)
{
	//------------
	I2C_Start();

	I2C_WriteByte(0x70);
	ack_status = Receive_ACK();
	if(ack_status) printf("7");
	else printf("7-n-");
	I2C_WriteByte(0xAC);
	ack_status = Receive_ACK();
	if(ack_status) printf("8");
	else printf("8-n-");
	I2C_WriteByte(0x33);
	ack_status = Receive_ACK();
	if(ack_status) printf("9");
	else printf("9-n-");
	I2C_WriteByte(0x00);
	ack_status = Receive_ACK();
	if(ack_status) printf("10");
	else printf("10-n-");
	I2C_Stop();
}

(2)、直接发送 0xAC命令(触发测量),此命令参数
有两个字节,第一个字节为 0x33,第二个字节
为0x00。

(3)、当接收完六个字节后,紧接着下一个字节是
CRC校验数据,用户可以根据需要读出,如果接
收端需要CRC校验,则在接收完第六个字节后发
ACK应答,否则发NACK结束,CRC初始值为0XFF。

I2C_Start();   //I2c 启动

	I2C_WriteByte(0x71);   //I2C写数据
	ack_status = Receive_ACK();//收到的应答信息
	readByte[0]= I2C_ReadByte();//I2C读取数据
	Send_ACK();//发送应答信息

	readByte[1]= I2C_ReadByte();
	Send_ACK();

	readByte[2]= I2C_ReadByte();
	Send_ACK();

	readByte[3]= I2C_ReadByte();
	Send_ACK();

	readByte[4]= I2C_ReadByte();
	Send_ACK();

	readByte[5]= I2C_ReadByte();
	SendNot_Ack();
	//Send_ACK();

	I2C_Stop();  //I2C停止

(4)、计算温湿度值。

if( (readByte[0] & 0x68) == 0x08 )
	{
		H1 = readByte[1];
		H1 = (H1<<8) | readByte[2];
		H1 = (H1<<8) | readByte[3];
		H1 = H1>>4;

		H1 = (H1*1000)/1024/1024;

		T1 = readByte[3];
		T1 = T1 & 0x0000000F;
		T1 = (T1<<8) | readByte[4];
		T1 = (T1<<8) | readByte[5];

		T1 = (T1*2000)/1024/1024 - 500;

		AHT20_OutData[0] = (H1>>8) & 0x000000FF;
		AHT20_OutData[1] = H1 & 0x000000FF;

		AHT20_OutData[2] = (T1>>8) & 0x000000FF;
		AHT20_OutData[3] = T1 & 0x000000FF;
	}
	else
	{
		AHT20_OutData[0] = 0xFF;
		AHT20_OutData[1] = 0xFF;

		AHT20_OutData[2] = 0xFF;
		AHT20_OutData[3] = 0xFF;
		printf("读取失败");

	}
	printf("\r\n");
	//根据温度和湿度的计算公式,计算输出结果,通过串口发生到上位机
	printf("温度:%d%d.%d",T1/100,(T1/10)%10,T1%10);
	printf("湿度:%d%d.%d",H1/100,(H1/10)%10,H1%10);
	printf("\r\n");

三、实验效果及总结

对着传感器吹一口气,可以看到湿度在增大,传感器运行正常。

Alt
总结:此次实验,使用了AHT20温湿度传感器,在此之前对其一无所知,如何使用是一个大问题。但在商家提供实验例程的情况下就容易了很多,只需去熟悉看懂主要的函数,就能自己在此基础上改进完成实验。面对新事物时,找一个现实的例程去对照,这是个很好的学习方法。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值