STM32-电源管理(实现低功耗)

news2025/1/17 9:05:31

电源管理

STM32 HAL库对电源管理提供了完善的函数和命令。

工作模式(高功耗->低功耗):运行、睡眠、停止、待机。

若备份域电源正常供电,备份域内的RTC都可以正常运行,备份域内的寄存器的数据会被保存,不受功耗模式影响。

上电复位(POR)和掉电复位(PDR)

当检测到VDD的电压低于阈值VPOR及VPDR时,无需外部电路辅助,STM32芯片会自动保持在复位状态,防止因电压不足强行工作而带来严重的后果。

在刚开始电压低于VPOR时(约1.92V),STM32保持在上电复位状态(POR,Power On Reset)。当VDD电压持续上升至大于VPOR时,芯片开始正常运行。

而在芯片开始正常运行的时候,当检测到VDD电压下降至低于VPDR阈值(约1.88V),会进入掉电复位状态(PDR,Power Down Reset)。

配置PVD监控功能

PVD可监控VDD的电压,当它低于阈值时可产生PVD中断以让系统进行紧急处理,这个阈值可以直接使用库函数PWR_PVDLevelConfig配置成某一个的阈值等级。

WFI和WFE命令

进入各种低功耗模式时都需要调用WFI或WFE命令,实质上都是内核指令,在库文件 core_cm3.h 或 cmsis_armcc.h 中把这些指令封装成了函数。

/* 等待中断。是一种暂停执行指令,暂停至任意中断产生后被唤醒 */
#define __WFI                             __wfi

/* 等待事件。是一种暂停执行指令,暂停至任意事件产生后被唤醒 */
#define __WFE                             __wfe

这两个指令,调用后都能进入低功耗模式,需要使用__WFI();和__WFE();来调用(因为__wfi和__wfe是编译器内置的函数,函数内部调用了相对应的汇编指令)。

具体可查《cortex-CM3/CM4权威指南》。

进入停止模式

直接调用WFI和WFE指令可以进入睡眠模式,而进入停止模式这还需要在调用指令前设置一些寄存器位,STM32 HAL库把这部分的操作封装到HAL_PWR_EnterSTOPMode()。

/**
  * @brief 进入停止模式
  * @note  在停止模式下所有I/O都会保持在停止前的状态.
  * @note  当使用中断或唤醒事件退出停止模式时,HSI RC振荡器被选择为系统时钟。
  * @note  当稳压器在低功率模式下工作时,从停止模式唤醒时会产生额外的启动延迟。
  *        通过在停止模式中保持内部稳压器打开,虽然启动时间减少,但消耗更高。
  * @param Regulator: 在停止模式下指定稳压器状态。
  *            @arg PWR_MAINREGULATOR_ON: 稳压器正常运行
  *            @arg PWR_LOWPOWERREGULATOR_ON: 稳压器低功耗运行
  * @param STOPEntry: 指定是否使用WFI或WFE指令进入停止模式。
  *            @arg PWR_STOPENTRY_WFI: 使用WFI指令进入停止模式
  *            @arg PWR_STOPENTRY_WFE: 使用WFE指令进入停止模式
  * @retval None
  */
void HAL_PWR_EnterSTOPMode(uint32_t Regulator, uint8_t STOPEntry)
{
    /* 检查参数 */
    assert_param(IS_PWR_REGULATOR(Regulator));
    assert_param(IS_PWR_STOP_ENTRY(STOPEntry));

    /* 清除PWR寄存器中的PDDS位以指定当CPU进入深度睡眠时进入停止模式 */
    CLEAR_BIT(PWR->CR,  PWR_CR_PDDS);

    /* 根据稳压参数值,通过在PWR寄存器中设置LPDS位来选择稳压模式 */
    MODIFY_REG(PWR->CR, PWR_CR_LPDS, Regulator);

    /* 设置内核系统控制寄存器的SLEEPDEEP位 */
    SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));

    /* 选择停止模式进入 */
    if (STOPEntry == PWR_STOPENTRY_WFI)
    {
        /* 请求等待中断 */
        __WFI();
    }
    else
    {
        /* 请求等待事件 */
        __SEV();
        PWR_OverloadWfe(); /* 本地重新定义WFE */
        PWR_OverloadWfe(); /* 本地重新定义WFE */
    }

    /* 以下的程序是当重新唤醒时才执行的,清除内核系统控制寄存器的SLEEPDEEP位 */
    CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
}

