当前位置: 首页 > article >正文

细说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。


http://www.kler.cn/a/509246.html

相关文章:

  • 基于.Net Core+Vue的文件加密系统
  • K8S开启/关闭审计日志
  • 【PyQt】图像处理系统
  • 动态路由vue-router
  • LabVIEW与WPS文件格式的兼容性
  • Redis 3.2.1在Win10系统上的安装教程
  • ASP.NET Core - 配置系统之配置提供程序
  • 【Linux系列】查看服务器是否使用了 SSD 的多种方法
  • Spark Streaming的核心功能及其示例PySpark代码
  • Linux——线程的慨念及控制
  • Oracle数据库传统审计怎么用
  • 网络安全构成要素
  • 基于高光谱数据的叶片水分估测方法研究 【Matlab Python Origin】
  • 用c语言实现哈希表:03_hashing (哈希函数设计)
  • Flask学习入门笔记
  • 电梯系统的UML文档05
  • java根据模板导出word,并在word中插入echarts相关统计图片以及表格
  • 爱德华EDWARDS EPX180EPX500 Vacuum Pumps instruction Manual
  • 基于 Java Spring Boot 开发的项目复盘与技术总结
  • Spring Boot应用关闭分析
  • Linux中的nc命令是网络工具中的一种,用于进行网络连接和数据传输。下面是nc命令的使用方法和实例:
  • 2.使用Spring BootSpring AI快速构建AI应用程序
  • PHP企业IM客服系统
  • 鸿蒙学习构建视图的基本语法(二)
  • Jenkins下载安装
  • 利用 LNMP 实现 WordPress 站点搭建