重构外部中断回调函数来区分外部中断具体引脚做具体对应的任务(STM32+CubeMX+HAL库+EXTI外部中断)

1.STM32CubeMX配置如下:

在这里插入图片描述
这里GPIO mode一共有六种分别是:在这里插入图片描述
(1)上升沿触发外部中断
(2)下降沿触发外部中断
(3)边沿触发外部中断
(4)上升沿触发外部事件
(5)下降沿触发外部事件
(6)边沿触发外部事件

2.CubeMX生成的代码:

/** Configure pins as
        * EXTI
*/
void MX_GPIO_Init(void)
{
	__HAL_RCC_GPIOD_CLK_ENABLE();  //使能GPIO时钟
	GPIO_InitTypeDef GPIO_InitStruct = {0};
	
	GPIO_InitStruct.Pin = DATA_433M_Pin;
	GPIO_InitStruct.Mode = GPIO_MODE_IT_RISING;  //上升沿触发
	GPIO_InitStruct.Pull = GPIO_PULLDOWN; //下拉
	HAL_GPIO_Init(DATA_433M_GPIO_Port, &GPIO_InitStruct);
	
	HAL_NVIC_SetPriority(EXTI2_3_IRQn, 0, 0);//设置中断优先级
	HAL_NVIC_EnableIRQ(EXTI2_3_IRQn);//使能外部中断
}

/******************************************************************************/
/* STM32G0xx Peripheral Interrupt Handlers                                    */
/* Add here the Interrupt Handlers for the used peripherals.                  */
/* For the available peripheral interrupt handler names,                      */
/* please refer to the startup file (startup_stm32g0xx.s).                    */
/******************************************************************************/

/**
  * @brief This function handles EXTI line 2 and line 3 interrupts.
  */
void EXTI2_3_IRQHandler(void)
{
  /* USER CODE BEGIN EXTI2_3_IRQn 0 */

  /* USER CODE END EXTI2_3_IRQn 0 */
  HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_2);
  /* USER CODE BEGIN EXTI2_3_IRQn 1 */

  /* USER CODE END EXTI2_3_IRQn 1 */
}

3.分析外部中断过程

我们可以看见这个外部中断包含了GPIO_Pin_2和GPIO_Pin_3两个外部中断源,这两个脚都可以触发这个外部中断,如果我们想区分具体是2还是3脚的时候,或者需要区分上升沿做不同事情,那么就需要自己重构回调函数来区分,然后执行不同的任务。我们发现里面只有一个函数,打开看看如下:

/**
  * @brief  Handle EXTI interrupt request.
  * @param  GPIO_Pin Specifies the port pin connected to corresponding EXTI line.
  * @retval None
  */
void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
  /* EXTI line interrupt detected */
  if (__HAL_GPIO_EXTI_GET_RISING_IT(GPIO_Pin) != 0x00u)
  {
    __HAL_GPIO_EXTI_CLEAR_RISING_IT(GPIO_Pin);
    HAL_GPIO_EXTI_Rising_Callback(GPIO_Pin);
  }

  if (__HAL_GPIO_EXTI_GET_FALLING_IT(GPIO_Pin) != 0x00u)
  {
    __HAL_GPIO_EXTI_CLEAR_FALLING_IT(GPIO_Pin);
    HAL_GPIO_EXTI_Falling_Callback(GPIO_Pin);
  }
}

可以发现里面里面一共两个大的判断,一个是上升沿,一个是下降沿。然后先清楚中断标志位,然后执行一个回调函数,这个回调函数是可以自己重构的,仔细看下这个回调函数:

/**
  * @brief  EXTI line detection callback.
  * @param  GPIO_Pin Specifies the port pin connected to corresponding EXTI line.
  * @retval None
  */
__weak void HAL_GPIO_EXTI_Rising_Callback(uint16_t GPIO_Pin)
{
  /* Prevent unused argument(s) compilation warning */
  UNUSED(GPIO_Pin);

  /* NOTE: This function should not be modified, when the callback is needed,
           the HAL_GPIO_EXTI_Rising_Callback could be implemented in the user file
   */
}