进入停止模式后,STM32的所有I/O都保持在停止前的状态,而当它被唤醒时,STM32使用HSI作为系统时钟(8MHz)运行,由于系统是在会影响很多外设的工作状态,所以一般我们在唤醒后会重新开始HSE,把系统时钟设置成原来的状态。

进入待机模式

STM32 HAL库把这部分的操作封装到HAL_PWR_EnterSTANDBYMode()。

/**
  * @brief 进入待机模式
  * @note  待机模式下,除以下情况外,所有I/O引脚均为高阻抗::
  *          - 复位引脚(仍然有效)
  *          - TAMPER pin if configured for tamper or calibration out.
  *          - WKUP pin (PA0) (如果使能WKUP唤醒功能).
  * @retval None
  */
void HAL_PWR_EnterSTANDBYMode(void)
{
    /* 选择待机模式 */
    SET_BIT(PWR->CR, PWR_CR_PDDS);

    /* 设置内核系统控制寄存器的SLEEPDEEP位*/
    SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));

    /* 存储操作完毕时才能进入待机模式,使用以下语句确保存储操作执行完毕 */
#if defined ( __CC_ARM)
    __force_stores();
#endif
    /* 请求等待中断 */
    __WFI();
}

待机模式也可以使用WFE指令进入的,如果有需要可以自行修改。

在进入待机模式后,除了被使能用于唤醒的I/O,其余I/O都进入高阻态,而从待机模式唤醒后,想防御复位STM32芯片,程序重新从头开始执行。

实验环节1:PWR_PVD监控

实验操作

使用外部可调电源,调节成5V输出,连接到开发板5V和GND排针给板子进行供电;

复位开发板,电压正常时LED为绿色;

向下调节可调电源的电压,大约降到4V时,LED为红色。(程序中控制PVD监控电压约为2.8V,当5V降到4V时,连接STM32的VDD电源会降于2.8V,产生PVD事件,在中断中控制亮红灯)。

注意:其他电源线都拔掉(包括下载器、USB线)。不能远高于5V而导致烧坏开发板。

PVD配置

void PVD_Config(void)
{
	PWR_PVDTypeDef sConfigPVD;
	
	/*使能 PWR 时钟 */
	__HAL_RCC_PWR_CLK_ENABLE();
	
	/* 配置 PVD 中断 */
	HAL_NVIC_SetPriority(PVD_IRQn, 0 ,0);
	HAL_NVIC_EnableIRQ(PVD_IRQn);  

	/* 配置PVD级别6 (PVD检测电压的阈值为2.8V,VDD电压低于2.8V时产生PVD中断,
	   具体数据可查询数据手册获知) 具体级别根据自己的实际应用要求配置*/
	sConfigPVD.PVDLevel = PWR_PVDLEVEL_6;
	sConfigPVD.Mode 	= PWR_PVD_MODE_IT_RISING_FALLING;
	HAL_PWR_ConfigPVD(&sConfigPVD);
	
	/* 使能PVD输出 */
	HAL_PWR_EnablePVD();
}

测试环节

void PVD_IRQHandler(void)
{
	HAL_PWR_PVD_IRQHandler();
}

void HAL_PWR_PVDCallback(void)
{
	LED红灯
}

void test(void)
{
	初始化
	LED绿灯

	// 配置PVD,当电压过低时,会进入中断服务函数,亮红灯
	PVD_Config();
	
	while(1)
	{}
}

实验环节2:PWR睡眠模式

实验操作

LED:绿灯正常运行,红灯睡眠状态,蓝灯刚被唤醒。

KEY:key1和key2配置成IO中断模式。

运行一段时间后自动进入睡眠时间,通过按键(key1或key2)唤醒。

