0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

使用IIC去驱动MPU6050时为什么总读取失败

strongerHuang 来源:知晓编程 作者:Firefly 2021-11-06 10:10 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

01概述

在之前的文章中《I2C基础原理及协议》中详细讲解了IIC协议,并且使用是NXP的官方手册,demo示例使用IIC读取RTC芯片,运行正常,没有任何问题。并且更新了《IIC踩过的坑》,讲述了在使用IIC读取RTC芯片时遇到的问题,并成功解决。

我以为我已经完全学会了IIC,但现实却打了脸,我在使用《STM32IIC详解》文中的IIC驱动,去驱动MPU6050时,总是读取失败。这个驱动明明是验证过的,为什么会有问题。让我一度很是郁闷。

02问题

不卖关子,直接说问题,是我之前的IIC驱动有问题。

问题1:

错误将CLK信号GPIO设置为推挽输出。应该设置为开漏输出。

042908fa-3dfe-11ec-82a9-dac502259ad0.png

问题2:

读取函数有bug。1处应该先左移再读取SDA的数据,然后删除2处的数据。

0464f400-3dfe-11ec-82a9-dac502259ad0.png

问题2:这个就是纯粹的bug了,大家应该看出来了。在RTC的驱动没有触发bug的原因是:在RTC的IIC接收数据中,实际应用中最高位为0,触发不了这个bug。而在MPU6050的IIC接收数据中就触发了这个bug。我也在感慨,有时候不是程序没有bug,而是可能没有触发。

问题1:这个问题,其实很简单,IIC协议中也提到过,很多大神也知道需要将MCU的IIC引脚设置为开漏输出。这一方面我也了解,但是没有在意,因为一直读取RTC一直“没有bug”。接下来我将细细和大家分享一下IIC为什么需要开漏输出,开漏输出和推挽输出有什么区别。精通的大佬可以出门左转了,想了解一下的同学欢迎继续往下看。

03开漏输出

STM32F207的GPIO框图如下

04c588b0-3dfe-11ec-82a9-dac502259ad0.png

普通输入模式下,上拉和下拉电阻(微弱)的存在。主要是由于P-MOS和N-MOS的存在分为下列两种模式

开漏模式:输出寄存器是 0 时,激活 N-MOS,而输出寄存器是 1 时,端口保持高阻态(P-MOS 不会被使能)

推挽输出:输出寄存器是 0 时,激活 N-MOS,而输出寄存器是 1 时,激活 P-MOS。

上面是我的在文章《STM32 GPIO详解》中的说明,GPIO的其他模式请看文章《STM32 GPIO详解》。上文说到开漏模式输出1时,端口保持高阻态,这个时候如果端口外上拉电阻,就可以输出电平1。

开漏输出的作用:

1:防止短路,在一些应用中,两个GPIO链接在一起(中间没有串电阻),或者在总线应用中,需要将MCU的多个GPIO连接在一起。如果都设置成推挽输出,当一个GPIO输出1,另一个输出0,那么就短路了,直接凉凉。如下图

如果换成开漏输出,GPIO的高电平是靠上拉电阻的,也就是VCC和GND之间会有个电阻,不会出现短路的问题。这样的电路就安全一些,所以部分总线采用开路输出。

2:线与:开漏输出还能实现线与,减少一个与门,简化电路。这个问题下文讲到。

04开漏输出在IIC的应用

IIC为什么需要开漏输出,除了上文说的到的防止短路,还有一个重要的因素就是线与。

首先我们先说一下线与功能:

线与逻辑,即两个输出端(包括两个以上)直接互连就可以实现“AND”的逻辑功能。在总线传输等实际应用中需要多个门的输出端并联连接使用,而一般TTL门输出端并不能直接并接使用,否则这些门的输出管之间由于低阻抗形成很大的短路电流(灌电流),而烧坏器件。

在硬件上,可用集电极开路门(OC门)或三态门(TS门)来实现。用OC门实现线与,应同时在输出端口加一个上拉电阻。

上面是数电知识,我的个人简单理解是:就是a,b两条线,两端接一块做输出,另两端做输入。如果输入都是高电平,那输出就是高电平,否则输出就是低电平。

那么线与在IIC中的应用是什么呢?

答案是:多主设备抢占总线的仲裁。

在之前IIC读取RTC或IIC读取MPU6050的情况,都是一个主机,一个从机。但IIC设计中可以支持多主机模式,那么就面临一个问题,当多个主机同时启动总线时,如果仲裁的问题。线与逻辑就起到了作用。

假设主设备A需要启动IIC,它需要在SCL高电平时,将SDA由高电平转换为低电平作为启动信号。主设备A在把SDA拉高后,它需要再检查一下SDA的电平。

SDA是高电平,说明主设备A可以占用总线,然后主设备A将SDA拉低,开始通信

SDA是低电平,说明有人已经捷足先登了,主设备A不能占用总线,结束通信。

如果主设备A拉高SDA时,已经有其他主设备将SDA拉低了。由于1 & 0 = 0 那么主设备A在检查SDA电平时,会发现不是高电平,而是低电平,说明其他主设备抢占总线的时间比它早,设备A只能放弃占用总线。如果是高电平, 则可以占用。

这就是IIC通信开漏输出的原因。上拉电阻的原因就是由于开漏输出的特性,需要上拉电阻在输出1时,提高驱动力。

05最后补充

最后说一下为什么之前使用推挽输出的IIC读取RTC没有问题,这个因为上拉电阻的阻值不同,RTC的上拉电阻即使推挽输出也可以正常拉高拉低电平。这个根据上文讲述的,可以查MCU的datasheet,确认IO的PMOS和NMOS的阻抗,计算一下电压。

