STM32 HAL库 I2C 工作出错 返回I2C_BUSY

使用stm32cubemx生成硬件I2C的代码
不过自动生成的代码,调用HAL_I2C_XXX的API工作不正常,返回错误代码为I2C_BUSY
使用STM32的I2C接口使用时需要注意很多细节,不过HAL库中官方已经为用户根据这些细节做了处理,可以直接使用。不过这个I2C代码并不稳定,有些板子可以用,另一些则出错.

出错现象为调用HAL_I2C的API时,返回I2C_BUSY

查看寄存器 BUSY位被置位

STM32 HAL库 I2C 工作出错 返回I2C_BUSY

在此基础上,替换过上拉电阻亦没有作用
在网上搜到一个回答解决了这个问题
STM32 HAL库 I2C 工作出错 返回I2C_BUSY
附下原文链接
https://electronics.stackexchange.com/questions/272427/stm32-busy-flag-is-set-after-i2c-initialization?answertab=votes#tab-top

解决方案 将自动生成的代码中,HAL_RCC_I2C1_CLK_ENABLE()放到GPIO初始化之前就不会出现这个问题,I2C工作正常

具体依据没有找到,不过在HAL的说明文档中有要求这样的初始化顺序,同时其他的模块如SPI或UART,也都是先初始化模块(SPI,UART),再初始化GPIO
STM32 HAL库 I2C 工作出错 返回I2C_BUSY

void HAL_I2C_MspInit(I2C_HandleTypeDef* i2cHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(i2cHandle->Instance==I2C1)
  {
  /* USER CODE BEGIN I2C1_MspInit 0 */
	__HAL_RCC_I2C1_CLK_ENABLE(); //添加时钟使能命令
  /* USER CODE END I2C1_MspInit 0 */
  
    __HAL_RCC_GPIOB_CLK_ENABLE();
    /**I2C1 GPIO Configuration    
    PB6     ------> I2C1_SCL
    PB7     ------> I2C1_SDA 
    */
    GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

    /* I2C1 clock enable */
    __HAL_RCC_I2C1_CLK_ENABLE();

    /* I2C1 interrupt Init */
    HAL_NVIC_SetPriority(I2C1_EV_IRQn, 0, 0);
    HAL_NVIC_EnableIRQ(I2C1_EV_IRQn);
  /* USER CODE BEGIN I2C1_MspInit 1 */

  /* USER CODE END I2C1_MspInit 1 */
  }
}

其他解决方式

即初始化时,重新置位下I2C,再重新初始化,虽然说这种方法也可行,只是总感觉不太合适。再没有找到合理的解决方式之前也用这个方式试过。
重置方式有两种

一、I2C寄存器复位

STM32 HAL库 I2C 工作出错 返回I2C_BUSY
换成代码可以是这样,
hi2c1.Instance->CR1 |= 0x8000;
hi2c1.Instance->CR1 &= ~0x8000;

不过这个操作之后I2C必须要重新初始化,可用于除错的处理吧

I2C1的复位操作也可以通过操作另外一个标志位达到复位的效果

二、APB寄存器复位

STM32 HAL库 I2C 工作出错 返回I2C_BUSY
置位I2C1RST来达到复位I2C的效果
HAL中有现成的库函数定义
__HAL_RCC_I2C1_FORCE_RESET()
__HAL_RCC_I2C1_RELEASE_RESET()

上一篇:Zabbix unreachable poller processes more than 75% busy解决方法


下一篇:RocketMQ 一行代码造成大量消息丢失