睡眠状态下,DAP下载器无法给STM32下载程序,可唤醒后再下载或按复位键使芯片处于复位状态下下载后松开复位键。

测试环节

int main(void)
{
	初始化

    while (1)
    {
        LED绿灯
        HAL_Delay(2000);

        LED红灯
        HAL_SuspendTick();	//暂停滴答时钟,防止通过滴答时钟中断唤醒
        //进入睡眠模式
        HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);
		
        //等待中断唤醒  K1或K2按键中断
		
        LED蓝灯
        HAL_ResumeTick();	//被唤醒后,恢复滴答时钟
        HAL_Delay(2000);
		
        //继续执行while循环
    }
}

实验环节3:PWR待机模式

实验操作

LED:绿灯表示本次复位是上电或引脚复位,红灯待机状态,蓝灯刚被唤醒。

KEY:key2配置成输入模式。

长按KEY2按键会进入待机模式,待机模式下KEY1按键可唤醒,唤醒后系统会复位。可通过检测PWR_CSR:WUF标志确定复位来源。

待机模式下,DAP下载器无法给STM32下载程序,可唤醒后再下载。

注意:由于WKUP引脚(PA0)必须使用上升沿才能唤醒待机状态的系统,所以硬件设计PA0引脚连接到KEY1,且按下KEY1时会在PA0引脚产生上升沿,从而可实现唤醒的功能。

测试环节

/**
  * @brief  用于检测按键是否被长时间按下
  * @param  无
  * @retval 1 :按键被长时间按下  0 :按键没有被长时间按下
  */
static uint8_t KEY2_LongPress(void)
{
    uint8_t downCnt = 0;		//记录按下的次数
    uint8_t upCnt = 0;			//记录松开的次数
	
    while (1)					//死循环,由return结束
    {	
        HAL_Delay(20);			//延迟一段时间再检测

        if (HAL_GPIO_ReadPin(KEY2_GPIO_PORT, KEY2_PIN) == SET)	//检测到按下按键
        {
            downCnt++;			//记录按下次数
            upCnt = 0;			//清除按键释放记录

            if (downCnt >= 100)	//按下时间足够
            {
                return 1; 		//检测到按键被时间长按下
            }
        }
        else
        {
            upCnt++; 			//记录释放次数

            if (upCnt > 5)		//连续检测到释放超过5次
            {
                return 0;		//按下时间太短,不是按键长按操作
            }
        }
    }
}

int main(void)
{
	初始化

    /* 使能电源管理单元的时钟,必须要使能时钟才能进入待机模式 */
	__HAL_RCC_PWR_CLK_ENABLE();

    //检测复位来源
    if (__HAL_PWR_GET_FLAG(PWR_FLAG_WU) == SET)
    {	// 复位前为待机模式
        __HAL_PWR_CLEAR_FLAG(PWR_FLAG_WU);
        LED蓝灯
    }
    else
    {
		// 复位前为正常运行
        LED绿灯
    }

    while (1)
    {
        // K2 按键长按进入待机模式
        if (KEY2_LongPress())
        {
            LED红灯
            HAL_Delay(1000);

            /*清除WU状态位*/
            __HAL_PWR_CLEAR_FLAG(PWR_FLAG_WU);

            /* 使能WKUP引脚的唤醒功能,使能PA0*/
            HAL_PWR_EnableWakeUpPin(0x00000100U);

            //暂停滴答时钟,防止通过滴答时钟中断唤醒
            HAL_SuspendTick();
			
            /* 进入待机模式 */
            HAL_PWR_EnterSTANDBYMode();
        }
    }
}

实验现象

开机正常运行绿灯。长按KEY2按键,显红灯,过1s后进入待机模式LED灭。按下KEY1按键退出待机模式自动复位,显蓝灯。按下复位键,重新运行绿灯。

实验环节4:PWR停止模式

实验操作

LED:绿灯正常运行,红灯停止状态,蓝灯刚被唤醒。

KEY:key1和key2配置成IO中断模式。

运行一段时间后自动进入停止时间,通过按键(key1或key2)唤醒。

