当前位置: 首页 > news >正文

GD32 MCU碰到IIC总线卡死怎么办?

大家在使用MCU IIC通信时,若碰到设备复位或者总线干扰等情况,可能会导致IIC总线卡死,表现上总线上SDA或者SCL其中一根线为低电平,IIC总线一直处于busy状态。此时若代码上一直等待总线空闲,则可能导致软件死机,为解决该问题,本视频提供了软件配置释放IIC总线的方法。

首先为大家介绍发生IIC总线卡死的两种情况:(1)主机在发送 START 信号后, 控制 SCL 产生 8 个时钟脉冲,然后拉低 SCL 信号为低电平,在这个时候,从设备输出应答信号,将 SDA 信号拉为低电平。如果这个时候主机异常复位, SCL 就会被释放为高电平。此时,如果从机没有复位,就会继续 I2C 的应答,将 SDA一直拉为低电平,直到 SCL 变为低电平,才会结束应答信号。 而由于 I2C 主机复位后检测总线的状态,如果 SDA 信号为低电平,则 I2C 总线被占用,会一直等待 SCL 和 SDA信号变为高电平,因此,在 I2C 主机等待从机释放 SDA 信号时, I2C 从机又在等待主机将 SCL 信号拉低以释放应答信号,两者相互等待, I2C 总线进入死锁状态 ;(2)主机在发送 START 信号后, 控制 SCL 产生 8 个时钟脉冲,然后拉低 SCL 信号为低电平,在这个时候,从设备输出应答信号,将 SDA 信号拉为低电平。如果这个时候主机异常复位, SCL 就会被释放为高电平。此时,如果从机没有复位,就会继续 I2C 的应答,将 SDA一直拉为低电平,直到 SCL 变为低电平,才会结束应答信号。 而由于 I2C 主机复位后检测总线的状态,如果 SDA 信号为低电平,则 I2C 总线被占用,会一直等待 SCL 和 SDA信号变为高电平。因此,在 I2C 主机等待从机释放 SDA 信号时, I2C 从机又在等待主机将 SCL 信号拉低以释放应答信号,两者相互等待, I2C 总线进入死锁状态 。

以下为两种复位IIC总线卡死的软件方法,大家可以尝试使用:

(1)将SDA和SCL配置为推挽输出,强制输出stop信号

在 I2C 主机复位后,主机检测 I2C 总线一直为 BUSY 状态,且超过设定的时间,则总线被锁死。可通过将 I2C 的 SCL 和 SDA 引脚初始化成普通 GPIO 功能,配置成推挽输出。 先拉高SCL 信号,在拉高 SDA 信号,模拟产生一个 STOP 信号,然后再配置为 I2C 的引脚复用功能。配置代码如下所示。

C
/*!
\brief reset i2c bus
\param[in] none
\param[out] none
\retval none
*/
void i2c_bus_reset()
{
GPIO_BC(GPIOB) |= GPIO_PIN_6 | GPIO_PIN_7;
gpio_init(GPIOB, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ,
GPIO_PIN_6|GPIO_PIN_7);
__nop();
__nop();
__nop();
__nop();
__nop();
GPIO_BOP(GPIOB) |= GPIO_PIN_6;
__nop();
__nop();
__nop();
__nop();
__nop();
GPIO_BOP(GPIOB) |= GPIO_PIN_7;
gpio_init(GPIOB, GPIO_MODE_AF_OD, GPIO_OSPEED_50MHZ, GPIO_PIN_6 |
GPIO_PIN_7);
}
/*!
\brief check the I2C is or not busy
\param[in] none
\param[out] none
\retval none
*/
void check_bus_status(void)
{
while(i2c_flag_get(I2C0,I2C_FLAG_I2CBSY))
{
if(--time_out == 0){
i2c_bus_reset();
}
}
}

(2)将SCL配置为推挽输出,强制输出9个clk

