STM32电源管理实现低功耗
STM32电源管理简介
电源对电子设备的重要性不言而喻,它是保证系统稳定运行的基础,而保证系统能稳定运行后,又有低功耗的要求。在很多应用场合中都对电子设备的功耗要求非常苛刻,如某些传感器信息采集设备,仅靠小型的电池提供电源,要求工作长达数年之久,且期间不需要任何维护;由于智慧穿戴设备的小型化要求,电池体积不能太大导致容量也比较小,所以也很有必要从控制功耗入手,提高设备的续行时间。STM32有专门的电源管理外设监控电源并管理设备的运行模式,确保系统正常运行,并尽量降低器件的功耗。
STM32电源管理系统
STM32的电源管理系统主要分为:
- ①备份域
- ②调压器供电电路
- ③ADC电源电路
备份域电路
STM32的备份域包括LSE振荡器、RTC、备份寄存器及备份SRAM这些器件,这部分的电路可以通过STM32的VBAT引脚获取供电电源,在实际应用中一般会使用3V的钮扣电池对该引脚供电。
在图中备份域电路的左侧有一个电源开关结构,它的功能类似图中的双二极管,在它的上方连接了VBAT电源,下方连接了VDD主电源(一般为3.3V),右侧引出到备份域电路中。当VDD主电源存在时,由于VDD电压较高,备份域电路通过VDD供电,当VDD掉电时,备份域电路由钮扣电池通过VBAT供电,保证电路能持续运行,从而可利用它保留关键数据
调压器供电电路
在STM32的电源系统中调压器供电的电路是最主要的部分,调压器为备份域及待机电路以外的所有数字电路供电,其中包括内核、数字外设以及RAM,调压器的输出电压约为1.2V,因而使用调压器供电的这些电路区域被称为1.2V域。
调压器可控制调节供电电路使系统运行在“运行模式”、“停止模式”以及“待机模式”下:
运行模式:调压器为 1.2 V 域(内核、存储器和数字外设)提供全功率。
停止模式:1.2V域运行在低功耗状态,1.2V区域的所有时钟都被关闭,相应的外设都停止了工作,但它会保留内核寄存器以及SRAM的内容;
待机模式:整个1.2V域都断电,该区域的内核寄存器及SRAM内容都会丢失(备份区域的寄存器及SRAM不受影响)。
ADC电源控制电路
为了提高转换精度,STM32的ADC配有独立的电源接口,方便进行单独的滤波。ADC的工作电源使用VDDA引脚输入,使用VSSA作为独立的地连接,VREF引脚则为ADC提供测量使用的参考电压。
STM32低功耗模式
很多单片机都有低功耗模式,STM32F4也不例外 ,运行状态下的HCLK为 CPU提供时钟,内核执行程序代码。当 CPU不需继续运行时,可以利用多个低功耗模式来节省功耗,例如等待某个外部事件时。
STM32F4按功耗由高到低排列具有运行、睡眠、停止和待机四种工作模式。
上电复位后STM32处于运行状态时,当内核不需要继续运行,就可以选择进入后面的三种低功耗模式降低功耗,这三种模式中,电源消耗不同、唤醒时间不同、唤醒源不同,用户需要根据应用需求,选择最佳的低功耗模式。这三种低功耗模式层层递进,运行的时钟或芯片功能越来越少,因而功耗越来越低。
STM32实现睡眠模式
在睡眠模式中,仅关闭了内核时钟,内核停止运行,但其片上外设,CM4核心的外设全都还照常运行。
有两种方式进入睡眠模式,它的进入方式决定了从睡眠唤醒的方式,分别是WFI(waitforinterrupt)和WFE(waitforevent),即由等待“中断”唤醒和由“事件”唤醒。睡眠模式的各种特性见下表
电气原理图
cubemx配置
因为蜂鸣器是无源蜂鸣器,所以需要用到PWM
代码实现
主函数
HAL_TIM_PWM_Start(&htim10, TIM_CHANNEL_1);
printf("usart is init\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);
while (1)
{
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_RESET);
printf("程序正常运行\n");
HAL_Delay(2000);
HAL_SuspendTick();//关闭systick中断,否则会被中断唤醒
printf("系统进入睡眠模式\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);//指示灯灭掉表示睡眠模式
HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);//进入睡眠模式
printf("系统已经被唤醒\n");
}
中断函数
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
if(GPIO_Pin == GPIO_PIN_0)
{
HAL_ResumeTick();
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 10);//蜂鸣器响一下
HAL_Delay(200);
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 0);
}
}
实验现象为:上电后指示灯亮,串口打印,两秒后进入睡眠模式,使用按键中断唤醒进入中断,然后继续执行函数
STM32实现停止模式
在停止模式中,进一步关闭了其它所有的时钟,于是所有的外设都停止了工作,但由于其1.2V区域的部分电源没有关闭,还保留了内核的寄存器、内存的信息。
所以从停止模式唤醒,并重新开启时钟后,还可以从上次停止处继续执行代码。停止模式可以由任意一个外部中断(EXTI)唤醒。在停止模式中可以选择电压调节器为开模式或低功耗模式,可选择内部FLASH工作在正常模式或掉电模式。
代码实现
主函数
HAL_TIM_PWM_Start(&htim10, TIM_CHANNEL_1);
printf("usart is init\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);
while (1)
{
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_RESET);
printf("程序正常运行\n");
HAL_Delay(2000);
printf("系统进入停止模式\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);//指示灯灭掉表示睡眠模式
HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI);//进入停止模式
printf("系统已经被唤醒\n");
}
中断函数
CLK_Resume是自己编写的时钟唤醒函数,需要重新设置时钟
void CLK_Resume(void)
{
//使能HSE
__HAL_RCC_HSE_CONFIG(RCC_HSE_ON);//打开HSE
while(__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY) == RESET);
//使能PLL
__HAL_RCC_PLL_ENABLE();//打开PLL
while(__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY) == RESET);
//选择PLL作为时钟源
__HAL_RCC_SYSCLK_CONFIG(RCC_SYSCLKSOURCE_PLLCLK);
while(__HAL_RCC_GET_SYSCLK_SOURCE() != 0x08);
}
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
if(GPIO_Pin == GPIO_PIN_0)
{
CLK_Resume();//恢复系统时钟HSE
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 10);//蜂鸣器响一下
HAL_Delay(200);
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 0);
}
}
STM32实现待机模式
待机模式,它除了关闭所有的时钟,还把1.2V区域的电源也完全关闭了,也就是说,从待机模式唤醒后,由于没有之前代码的运行记录,只能对芯片复位,重新检测boot条件,从头开始执行程序。它有四种唤醒方式,分别是WKUP(PA0)引脚的上升沿,RTC闹钟事件,NRST引脚的复位和IWDG(独立看门狗)复位。
代码实现
HAL_TIM_PWM_Start(&htim10, TIM_CHANNEL_1);
printf("usart is init\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);
while (1)
{
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_RESET);
printf("程序正常运行\n");
HAL_Delay(2000);
printf("系统进入待机模式\n");
HAL_GPIO_WritePin(GPIOF, GPIO_PIN_8, GPIO_PIN_SET);//指示灯灭掉表示睡眠模式
HAL_PWR_EnableWakeUpPin(PWR_WAKEUP_PIN1);//使能唤醒引脚
__HAL_PWR_CLEAR_FLAG(PWR_FLAG_WU);//标志位为1,表示需要去唤醒
HAL_PWR_EnterSTANDBYMode();
printf("系统已经被唤醒\n");
}
中断函数
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
if(GPIO_Pin == GPIO_PIN_0)
{
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 10);//蜂鸣器响一下
HAL_Delay(200);
__HAL_TIM_SetCompare(&htim10, TIM_CHANNEL_1, 0);
}
}
实验现象:待机模式被唤醒后,无法继续执行之前的代码,重新复位从头开始执行程序