细说STM32F407单片机电源低功耗SleepMode模式及应用示例

目录

一、STM32F4的低功耗模式

1、睡眠(Sleep)模式

2、停止(Stop)模式

3、待机(Standby)模式

二、睡眠模式

1、进入睡眠模式

2、睡眠模式的状态

3、退出睡眠模式

4、SysTick的影响

三、应用示例

1、工程配置

(1) 时钟、DEBUG、GPIO、CodeGenerator

(2) USART6

(3) NVIC

2、软件设计

(1) main.h

(2)main.c

四、下载并调试


        电池供电的嵌入式系统一般非常注意功耗控制,尽量使系统的功耗最低。STM32F4系列MCU提供了多种运行模式,CubeMX也提供了功耗分析的功能。本文介绍STM32F4的SleepMode功耗模式,以及如何通过该功耗模式的控制实现系统的低功耗。

一、STM32F4的低功耗模式

        系统复位后,MCU处于正常运行模式。在正常运行模式下,CPU由HCLK时钟信号驱动连续执行程序指令。用户可以采取一些措施降低系统正常运行时的功耗,例如,可以降低HCLK时钟频率,或者将不使用的外设的时钟信号关闭。

        从main()函数的代码可以看出,在执行完各种初始化后,最后都是执行一个while()死循环。在while()循环里,通过轮询方式处理各种事务,或通过中断响应处理各种事务。在正常运行模式下,while()循环里的程序代码是一直执行的,即使一行代码都没有。所以在正常运行模式下,一般的嵌入式系统的CPU计算时间都是浪费的。

        除了正常运行模式,STM32F4系列MCU还有3种低功耗模式。

1、睡眠(Sleep)模式

        Cortex-M4内核时钟停止,1.2V调压器正常工作,外设保持运行。通过WFI(wait for interrupt)或WFE(wait for event)指令进入睡眠模式。进入睡眠模式后,CPU不再执行新的代码。CPU可以被中断或事件唤醒,唤醒后继续执行进入睡眠点之后的代码。

2、停止(Stop)模式

        1.2V域所有时钟都停止,所有外设停止工作,内部调压器可以处于运行或低功耗模式,内部SRAM和寄存器的内容被保留,HSI和HSE振荡器关闭。通过EXTI中断或EXTI事件唤醒,CPU从停止处继续执行代码。

3、待机(Standby)模式

        调压器停止,1.2V域断电,内部SRAM和寄存器的内容丢失。只能通过SYS_WKUP引脚的上升沿、RTC闹钟事件、RTC唤醒事件、RTC入侵事件、NRST引脚外部复位等唤醒。从待机模式唤醒相当于系统复位,程序从头开始执行。

        在这3种低功耗模式中,待机模式功耗最低,但是从待机模式唤醒相当于系统复位,程序从头开始执行。睡眠模式和停止模式都能停止CPU的程序执行,被唤醒后,从程序停止处继续执行。应根据系统的实际功能需求选择合适的低功耗模式。

二、睡眠模式