在 I2C 主机中增加 I2C 总线恢复程序。每次 I2C 主设备复位后,如果检测到 SDA 数据线被拉低,则控制 I2C 中的 SCL 时钟线产生 9 个时钟脉冲(针对 8 位数据的情况),这样 I2C 从设备就可以完成被挂起的操作,从死锁状态中恢复过来。  

I2C 主机通过将 SCL 引脚初始化为普通 GPIO 功能,配置成推挽输出。保证连续发送 9 个时钟脉冲,为保证后续 I2C 正常通信,先将 I2C 模块复位,再置位,最后再配置为 I2C 的引脚复用功能。配置代码如下所示。

C
/*!
\brief reset i2c bus
\param[in] none
\param[out] none
\retval none
*/
void i2c_bus_reset()
{
uint8_t I = 0;
gpio_init(GPIOB, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_6);
/* SCL output clock signal */
for(I = 0; I < 10; i++){
gpio_bit_reset(GPIOB, GPIO_PIN_6);
delay_1us(2);
gpio_bit_set(GPIOB, GPIO_PIN_6);
delay_1us(2);
}
/* reset I2C */
i2c_software_reset_config(I2C0, I2C_SRESET_RESET);
i2c_software_reset_config(I2C0, I2C_SRESET_SET);
gpio_init(GPIOB, GPIO_MODE_AF_OD, GPIO_OSPEED_50MHZ, GPIO_PIN_6 |
GPIO_PIN_7);
}
/*!
\brief check the I2C is or not busy
\param[in] none
\param[out] none
\retval none
*/
void check_bus_status(void)
{
while(i2c_flag_get(I2C0,I2C_FLAG_I2CBSY))
{
if(--time_out == 0){
i2c_bus_reset();
}
}
}

如有其他问题或建议,欢迎评论区讨论。

更多GD32 MCU相关咨询:https://www.gd32bbs.com/

http://www.lryc.cn/news/421024.html

相关文章:

  • 算法——动态规划:0/1 背包问题
  • 又是奇瑞,“统一下班时间”过去不久,最近又整新活了...
  • ubuntu24.04lts cmake编译 opencv4.5.4 contrib的一些问题
  • 大数据面试SQL(三):每分钟在线直播人数
  • python中执行mysql操作并将python脚本共享
  • HTTP、HTTPS、SOCKS5三种协议特点
  • 在ubuntu、centos、openEuler安装Docker
  • 公共命名空间的例子3
  • 【云存储】SDS软件定义存储,数据存储的类型与技术方案(块/文件/对象,Ceph、RBD等)
  • 第31课 Scratch入门篇:小画家(舞台上画画)
  • QT UI界面之ListView
  • freeRTOS互斥量(mutex)
  • 基于GeoTools使用JavaFx进行矢量数据可视化实战
  • zabbix的setup无法进入第二步
  • 代码随想录算法训练营第四十六天 | 115. 不同的子序列、583. 两个字符串的删除操作、72. 编辑距离、编辑距离总结篇
  • 宝塔安装nginx失败报错“检测到系统组件wget不存在,无法继续安装”
  • C++之运算符重载系列深入学习:从入门到精通!
  • 国赛分析。。。。
  • 无缝融入,即刻智能[一]:Dify-LLM大模型平台,零编码集成嵌入第三方系统,42K+星标见证专属智能方案
  • PLSQL导入导出ORACLE数据提示失败问题修改PLSQL配置
  • 从Shift+F6到雪花算法:IDEA开发中的那些坑与解法
  • Linux知识点总结
  • Gradio 快速开发网页应用
  • spring使用validation参数及全局异常检测
  • 学习笔记 韩顺平 零基础30天学会Java(2024.8.8)
  • 45.跳跃游戏
  • Golang | Leetcode Golang题解之第328题奇偶链表
  • 【ARM】CMSIS 软件标准接口
  • Qt 小功能:加载等待动画——转圈圈
  • 【Linux进程篇】进程终章:POSIX信号量线程池线程安全的单例模式自旋锁读者写者问题