还有一个简单粗暴的办法,直接使用示波器看波形也可以发现问题。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 芯片
    +关注

    关注

    463

    文章

    54686

    浏览量

    471245
  • 通信
    +关注

    关注

    18

    文章

    6490

    浏览量

    140379
  • I2C
    I2C
    +关注

    关注

    28

    文章

    1567

    浏览量

    131994

原文标题:I2C通信中的坑

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    MC33665A CRC 读取失败的原因?如何处理?

    失败。我尝试进行调试时,问题出在Phy_665a_ReadSysCrcReqSendSpiVariant函数上,该函数试图读取MC33665_SYS_CFG_CRC_OFFSET寄存器。读取操作虽然成功,但当我在完成SYS_CF
    发表于 04-27 07:19

    用EB配置MEM和FLS时,读取和发送的数据验证失败,怎么解决?

    当我用EB配置MEM和FLS时,运行EB提供的官方demo写入8192个数据点,运行代码后,读取和发送的数据验证失败。通过通过内存观察闪存内容,我可以看到rxBuffer中存在错误,写入闪存的数据也不正确。
    发表于 04-24 08:12

    是否有用于读取 FCCU 数据/锁步失败的寄存器?

    我们正在研究 AEB 应用的S32K344和S32K358。 在该安全功能中,对于锁步故障(核心 0 和核心 1 结果不匹配),我们需要读取 FCCU 数据。为此,我们需要 SAF 包。 还有其他方法可以知道吗,锁步失败。 是否有用于
    发表于 04-03 06:43

    如何使用 VisionFive IIC 读取 SHTC3 数据?

    使用 VisionFive IIC 读取 SHTC3 数据
    发表于 03-27 07:09

    开源项目低功耗蓝牙智能骰子总体介绍

    一款将传统桌游道具与低功耗蓝牙结合的创意!这款智能骰子基于 nRF52840 核心的低功耗模组打造,集成 MPU6050 六轴惯性测量单元,精准检测掷骰动作与朝向,掷出后 LED 自动点亮显示点数,蓝牙 5.0 低能耗可与安卓设备配对联动。
    的头像 发表于 02-27 09:08 659次阅读
    开源项目低功耗蓝牙智能骰子总体介绍

    CW32L012解算MPU6050姿态数据

    我们仅仅获取了MPU6050的三轴加速度和角速度,要想得到姿态角,需要利用读取的数据进行姿态解算。
    的头像 发表于 01-05 16:42 2040次阅读
    CW32L012解算<b class='flag-5'>MPU6050</b>姿态数据

    CW32L012读取MPU6050姿态数据

    随着物联网、可穿戴设备、工业倾角检测等场景的普及,低成本、低功耗的姿态检测方案成为嵌入式领域的研究热点。MPU6050 作为集成三轴加速度计和三轴陀螺仪的六轴传感器,凭借低成本、小体积的优势被
    的头像 发表于 01-05 16:36 1347次阅读
    CW32L012<b class='flag-5'>读取</b><b class='flag-5'>MPU6050</b>姿态数据

    RA MCU众测宝典 | IIC之【RA2E1】IIC通信的OLED显示

    ,到OLED驱动函数编写、图像文字取模与显示,一起感受仅用SDA和SCL两条线,就能实现设备间数据交互与可视化的便捷魅力。开启宝典简介IIC通信协议IIC(Inte
    的头像 发表于 01-01 10:04 3941次阅读
    RA MCU众测宝典 | <b class='flag-5'>IIC</b>之【RA2E1】<b class='flag-5'>IIC</b>通信的OLED显示

    用dma加iic读取mpu6050数据遇到的疑问求解

    我在用dma加iic读取mpu6050数据时,试着用dma传输4个数,但是用freemaster查看数据以后,发现只有数组里面的第一个数有值,其他的都为0,而且发现没进dma传输完成中断,把传输数量改成1以后,就能进入dma传输
    发表于 12-09 07:41

    电机驱动emc整改:不过关?可能是接地方式错了

    电机驱动emc整改:不过关?可能是接地方式错了|深圳南柯电子
    的头像 发表于 10-14 15:44 1098次阅读

    TT电机,Arduino Uno,L298N 5AD ,MPU6050最简自平衡机器人资料

    ;MPU6050_6Axis_MotionApps20.h\" //https://github.com/jrowberg/i2cdevlib/tree/master/Arduino/MPU6050 MPU6050
    发表于 10-06 13:12

    请问IIC的设备驱动兼容SMbus协议吗?

    请问IIC的设备驱动兼容SMbus协议么?
    发表于 09-29 09:53

    【RA4M2-SENSOR】串口传输MPU6050陀螺仪数据测评

    ); printf(\"hollow Acce\"); 接着配置mpu6050读取数据代码,将如下文件复制到工程中,并且添加到项目中: 配置I2C引脚: 代码主要在bsp_iic.h中进行修改
    发表于 09-14 11:41

    小安派BW21-CBV-Kit入门教程之DMP6读取MPU6050数据

    本示例将演示使用 DMP6 从 MPU6050 获取数据。MPU6050 配备了数字运动处理器 (DMP),用于处理运动算法的计算,如转换为三轴偏航/俯仰/滚动平面、转换为四元数或转换为欧拉角。
    的头像 发表于 06-19 14:20 1265次阅读
    小安派BW21-CBV-Kit入门教程之DMP6<b class='flag-5'>读取</b><b class='flag-5'>MPU6050</b>数据

    通过control center上位机读取USB芯片CY7C68013A数据失败的原因?怎么解决?

    大家好,通过control center上位机读取CY7C68013A芯片的数据,连续读取512个字节能够成功,读取2个字节失败,错误码是997,一开始
    发表于 05-30 06:43