凡亿专栏 | 【STM32 cubemx】0011 HAL库开发:I2C总线访问加速度传感器ADXL345
【STM32 cubemx】0011 HAL库开发:I2C总线访问加速度传感器ADXL345

本节我们介绍I2C总线,并使用stm32的I2C总线来访问加速度传感器ADXL345。


1)I2C基础知识

I2C总线通信比较适合设备内部各芯片间的通信,它只需要两根信号线。

I2C可以挂载多个主机和从机,通信总是由主机发起。每个从机都有唯一的地址,主机通过地址决定访问哪个从机。

I2C总线的两根线,SCL为时钟线,SDA为数据线;所有的器件对这两根线的输出操作只能拉低,当释放总线时,由总线上的上拉电阻将电平拉高。所以硬件连接上,上拉电阻是必须的,否则不能通信。


I2C空闲时,所有器件释放总线,SCL和SDA都被上拉电阻拉到高电平;

I2C的起始条件:SCL为高电平时,SDA由高电平向低电平切换;表示开始传送数据;一般是需要通信的主机发起,起始条件的图示如下:

128febeabab0e8cd32e95e75bc6340.jpg


I2C的停止条件:SCL为高电平时,SDA由低电平向高电平跳变;表示结束传送数据;一般也是主机最后结束通信,停止条件的图示如下:

32eb141e0fe1a0f399d4f480b30711.jpg


传输数据时,在SCL时钟线为高电平时,SDA数据线上的电平不允许被修改;SCL时钟线为低电平时,SDA数据线上的电平可变为高/低,如下图所示:

d6a36f0be749f2c94e8627c94fa2ed.jpg


I2C的ACK和NACK,都是回应,ACK是将SDA线拉低,NACK是将SDA总线释放(拉高);ACK和NACK都是回应,可以是主机回应从机、也可以是从机回应主机。具体来说(以ADXL345芯片为例):

主机发起通信后,如果要向从机写入数据,则每传输一个字节都需要等待从机回应ACK,如下图所示;具体到实际的操作,就是主机每发完一个字节的数据,会释放总线,等待从机回应ACK(即等待从机把SDA线拉低)。最后通信完成后,主机发送停止条件:

95d4940c3d17e5fa28f19da00b2342.jpg


主机发起通信后,如果要从从机处读取数据,则发送地址时,需要等待从机回应ACK;在从机向主机回复数据时,主机要回应ACK,当主机读完最后一个字节不再读取时,就回应NACK;写入数据的过程如下图:

fa58506be4768a34f937867fdd9773.jpg


上面的例子是以加速度传感器ADXL345芯片为例,其他芯片在使用I2C通信时,可能有细微的不同,但基本上都是这几种状态组合起来的。


2)stm32的I2C总线使用

Stm32带了硬件I2C,下面我们就使用stm32的硬件i2c读取加速度传感器ADXL345的数值。

我们仍然以串口的工程为基础,在它上面添加设置,如下图,选择I2C1接口,其他都默认:

c0d508fd3e89e524f745c599ca2d12.jpg

选择之后可以发现引脚PB6和PB7被占用为I2C的引脚,其中PB6是SCL、PB7是SDA。


在硬件连接上,我们也需要将ADXL345芯片的SCL和SDA连接到PB6和PB7,并且SDA和SCL都要用电阻上拉到电源,ADXL345芯片的相关原理图如下:

60b888641694db209f49d38c680ac0.jpg


在cubemx中生成工程代码,在keil中打开。Stm32的hal库已经将i2c的初始化、i2c的读写操作封装成了函数,我们直接调用即可。


这里先需要修改一个bug,如下图:

__HAL_RCC_I2C1_CLK_ENABLE(); 这一句使能i2c时钟的语句,cubemx生成的代码在GPIO初始化之后,这样不能设置成功,需要把它提前到GPIO初始化之前:

c8570a0ff4708c03a4932161b882cc.jpg


在Main函数中添加初始化ADXL345的代码:

c2c5ce221c9e6e3144ad94a548e25e.jpg

简单说明一下,前面写入的4个值是设置芯片的工作模式,最后写入的三个值是x、y、z三个方向的校准参数。由于ADXL345传感器有初始误差(网上有说法是X、Y方向误差不大,Z轴可能误差达到几个g;我的这个芯片X、Y、轴基本正常,Z轴有大约0.6g左右的误差),可以写入一组测好的参数,让它输出时自动减掉这个值,达到去除初始误差的目的,我这里在Z轴写入了f0,校准后基本正常。


主循环内添加循环读取ADXL345数值的代码:

ce8e3a792efe1dab75fa32b63c09e5.jpg


编译下载运行,可以看到数据输出:

c6f343ad5187d5e2e436680cde4e8b.jpg


变换传感器方向,可以看到测出的重力加速度的数值变化。


3)stm32的硬件I2C问题备忘

在使用stm32的硬件i2c时,查了很多资料,挺多网友说stm32的硬件i2c有问题,我用的不多,但是汇总了一些网上的讨论,以作为备忘,万一哪天必须用硬件i2c时也好排查:

a) cubemx生成的代码,初始化i2c的时钟要提到GPIO初始化之前;

b) I2c的速率不能过高,有说法是50k以下基本无问题,100k以上运行时间久了会出问题;

c) 某些带FSMC模块的stm32型号,由于FSMC和I2C1模块共用了PB7引脚,会使得I2C1不能正常启动;即使没有使用FSMC只是打开了FSMC的时钟也会影响,解决办法是改用I2C2或者重定义I2C的时钟线和数据线到PB8、9引脚;或者关掉FSMC的时钟;

d)硬件i2c不能被中断打断,否则会出问题;如果使用中断,建议i2c的中断设置为最高优先级。

鉴于有些问题确实不好验证,本人后面开发都使用软件i2c了,效率低一些但可靠性高,并且移植到其他芯片上也比硬件i2c方便很多。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表凡亿课堂立场。文章及其配图仅供工程师学习之用,如有内容图片侵权或者其他问题,请联系本站作侵删。
相关阅读
进入分区查看更多精彩内容>
精彩评论

暂无评论