总结一下首次使用HAL库STM32f030硬件IIC从机中断收发

总结一下首次使用HAL库STM32f030硬件IIC从机中断收发,第1张

总结一下首次使用HAL库STM32f030硬件IIC从机中断收发

首先IIC的概念就略过了。这个网上写的很详细。
从CUBEMX配置完代码开始吧。

手上的项目是一主机,七个从机,从机使用中段收发。
关于地址是一个大坑,后续的读写中断也是个坑。

从机地址:
硬件为I2C_ADDRESSINGMODE_7BIT时的从机地址时,从机的地址需要左移一位,最低为表示读或者写。比如从机地址I2COwnAddr = 0x0A,左移一位配置为:
hi2c1.Init.OwnAddress1 = I2COwnAddr<<1;//从机的地址设置需要左移一位
这样从机的配置初始化函数

void MX_I2C1_Init(void)
{

  

  

  

  
  hi2c1.Instance = I2C1;
  hi2c1.Init.Timing = 0x2000090E;
  hi2c1.Init.OwnAddress1 = I2COwnAddr<<1;//从机的地址设置需要左移一位
  hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
  hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
  hi2c1.Init.OwnAddress2 = 0;
  hi2c1.Init.OwnAddress2Masks = I2C_OA2_NOMASK;
  hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
  hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
  if (HAL_I2C_Init(&hi2c1) != HAL_OK)
  {
    Error_Handler();
  }
  
  if (HAL_I2CEx_ConfigAnalogFilter(&hi2c1, I2C_ANALOGFILTER_ENABLE) != HAL_OK)
  {
    Error_Handler();
  }
  
  if (HAL_I2CEx_ConfigDigitalFilter(&hi2c1, 0) != HAL_OK)
  {
    Error_Handler();
  }
  
	
	#ifndef MASTER
//	HAL_I2C_Slave_Receive_IT(&hi2c1, I2CBuf_RxData, sizeof(I2CBuf_RxData));    // 启动中断接收
	HAL_I2C_Slave_Transmit_IT(&hi2c1, I2CBuf_TxData, sizeof(I2CBuf_TxData));    // 启动中断发送
//	HAL_I2C_EnableListen_IT(&hi2c1);

	#endif
  

}

然后主机的地址可以忽略,在主机中读取从机的数据,地址需要时从机地址左移一位与上1:
//查询读取从机6个字节数据,从机地址为0x0A
HAL_I2C_Master_Receive(&hi2c1,(0x0A<<1)|0x01,I2CBuf_RxData,6,1000);

向从机发送数据:
HAL_I2C_Master_Transmit(&hi2c1,(0x0A<<1)|0x00,I2CBuf_TxData,6,1000);

中断收发:
主机的比较简单,使用查询来读取发送。从机使用中断就麻烦了。
从机的HAL库同时配置了接收和发送之后,查看寄存器只有接收中断,不知道是怎么回事,这里我就分开用了。因为从机只需要就收一次配置,我就先配置为接收中断,配置完成以后转换为发送中断。(有其他方法解决,使用ADD地址中断,在ADD回调函数里面来判断收发。)
配置函数就是上面的,从机模式可以先使能接受中断:
HAL_I2C_Slave_Receive_IT(&hi2c1, I2CBuf_RxData, sizeof(I2CBuf_RxData)); // 启动中断接收
在这个函数里面,有接收中断,地址终端,监听中断。调用了下面这个

    
    
    
    I2C_Enable_IRQ(hi2c, I2C_XFER_RX_IT | I2C_XFER_LISTEN_IT);

当ADD中断产生后TXIS置位TXIE打开,开始发送数据

好了,发送完成了,然后就是比较坑的了,完成后中断是会被关闭的。。。
所以啊还需要在发送完成回调函数里面再次打开,简单一句,浪费我好多天才跟踪到(划水划水)

void HAL_I2C_SlaveTxCpltCallback(I2C_HandleTypeDef *hi2c)
{
	HAL_I2C_Slave_Transmit_IT(&hi2c1, I2CBuf_TxData, sizeof(I2CBuf_TxData));    // 启动中断发送
}

接收函数也是如此,想再次接受数据就需要再次打开

void HAL_I2C_SlaveRxCpltCallback(I2C_HandleTypeDef *hi2c)
{
	if(hi2c->Instance == I2C1)
	{
		//PROGRESS READ DATA//SET Static Value
		if(I2CBuf_RxData[0] == I2CBuf_RxData[1])
		{
		//处理数据
		}
		memset(I2CBuf_RxData,0,sizeof(I2CBuf_RxData));
		HAL_I2C_Slave_Receive_IT(hi2c, I2CBuf_RxData, sizeof(I2CBuf_RxData));
	}
}

当然终端还需要一个错误处理函数。总线很容易锁定。

void HAL_I2C_ErrorCallback(I2C_HandleTypeDef *hi2c)
{
	if(hi2c->Instance == I2C1)
	{
		ResetFlag = 1;
		memset(I2CBuf_RxData,0,sizeof(I2CBuf_RxData));
		HAL_I2C_DeInit(hi2c);
		MX_I2C1_Init();
//		HAL_I2C_Slave_Receive_IT(hi2c, I2CBuf_RxData, sizeof(I2CBuf_RxData));
		HAL_I2C_Slave_Transmit_IT(&hi2c1, I2CBuf_TxData, sizeof(I2CBuf_TxData));    // 启动中断发送
	}
}

锁定之后需要总线复位,这里参考网上的改了一点,效果不错。

void HAL_I2C_MspInit(I2C_HandleTypeDef* i2cHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(i2cHandle->Instance==I2C1)
  {
  

  

    
    __HAL_RCC_I2C1_CLK_ENABLE();//放到了前面
    __HAL_RCC_GPIOB_CLK_ENABLE();
    //初始化引脚拉高拉低,复位IIC
    GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_PULLUP;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6|GPIO_PIN_7, GPIO_PIN_SET); 
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6|GPIO_PIN_7, GPIO_PIN_RESET);
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6|GPIO_PIN_7, GPIO_PIN_SET);  
    i2cHandle->Instance->CR1 = I2C_CR1_SWRST;   //复位I2C
    i2cHandle->Instance->CR1 = 0;              //解除复位
		
    
    GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Alternate = GPIO_AF1_I2C1;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
		
    
    HAL_NVIC_SetPriority(I2C1_IRQn, 0, 0);
    HAL_NVIC_EnableIRQ(I2C1_IRQn);
  

  
  }
}

应该没有了吧,调了这么多天,总结一下做个笔记,希望能帮到你。

对了,还看到使用地址中断,在地址回调函数里面判断收发,在调用对应函数的,可以同时中断收发,地址在下面,写的很详细

void HAL_I2C_AddrCallback(I2C_HandleTypeDef *hi2c, uint8_t TransferDirection, uint16_t AddrMatchCode)
{
  if(TransferDirection == I2C_DIRECTION_TRANSMIT)
  {
	if(HAL_I2C_Slave_Seq_Receive_IT(&hi2c1, i2c.RxData, sizeof(i2c.RxData), I2C_FIRST_frame) != HAL_OK)
	{

	}	  
	  
  }
  else if(TransferDirection == I2C_DIRECTION_RECEIVE)
  {
	if(HAL_I2C_Slave_Seq_Transmit_IT(&hi2c1, i2c.TxData, sizeof(i2c.TxData), I2C_LAST_frame)!= HAL_OK)
	{
		
	}  
  }
}
————————————————
版权声明:本文为CSDN博主「@残梦」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_36561846/article/details/117474070

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5703096.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存