待机模式下,DAP下载器无法给STM32下载程序,可唤醒后再下载。

注意:由于WKUP引脚(PA0)必须使用上升沿才能唤醒待机状态的系统,所以硬件设计PA0引脚连接到KEY1,且按下KEY1时会在PA0引脚产生上升沿,从而可实现唤醒的功能。

测试环节

/**
  * @brief  从停止模式唤醒后配置系统时钟:启用HSE、PLL并选择PLL作为系统时钟源。

  * @param  无
  * @retval 无
  */
static void SYSCLKConfig_STOP(void)
{
	RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
	RCC_OscInitTypeDef RCC_OscInitStruct = {0};
	uint32_t pFLatency = 0;

	/* 启用电源控制时钟 */
	__HAL_RCC_PWR_CLK_ENABLE();

	/* 根据内部RCC寄存器获取振荡器配置 */
	HAL_RCC_GetOscConfig(&RCC_OscInitStruct);

	/* 从停止模式唤醒后重新配置系统时钟: 启用HSE和PLL */
	RCC_OscInitStruct.OscillatorType  = RCC_OSCILLATORTYPE_HSE;
	RCC_OscInitStruct.HSEState        = RCC_HSE_ON;
	RCC_OscInitStruct.PLL.PLLState    = RCC_PLL_ON;
	if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
	{
		while(1) { ; }
	}

	/* 根据内部RCC寄存器获取时钟配置 */
	HAL_RCC_GetClockConfig(&RCC_ClkInitStruct, &pFLatency);

	/* 选择 PLL 作为系统时钟源, 并配置 HCLK、PCLK1 和 PCLK2时钟分频系数 */
	RCC_ClkInitStruct.ClockType     = RCC_CLOCKTYPE_SYSCLK;
	RCC_ClkInitStruct.SYSCLKSource  = RCC_SYSCLKSOURCE_PLLCLK;
	if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, pFLatency) != HAL_OK)
	{
		while(1) { ; }
	}
}