凡是前面带__weak的我们都是可以再重构而不会报错的。

4.重构回调函数

//下降沿回调函数
void HAL_GPIO_EXTI_Rising_Callback(uint16_t GPIO_Pin)
{
 if(GPIO_Pin == GPIO_PIN_2)
  {
    SEGGER_RTT_printf(0,"GPIO_Pin_2 Rising exti!\r\n");
  }
  else if(GPIO_Pin == GPIO_PIN_3)
    {
    SEGGER_RTT_printf(0,"GPIO_Pin_3 Rising exti!\r\n");
  }
}
//上升沿回调函数
void HAL_GPIO_EXTI_Falling_Callback(uint16_t GPIO_Pin)
{
 if(GPIO_Pin == GPIO_PIN_2)
  {
    SEGGER_RTT_printf(0,"GPIO_Pin_2 Falling exti!\r\n");
  }
  else if(GPIO_Pin == GPIO_PIN_3)
    {
    SEGGER_RTT_printf(0,"GPIO_Pin_3 Falling exti!\r\n");
  }
}

5.调试结果

在这里插入图片描述

6.增加GPIO端口判断

1)解决办法:

进中断回调函数的时候通过if(HAL_GPIO_ReadPin(GPIOD,DATA_433M_Pin)==1)读取对应引脚电平状态来区分

void HAL_GPIO_EXTI_Rising_Callback(uint16_t GPIO_Pin)
{
 if(GPIO_Pin == GPIO_PIN_2)
  {
		if(HAL_GPIO_ReadPin(GPIOD,DATA_433M_Pin)==1) 
            SEGGER_RTT_printf(0,"GPIO_Pin_2 Rising exti!\r\n");
  }
  else if(GPIO_Pin == GPIO_PIN_3)
    {
    SEGGER_RTT_printf(0,"GPIO_Pin_3 Rising exti!\r\n");
  }
}
2)实测如下:

在这里插入图片描述
第一个是if(HAL_GPIO_ReadPin(GPIOD,DATA_433M_Pin) == 1)的结果,因为我们是上升沿,所以触发中断的时候肯定是个高电平,所以这个时候是可以正常打印RTT的。
第二个是if(HAL_GPIO_ReadPin(GPIOD,DATA_433M_Pin) == 0)的结果,因为我们是上升沿,所以触发中断的时候肯定是个高电平,而我们条件是低电平才会正常打印,所以退出低功耗的时候没有看见RTT打印的log。

  • 2
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
STM32 HAL 中,外部中断回调函数需要通过 HAL_GPIO_EXTI_Callback() 函数来实现。具体步骤如下: 1. 在 CubeMX 中配置外部中断,选择相应的 GPIO 引脚和触发方式。 2. 在 main() 函数中初始化 GPIO 和中断。 3. 在中断回调函数 HAL_GPIO_EXTI_Callback() 中编写相应的处理代码。 例如,如果我们想在外部中断触发时改变 LED 灯的状态,可以按照以下步骤操作: 1. 在 CubeMX 中配置外部中断,选择相应的 GPIO 引脚和触发方式。 2. 在 main() 函数中初始化 GPIO 和中断。例如,我们将 PA0 引脚配置为外部中断,连接到一个按键,同时将 PB0 引脚配置为输出,连接到一个 LED 灯。初始化代码如下: ``` /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); /*Configure GPIO pin : PA0 */ GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_IT_FALLING; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); /*Configure GPIO pin : PB0 */ GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); /* EXTI interrupt init*/ HAL_NVIC_SetPriority(EXTI0_IRQn, 0, 0); HAL_NVIC_EnableIRQ(EXTI0_IRQn); ``` 3. 在中断回调函数 HAL_GPIO_EXTI_Callback() 中编写相应的处理代码。例如,我们在按键按下时改变 LED 灯的状态。代码如下: ``` void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { if (GPIO_Pin == GPIO_PIN_0) { HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_0); } } ``` 这样,当按键按下时,会触发外部中断,进而调用 HAL_GPIO_EXTI_Callback() 函数,从而改变 LED 灯的状态。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值