1、进入睡眠模式

        通过执行Cortex-M4内核的WFI(Wait For Interrupt)指令或WFE(Wait For Event)指令可以进入睡眠模式。根据Cortex-M4F系统控制寄存器(System Control Register,SCR)的SLEEPONEXIT位的设置,有两种进入睡眠模式的方式。

  • 立即睡眠:如果SLEEPONEXIT位是0,MCU在执行WFI指令或WFE指令时,立即进入睡眠模式。
  • 退出时睡眠:如果SLEEPONEXIT位是1,MCU在退出优先级最低的中断ISR后,立即进入睡眠模式。
  •         在进入睡眠模式之前,可以调用HAL的驱动函数设置SLEEPONEXIT的值,这两个函数原型如下:

    void HAL_PWR_EnableSleepOnExit(void)	//将SLEEPONEXIT位置1
    void HAL_PWR_DisableSleepOnExit(void)	//将SLEEPONEXIT位清零

            进入睡眠模式的HAL函数是HAL_PWR_EnterSLEEPMode(),其源代码如下:

    void HAL_PWR_EnterSLEEPMode(uint32_t Regulator, uint8_t SLEEPEntry)
    {
      /* Prevent unused argument(s) compilation warning */
      UNUSED(Regulator);
    
      /* Check the parameters */
      assert_param(IS_PWR_REGULATOR(Regulator));
      assert_param(IS_PWR_SLEEP_ENTRY(SLEEPEntry));
    
      /* Clear SLEEPDEEP bit of Cortex System Control Register */
      CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
    
      /* Select SLEEP mode entry -------------------------------------------------*/
      if(SLEEPEntry == PWR_SLEEPENTRY_WFI)
      {   
        /* Request Wait For Interrupt */
        __WFI();
      }
      else
      {
        if(SLEEPEntry != PWR_SLEEPENTRY_WFE_NO_EVT_CLEAR)
        {
          /* Clear all pending event */
          __SEV();
          __WFE();
        }
    
        /* Request Wait For Event */
        __WFE();
      }
    }

             其中,参数Regulator表示调压器在睡眠模式下的状态。其取值使用如下宏定义常量。

    PWR_MAINREGULATOR_ON,调压器正常运行。
    PWR_LOWPOWERREGULATOR_ON,调压器处于低功耗模式。

            但是参数Regulator的取值在这个函数中并没有意义,因为STM32F4系列MCU在睡眠模式下,调压器总是处于运行状态,而不能是低功耗状态。这个参数是为了与低功耗系列的STM32F MCU的驱动函数相兼容。

            参数SLEEPEntry表示以何种指令进入睡眠模式,WFI指令或WFE指令。其取值使用如下宏定义常量。

    PWR_SLEEPENTRY_WFI,使用WFI指令进入睡眠模式。
    PWR_SLEEPENTRY_WFE,使用WFE指令进入睡眠模式。

            函数HAL_PWR_EnterSLEEPMode()内部会首先将系统控制寄存器SCR的SLEEPDEEP位清零,这个位如果置1就是深度睡眠模式,在进入停止模式时才将SLEEPDEEP位置1。

    2、睡眠模式的状态

            进入睡眠模式后,系统的状态如下:

  • CPU的时钟关闭,CPU停止运行,也就是程序暂停。
  • 所有外设的时钟不停止,外设正常运行,所有I/O引脚的状态与运行时相同。
  • 调压器正常运行。
  • 3、退出睡眠模式

            如果使用WFI指令进入睡眠模式,则NVIC确认的任何外设中断都可以将MCU唤醒。由中断唤醒后,先执行中断的ISR,然后执行WFI指令后面的程序。

            如果使用WFE指令进入睡眠模式,MCU将在有事件发生时立即退出睡眠模式,并执行WFE后的程序。唤醒事件可以通过以下方式产生。

  • 在外设的控制寄存器中使能一个中断事件,但是不在NVIC中使能其全局中断,同时使能系统控制寄存器SCR中的SEVONPEND(Send Event on Pending bit)位。当MCU从WFE恢复时,需要清除相应外设的事件中断标志位和外设NVIC中断挂起位。
  • 配置一个外部或内部EXTI线为事件模式。当CPU从WFE中恢复时,因为对应事件线的挂起位没有被置位,不必清除相应外设的中断标志位或NVIC中断通道挂起位。HAL库中有两个函数用于设置系统控制寄存器SCR中的SEVONPEND位的值。
  • void HAL_PWR_EnableSEVOnPend(void)	//SEVONPEND位置1
    void HAL_PWR_DisableSEVOnPend(void)	//SEVONPEND位清零

            从睡眠模式唤醒的响应没有任何延迟,是3种低功耗模式中唤醒响应最快的。

    4、SysTick的影响

            由于睡眠模式可以由任意中断或事件唤醒,而MCU在HAL初始化时就开启了Cortex-M内核的SysTick定时器,这个定时器每隔1ms中断一次。如果MCU处于睡眠状态,SysTick定时器的中断会将MCU从睡眠模式唤醒。

            如果要使睡眠模式不受SysTick中断的影响,需要在进入睡眠状态之前停止SysTick定时器,从睡眠状态恢复后又立即开启SysTick定时器,因为延时函数HAL_Delay()需要用到SysTick定时器。文件stmf4xx_hal.h定义了两个控制SysTick定时器的函数,两个函数原型定义如下:

    void HAL_SuspendTick(void);	//暂停SysTick定时器的运行
    void HAL_ResumeTick(void);	//恢复SysTick定时器的运行

    三、应用示例

    本文将创建一个示例项目,测试系统的睡眠模式。继续使用旺宝红龙开发板STM32F407ZGT6 KIT V1.0。示例功能和操作流程如下。

  • 将连接KeyRight键的PF6引脚配置为外部中断EXTI6。
  • 在主程序的while循环里,使系统进入睡眠状态后,按下KeyRight键把系统从睡眠状态唤醒。
  •         本示例要用到USART6、LED1(PA6)和KeyRight键,但是需要将连接KeyRight键的PF6引脚重新设置为外部中断线EXTI6,并设置上拉和下跳沿触发中断。本示例中KeyRight和LED1的引脚GPIO设置结果如图所示。用户还需要在NVIC中开启EXTI6的中断。

            本文引用KEYLED文件夹里的文件,其使用方法和管脚配置请看参考文章。

            参考文章:细说STM32F407单片机以DMA方式读写外部SRAM的方法_片外sram访问熟读-CSDN博客  细说STM32F407单片机以DMA方式读写外部SRAM的方法_片外sram访问熟读-CSDN博客

    1、工程配置

    (1) 时钟、DEBUG、GPIO、CodeGenerator

            外部时钟,25MHz,设置到HCLK=168MHz,PCLK1=42MHz,PCLK2=84MHz,其它,都设置成168MHz。

            DEBUG,选择serial wire,CodeGenerator的设置同参考文章。

     

    (2) USART6

            使用管脚PG9、PG14,默认其它参数。

    (3) NVIC

    2、软件设计

    (1) main.h

            即使PF6管脚被定义多重功能, IDE也自动生成如下驱动。

    /* Private defines -----------------------------------------------------------*/
    #define KeyRight_Pin GPIO_PIN_6
    #define KeyRight_GPIO_Port GPIOF
    #define KeyRight_EXTI_IRQn EXTI9_5_IRQn
    #define LED1_Pin GPIO_PIN_6
    #define LED1_GPIO_Port GPIOA

    (2)main.c

    /* USER CODE BEGIN 2 */
      printf("Demo22_1_SleepMode:Test Sleep Mode.\r\n\r\n");
      LED1_ON();
      HAL_Delay(1000);	//系统复位后,LED1 1秒后进入睡眠状态
      /* USER CODE END 2 */
      /* USER CODE BEGIN 3 */
    	printf("Press KeyRight[S5] to wake up.\r\n");
    	printf("After entering SleepMode,");
    	printf("the LED1 flashes until the MCU wakes up.\r\n\r\n");
    	LED1_OFF();
    	HAL_SuspendTick();	//使SysTick定时器暂停
    
    	/* 进入睡眠状态,对于正常的中断,WFI和WFE两种参数都可以唤醒,因为中断肯定是事件 */
    	HAL_PWR_EnterSLEEPMode(PWR_LOWPOWERREGULATOR_ON, PWR_SLEEPENTRY_WFI);
    
    	/* 按键的EXTI中断唤醒后执行下面的代码 */
    	HAL_ResumeTick();	//恢复SysTick定时器
    	printf("Resumed from SleepMode.\r\n\r\n");
    	for(uint8_t i=0; i<9;i++)	//使LED1 闪烁,延时也可消除按键抖动影响
    	{
    	  	LED1_Toggle();
    	  	HAL_Delay(500);
    	}
      }
      /* USER CODE END 3 */

            本示没有为外部中断EXTI6编写回调函数代码,所有用户代码都在main()函数里。

            在while循环里,在使系统进入睡眠状态之前先熄灭LED1,调用函数HAL_SuspendTick()使SysTick定时器暂停,然后调用HAL_PWR_EnterSLEEPMode()函数,用WFI指令进入睡眠模式。在进入睡眠模式后,CPU时钟停止,程序就暂停了。

            在用户按下KeyRight[S5]键时产生EXTI中断,系统被唤醒,继续执行后面的代码。程序先调用函数HAL_ResumeTick()恢复SysTick定时器的运行,因为后面的代码里需要使用HAL_Delay()函数,要用到SysTick定时器。

            本示例在执行函数HAL_PWR_EnterSLEEPMode()进入睡眠模式时,使用WFI方式或WFE方式的效果是一样的,因为中断必然是事件引起的,而事件不一定产生中断。

    /* USER CODE BEGIN 4 */
    // 唤醒SleepMode也可以在这里通过调用回调函数实现
    
    int __io_putchar(int ch)
    {
    	HAL_UART_Transmit(&huart6,(uint8_t*)&ch,1,0xFFFF);
    	return ch;
    }
    /* USER CODE END 4 */

    四、下载并调试

            运行时可以看到示例程序是按期望运行的。系统提示进入睡眠模式后,按KeyRight键可唤醒系统,唤醒后显示提示信息,并且使LED1闪烁几次,然后又进入睡眠状态。要唤醒系统,需要再按KeyRight键。

            如果将程序中调用函数HAL_SuspendTick()的那行语句注释掉,也就是不暂停SysTick定时器,会发现运行时LED1一直闪烁。这是因为在进入睡眠状态后,SysTick定时器中断就将系统唤醒了,而SysTick定时器中断的触发周期是1ms,所以系统睡眠不超过1ms就被唤醒了。在这个程序中,while循环里的代码不会被CPU一直高速循环执行,进入睡眠模式会使CPU暂停执行程序,被唤醒后才继续执行睡眠点之后的代码。本示例为了演示的需要,在系统被唤醒后,用for循环执行了约4000ms的程序(也起到消除按键抖动影响的作用)。在实际的系统中,程序可能大部分时间处于睡眠状态,执行程序的时间可能很短,例如,睡眠1000ms,执行程序才1ms,这样可以大大降低系统的功耗。

            在CubeMX中,为本项目进行功耗计算,只设置了RUN和SLEEP两种步骤,步骤设置中只开启系统实际用到的外设。运行模式下的耗电流是47.18mA,而睡眠模式下的电流是13.18mA。

    作者:wenchm

    物联沃分享整理
    物联沃-IOTWORD物联网 » 细说STM32F407单片机电源低功耗SleepMode模式及应用示例

    发表回复