int main(void) 
{
	uint32_t 	SYSCLK_Frequency = 0; 
	uint32_t 	HCLK_Frequency = 0;
	uint32_t 	PCLK1_Frequency = 0;
	uint32_t 	PCLK2_Frequency = 0;
	uint32_t 	SYSCLK_Source = 0;
	
	初始化

	while(1)
	{	
		LED绿灯
		HAL_Delay(2000);		

		// 进入停止模式,亮红灯,按KEY1或KEY2按键可唤醒
		LED_RED;
		HAL_SuspendTick();			//暂停滴答时钟,防止通过滴答时钟中断唤醒
		/* 进入停止模式,设置电压调节器为低功耗模式,等待中断唤醒 */
		HAL_PWR_EnterSTOPMode(PWR_MAINREGULATOR_ON, PWR_STOPENTRY_WFI);
		
		// 等待中断唤醒  K1或K2按键中断	
		
		// 被唤醒,亮蓝灯指示
		LED蓝灯
		SystemCoreClockUpdate();	//根据时钟寄存器的值更新SystemCoreClock变量
		//获取唤醒后的时钟状态	
		SYSCLK_Frequency = HAL_RCC_GetSysClockFreq(); 
		HCLK_Frequency   = HAL_RCC_GetHCLKFreq();
		PCLK1_Frequency  = HAL_RCC_GetPCLK1Freq();
		PCLK2_Frequency  = HAL_RCC_GetPCLK2Freq();
		SYSCLK_Source    = __HAL_RCC_GET_SYSCLK_SOURCE();
		 
		/* 从停止模式唤醒后配置系统时钟:启用HSE、PLL*/
		/* 选择PLL作为系统时钟源(HSE和PLL在停止模式下被禁用)*/
		SYSCLKConfig_STOP();
		
		HAL_ResumeTick();			//被唤醒后,恢复滴答时钟
		//获取重新配置后的时钟状态
		SYSCLK_Frequency = HAL_RCC_GetSysClockFreq(); 
		HCLK_Frequency   = HAL_RCC_GetHCLKFreq();
		PCLK1_Frequency  = HAL_RCC_GetPCLK1Freq();
		PCLK2_Frequency  = HAL_RCC_GetPCLK2Freq();
		SYSCLK_Source    = __HAL_RCC_GET_SYSCLK_SOURCE();
		
		HAL_Delay(2000);	

		//继续执行while循环
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1154196.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

制作一个可以arm架构下运行的docker镜像(for Python)

看完本篇文章,你将得到一个可以arm架构下运行的python 基础镜像。 题外话 这里直接说docker镜像有点儿草率,因为目前很多容器都是Podman了。 podman的介绍 arm和aarch傻傻分不清楚 现在这两个是一样的意思了。 arm64和aarch64之间的区别 开始制作镜…

笔记48:51序列模型--课程笔记

本地笔记地址:D:\work_file\DeepLearning_Learning\03_个人笔记\3.循环神经网络\序列模型 a a a a a a a a a a a a a a a a a a 8.1. 序列模型 — 动手学深度学习 2.0.0 documentation

双十一真香数码好物推荐!相信总有一款适合你!

在这个数字化时代,科技产品已经渗透到我们的日常生活,成为不可或缺的一部分。随着电商平台不断举办各种促销活动,双11购物狂欢节已经成为备受瞩目的盛事。在这场购物狂欢中,数码产品是热门之选。我们都期望在这个特殊的时刻找到高…

微信小程序 如何订阅消息?

微信公众号 微信公众号分为两种类型:订阅号和服务号,它们在功能和使用方式上有一些区别。 1.订阅号(Subscription Account): 主要用于向用户提供信息、资讯、新闻等更新性内容。每天可以发送1条群发消息给所有关注者…

墨者学院 身份认证失效漏洞实战

一、题目信息 二、漏洞利用 1.通过抓包,抓取test登录信息 发现card_id号以及一些回显的账号密码信息 搜索了一下这个id,发现是测试的头像 2.修改id号 分析请求包的逻辑,发现是请求了头像资源后再去请求头像id的详情包,所以根据…

el-dropdown自定义样式,不影响其他组件

原来的样式: 修改后的样式: 给el-dropdown-menu添加类名dropdown-menu <el-dropdown-menu slot"dropdown" class"dropdown-menu"><router-link to"/user/profile"><el-dro…

mysql出现Deadlock死锁排查

参考: https://www.yisu.com/zixun/664543.html 查看mysql死锁日志 show engine innodb status找到信息中LATEST DETECTED DEADLOCK这一行&#xff0c;可以看到mysql的死锁信息详情 ------------------------ LATEST DETECTED DEADLOCK ------------------------ 2023-10-31…

VS LiveShare使用操作介绍

VS LiveShare的使用教程 文章简介下载过程 文章简介 本篇文章主要介绍了如何安装和使用LiveShare的过程。 下载过程 1.在扩展->管理扩展&#xff0c;搜索Live Share后&#xff0c;下载对应的安装包&#xff0c;安装后对VS进行重启 2.安装后界面右上角会出现Live Share标…

酷克数据出席永洪科技用户大会 携手驱动商业智能升级

10月27日&#xff0c;第7届永洪科技全国用户大会在北京召开。酷克数据作为国内云原生数仓代表企业&#xff0c;受邀出席本次大会&#xff0c;全面展示了云数仓领域最新前沿技术&#xff0c;并进行主题演讲。 携手合作 助力企业释放数据价值 数据仓库是商业智能&#xff08;BI…

什么是接口自动化测试?接口自动化测试的目的是什么?

1、什么是接口测试 接口测试是对系统或组件之间的接口的测试。主要用于检测外部系统与系统间以及内部各个子系统间的交互点。测试重点是检查数据交换、传递和控制管理过程&#xff0c;以及系统间的相互逻辑依赖关系等。 2、接口测试的目的 1> 尽早介入软件测试流程&#…

ubuntu(18.04) 安装 blast 并在php中调用

1、下载 https://ftp.ncbi.nlm.nih.gov/blast/executables/blast/LATEST/2、解压&#xff0c;配置环境变量 tar zvxf ncbi-blast-2.14.1-x64-linux.tar.gz解压后改名为 blast 配置环境变量&#xff0c;可以不配置 使用的时候直接绝对路径使用&#xff08;本次使用绝对路径&am…

我用好说 AI 做二次元人设

你有没有想过自己做一部原创作品&#xff1f; 就像开发《星露谷物语》那样&#xff0c;自己把控作品的 角色、故事、载体、宣传 等方方面面&#xff0c;让 idea 不再只是灵光一闪。 以前是 “万事开头难”&#xff0c;可能第一步都举步维艰。但现在有了 AI 就不同了&#xff…

iOS 系统获取 Bundle ID、平台公钥、签名 MD5 值的指引

1. 获取 Bundle ID&#xff1a;使用 APP 对应的 IOS 开发者账号登录 Developer 控制台&#xff0c;找到下图标识符&#xff08;英文&#xff09;&#xff0c;单击进入 Certificates,Identifiers&Profiles 页面。 2. 在 Certificates,Identifiers&Profiles 页面&#xf…

Playwright已经是目前最好的测试自动化工具了吗?

作者观点&#xff1a;很长时间以来&#xff0c;Selenium是QA工程师寻求测试自动化解决方案的首选测试框架。它能够测试任何浏览器&#xff08;这在IE浏览器的统治时期尤其重要&#xff09;和任何平台。然而&#xff0c;现在看来&#xff0c;那个时代已经过去了。 今天&#xf…

Origin科研绘图与学术图表绘制从入门到精通

目录 前言 1. PDF基线的绘制 2. XRD图的绘制 3. 点线图的绘制 4. 如何快速掌握Orign 前言 Hello小伙伴好&#xff0c;最近在忙于三篇小论文&#xff0c;在大量的处理数据和画图中。对于一篇好的学术论文图是不可缺少的&#xff0c;可以说如果你的数据平平无奇&#xff…

【JavaEE初阶】 认识文件与Java中操作文件

文章目录 &#x1f334;认识文件&#x1f6a9;树型结构组织和目录&#x1f6a9;文件路径&#xff08;Path&#xff09;&#x1f6a9;知识扩展 &#x1f38d;Java 中操作文件&#x1f6a9;File 概述&#x1f4cc;属性&#x1f4cc;构造方法&#x1f4cc;方法 &#x1f6a9;File使…

LIS检验科信息管理系统源码 标本核收、采集检验结果、书写检验报告、质量控制

LIS检验科信息管理系统源码 LIS是采用软件解决数据接收方式的医疗系统之一&#xff0c;强化了病人从开检验单检验&#xff0c;到最后检验报告的各项步骤&#xff0c;更替代了以前的缓慢、不可靠的纯手工数据接收方式&#xff0c;使得医疗检验的步骤更有条理、效率更高、检验结果…

一致性哈希揭秘,深入解析其工作原理

前言 在进行一致性哈希介绍前&#xff0c;先思考2个问题&#xff1a; 什么是Hash一致性Hash和Hash的关系是什么 对于第一个问题Hash的定义 Hash也成散列&#xff0c;基本原理就是把任意长度的输入&#xff0c;通过hash算法变成固定长度的输出。 对于第二个问题&#xff0c…

关于 MapboxGL 在 Vue 中的简单使用

前言问题 关于我为什么使用了 在线的 js引入方法&#xff0c;而不是使用 npm 直接下载依赖问题&#xff0c;之前有一篇文章讲过原因&#xff1a;关于 Vue-iClient-MapboxGL 的使用注意事项 网上提供的 vue-iclient-mapboxgl 比较多&#xff0c;但是我这里使用的是 iclient-su…

win11系统msvcp120.dll丢失的解决方法,亲测有效的详细方法

在计算机使用过程中&#xff0c;我们常常会遇到一些错误提示&#xff0c;其中之一就是“msvcp120.dll丢失”这个错误通常会导致某些应用程序无法正常运行。为了解决这个问题&#xff0c;我们需要采取一些修复措施。本文将介绍五个修复msvcp120.dll丢失的方法&#xff0c;帮助大…