STM32 低功耗停机模式(STOP)管脚中断(INTERRUPT)唤醒实现及优化

本文介绍了如何在STM32L4系列芯片上实现低功耗停机模式(STOP)并使用管脚中断进行唤醒。详细阐述了STOP模式的配置、中断设置和进入STOP模式的库函数使用。同时,针对存在多种中断源的情况,提出了优化方案,确保只有指定中断能唤醒STOP模式,增强了系统在低功耗模式下的稳定性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

STM32 低功耗停机模式(STOP)管脚中断(INTERRUPT)唤醒实现及优化

1. 介绍

STM32具有多种低功耗模式,当前以STM32L4系列的低功耗模式最为丰富,此处基于STM32L476和STM32CUBEIDE环境介绍停机模式(STOP)管脚中断唤醒的实现(HAL库)。STOP模式只是停止代码执行,唤醒(其实是continue的作用)后继续执行后面的代码,而不是重启之后从初始代码开始执行。

2. 低功耗模式

STM32L4的低功耗模式,相比其它系列的芯片,多了几种:
在这里插入图片描述
三种STOP模式的区别,可以参考文档:RM0351 Reference manual

STM32L4系列各种低功耗模式的特性总结如下:
在这里插入图片描述

3. 管脚中断配置

STOP模式可以通过通用管脚中断(Interrupt)或事件(Event)的方式唤醒,这里介绍通用管脚中断唤醒的方式, 选择一个GPIO管脚进行配置,这里用PC13作为唤醒源。

设置PC13为GPIO_EXTI方式:
在这里插入图片描述
为PC13选择其中一种中断触发方式,包括上升沿触发,下降沿触发和电平变化触发(即上升沿下降沿都触发):
在这里插入图片描述
然后使能中断:
在这里插入图片描述
保存后生成代码:
在这里插入图片描述

4. STOP模式进入及中断唤醒

通过HAL库函数可进入STOP模式,其中库函数有一个参数,指定唤醒时的来源,有两个可选:PWR_STOPENTRY_WFI 和 PWR_STOPENTRY_WFE, 分别对应管脚中断唤醒和管脚事件唤醒,这里我们用PWR_STOPENTRY_WFI,即管脚中断唤醒。

如果要进入STOP0模式,实施下面代码:

HAL_PWREx_EnterSTOP0Mode(PWR_STOPENTRY_WFI);

如果要进入STOP1模式,实施下面代码:

HAL_PWREx_EnterSTOP1Mode(PWR_STOPENTRY_WFI);

如果要进入STOP2模式,实施下面代码:

HAL_PWREx_EnterSTOP2Mode(PWR_STOPENTRY_WFI);

在执行到上述代码时,MCU就进入了STOP状态,等待管脚上发生信号变化产生的中断唤醒。在接受到中断唤醒后,会先进入中断处理函数,再出来执行进入STOP模式代码后面的代码。再PC13的管脚中断处理函数里,需要重新调用初始化时钟配置函数,以使得因为进入STOP模式关闭的时钟重新打开。PC13管脚的中断处理函数:

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
   if (GPIO_Pin==GPIO_PIN_13)
    {
	  SystemClock_Config();
	}
}

这样STOP模式的进入和唤醒就实现了。

5. STOP模式进入及中断唤醒的优化

当设计中存在多种中断源,包括调试器中断和其它管脚中断时,这些中断不是想用于唤醒STOP状态的中断,此时就需要增加必要的设计,实现:

  1. 当指定的中断源产生中断时,唤醒STOP状态;
  2. 当非指定的中断源产生中断唤醒STOP后,立即重新进入STOP模式。

实现方式如下,先增加一个判断变量flag_recog:

uint8_t flag_recog = 0;

在进入STOP0模式时,采用如下代码:

flag_recog = 0;
while(flag_recog == 0) HAL_PWREx_EnterSTOP0Mode(PWR_STOPENTRY_WFI);

在进入STOP1模式时,采用如下代码:

flag_recog = 0;
while(flag_recog == 0) HAL_PWREx_EnterSTOP1Mode(PWR_STOPENTRY_WFI);

在进入STOP2模式时,采用如下代码:

flag_recog = 0;
while(flag_recog == 0) HAL_PWREx_EnterSTOP2Mode(PWR_STOPENTRY_WFI);

这样,当异常中断引起唤醒时,因为flag_recog保持为0,所以会再次执行进入STOP模式的代码。

当指定的唤醒中断到来时,如这里的PC13,则修改中断处理函数为:

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	if (GPIO_Pin==GPIO_PIN_13)
	{
		flag_recog = 1;
		SystemClock_Config();
	}
}

从而在中断处理函数里将flag_recog设置为非0,在退出中断处理函数后,也会退出while(flag_recog == 0)的循环,执行后面的代码。

通过以上优化,提高了STOP低功耗模式应用的可靠性,避免异常时间的唤醒。

6. 库函数对应关系

STM32L4系列将STOP模式分为了STOP0, STOP1和STOP2模式,实际上,STOP0和STOP1和以前的STOP模式在库函数上有对应关系。譬如以前的STOP模式的库函数在STM32L4的定义:

/**
  * @brief Enter Stop mode
  * @note  This API is named HAL_PWR_EnterSTOPMode to ensure compatibility with legacy code running
  *        on devices where only "Stop mode" is mentioned with main or low power regulator ON.
  * @note  In Stop mode, all I/O pins keep the same state as in Run mode.
  * @note  All clocks in the VCORE domain are stopped; the PLL, the MSI,
  *        the HSI and the HSE oscillators are disabled. Some peripherals with the wakeup capability
  *        (I2Cx, USARTx and LPUART) can switch on the HSI to receive a frame, and switch off the HSI
  *        after receiving the frame if it is not a wakeup frame. In this case, the HSI clock is propagated
  *        only to the peripheral requesting it.
  *        SRAM1, SRAM2 and register contents are preserved.
  *        The BOR is available.
  *        The voltage regulator can be configured either in normal (Stop 0) or low-power mode (Stop 1).
  * @note  When exiting Stop 0 or Stop 1 mode by issuing an interrupt or a wakeup event,
  *         the HSI RC oscillator is selected as system clock if STOPWUCK bit in RCC_CFGR register
  *         is set; the MSI oscillator is selected if STOPWUCK is cleared.
  * @note  When the voltage regulator operates in low power mode (Stop 1), an additional
  *         startup delay is incurred when waking up.
  *         By keeping the internal regulator ON during Stop mode (Stop 0), the consumption
  *         is higher although the startup time is reduced.
  * @param Regulator: Specifies the regulator state in Stop mode.
  *          This parameter can be one of the following values:
  *            @arg @ref PWR_MAINREGULATOR_ON  Stop 0 mode (main regulator ON)
  *            @arg @ref PWR_LOWPOWERREGULATOR_ON  Stop 1 mode (low power regulator ON)
  * @param STOPEntry: Specifies Stop 0 or Stop 1 mode is entered with WFI or WFE instruction.
  *          This parameter can be one of the following values:
  *            @arg @ref PWR_STOPENTRY_WFI  Enter Stop 0 or Stop 1 mode with WFI instruction.
  *            @arg @ref PWR_STOPENTRY_WFE  Enter Stop 0 or Stop 1 mode with WFE instruction.
  * @retval None
  */
void HAL_PWR_EnterSTOPMode(uint32_t Regulator, uint8_t STOPEntry)
{
  /* Check the parameters */
  assert_param(IS_PWR_REGULATOR(Regulator));

  if(Regulator == PWR_LOWPOWERREGULATOR_ON)
  {
    HAL_PWREx_EnterSTOP1Mode(STOPEntry);
  }
  else
  {
    HAL_PWREx_EnterSTOP0Mode(STOPEntry);
  }
}

因此通过调用STOP模式的库函数,设置不同的输入参数,可以等效调用STOP0和STOP1模式。对于非STM32L4系列,可以调用HAL_PWR_EnterSTOPMode库函数进入STOP模式。

–End–

### 使用 HAL 库在 STM32L0 上实现低功耗模式配置 #### 1. 配置 STM32CubeMX 工具 通过 STM32CubeMX 可以简化低功耗模式的初始化设置。启动该工具并创建新项目,选择目标设备为 STM32L0 系列微控制器。 - 设置系统时钟源和频率。 - 启用必要的外设功能模块。 - 进入 "Pinout & Configuration" 页面,在 “System Core” 中找到 `PWR` 和 `RCC` 模块来调整电源管理和复位与时钟控制参数[^2]。 #### 2. 编写进入 STOP 或 STANDBY 模式的函数 编写应用程序代码之前先定义好要使用的具体硬件资源以及相应的中断服务程序 (ISR),这里给出一个简单的例子展示如何利用 RTC 定时器从停止状态唤醒: ```c #include "stm32l0xx_hal.h" void EnterStopMode(void){ /* Enable Power Clock */ __HAL_RCC_PWR_CLK_ENABLE(); /* Clear Wakeup flag before entering Stop mode */ WRITE_REG(PWR->CR, PWR_CR_CWUF); /* Select the regulator state in Stop mode */ SET_BIT(PWR->CR,PWR_CR_LPDS); /* Set SLEEPDEEP bit of Cortex System Control Register */ SCB->SCR |= SCB_SCR_SLEEPDEEP_Msk; /* Request Wait For Interrupt */ __WFI(); } ``` 此段代码展示了如何使能电源管理单元时钟、清除任何现存的唤醒标志位、设定低压检测选项、激活深度睡眠指示符并将处理器置于等待事件/中断的状态下运行[^1]。 对于更复杂的场景比如结合外部按键触发退出 Standby Mode,则需进一步配置 EXTI 外部中断线,并注册对应的回调处理逻辑。 #### 3. 实现完整的唤醒机制 为了确保当指定条件达成时可以从低能耗状态下恢复过来继续执行后续操作,还需要完成如下工作: - 如果采用定时方式作为唤醒源的话应该预先配置好实时时钟(RTC); - 对于基于 I/O 引脚变化而产生的唤醒信号则要相应地开启边沿敏感型外部中断请求; - 当然也可以组合上述两种方法甚至更多种不同的唤醒途径一起使用以适应实际应用场景的需求。 下面是一个简单示例说明怎样借助按压开关产生脉冲从而打断休眠过程重新回到正常工作的流程片段: ```c // 假定已经完成了对 GPIOA_PIN_0 的初始化以便用来监测物理按钮动作 __HAL_GPIO_EXTI_CLEAR_IT(GPIO_PIN_0); // 清除可能存在的未决 IT 请求 while(1){ // 执行某些任务... if(/*满足特定条件下*/true){ EnterStopMode(); // 调用前面定义好的停机接口 // 此处会一直停留直到发生有效的唤醒事件为止... // 继续其他业务逻辑... } } /* 用户自定义的EXTI线路中断服务子程序模板*/ void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){ switch(GPIO_Pin){ case GPIO_PIN_0 : // Do something after wakeup by button press. break; default: ; } } ``` 这段代码实现了基本的按键唤醒功能,其中包含了对外围输入端口 PA0 发生下降沿跳变后的响应措施描述。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

PegasusYu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值