STM32F4外设配置速查【低功耗唤醒部分】

低功耗模式

一般在系统或电源复位后,mcu在运行状态下由HCLK为CPU提供时钟,内核执行代码,当CPU不需要运行时可利用多种低功耗模式来节省功耗,等待某事件触发时才唤醒

stm32f4xx有三种低功耗模式

睡眠模式仅内核停止,外设如NVIC、systick等仍运行
停止模式所有时钟停止,1.8V内核电源工作,备份寄存器、待机电路等都有供电,寄存器、SRAM数据保留
待机模式1.8V内核电源关闭,仅有备份寄存器和待机电路维持供电,寄存器、SRAM清空,功耗最低

运行模式下,也可通过降低系统时钟关闭未被使用的外设时钟来降低功耗

调压器

嵌入式线性调压器为除备份域和待机电路外所有数字电路供电,需要连接两个外部电容到专用引脚 V C A P _ 1 V_{CAP\_1} VCAP_1 V C A P _ 2 V_{CAP\_2} VCAP_2为激活和停用调压器,需要将特定引脚连接到VSS或VDD,具体引脚由封装决定

低功耗模式开启方法

模式进入唤醒对1.2V域时钟的影响对VDD域时钟的影响调压器
睡眠WFI/WFE任意中断/唤醒事件CPU CLK关闭,其他无影响开启
停止PDDS位和LPDS位+SLEEPDEEP位+WFI/WFE任意EXTI线(可在EXTI寄存器中配置,包括内部线和外部线)全部关闭HSI、HSE关闭开启或处于低功耗模式
待机PDDS位+SLEEPDEEP位+WFI/WFEWKUP引脚上升沿、RTC闹钟(A或B)、RTC唤醒事件、RTC入侵事件、RTC时间戳事件、NRST引脚外部复位、IWDG复位全部关闭HSI、HSE关闭关闭

WFI置位可用任意中断唤醒

WFE置位可用唤醒事件唤醒

理想状态下,待机模式只需要2.2uA电流,典型电流为350uA

stm32f4的特别说明

使能了RTC闹钟中断或RTC周期性唤醒等中断时,进入待机模式前,必须进行以下处理:

  1. 禁止RTC中断
  2. 清零对应中断标志位
  3. 清除PWR唤醒(WUF)标志(通过设置PWR_CR->CWUF位实现)
  4. 重新使能RTC对应中断
  5. 进入低功耗模式

详情参考stm32f4xx芯片手册

stm32f4xx的待机模式启用-唤醒配置

注意:使用前需引入stm32f4xx_pwr.c库文件

进入待机配置步骤:

  1. 使能电源时钟
  2. 关闭RTC相关中断
  3. 设置WK_UP引脚为唤醒源
  4. 设置SLEEPDEEP位、PDDS位、执行WFI指令,进入待机模式

wkup.h

#ifndef __WKUP_H
	#define __WKUP_H
	#include "sys.h"
	
	u8 Check_WKUP(void);//检测WKUP脚信号
	void WKUP_enter_standbymode(void);//系统进入待机模式
	void WKUP_init(void);//待机唤醒初始化
	
#endif

wkup.c

#include "wkup.h"
#include "delay.h"

//检测WKUP脚信号
//返回1:连续按下3s以上;返回0:错误的触发(连按3s以下)
u8 Check_WKUP(void)
{
	u8 t=0;
	u8 tx=0;//记录松开的次数

	while(1)
	{
		if(WKUP_KD)//WKUP已经按下
		{
			t++;
			tx=0;
		}
		else
		{
			tx++;
			if(tx>3)//超过90ms内没有WKUP信号
				return 0;
		}
		
		delay_ms(30);
		
		if(t>=100)//按下超过3s
			return 1;
	}
}
/*此函数效率较低,可使用定时器来重写*/

//系统进入待机模式
void WKUP_enter_standbymode(void)
{
	while(WKUP_KD);//等待WKUP按键松开(在有RTC中断时,必须等WKUP松开再进入待机)

	RCC_AHB1PeriphResetCmd(0x04FF,ENABLE);//复位所有GPIO口
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);//使能电源配置时钟
	PWR_BackupAccessCmd(ENABLE);//使能备份域访问

	RTC_ITConfig(RTC_IT_TS|RTC_IT_WUT|RTC_IT_ALRA|RTC_IT_ALRB,DISABLE);//关闭RTC相关中断
	RTC_ClearITPendingBit(RTC_IT_TS|RTC_IT_WUT|RTC_IT_ALRA|RTC_IT_ALRB);//清除中断标志位
	
	PWR_ClearFlag(PWR_FLAG_WU);//清除WKUP标志
	
	PWR_WakeUpPinCmd(ENABLE);//使能待机唤醒功能
	PWR_EnterSTANDBYMode();//进入待机模式
}

//以PA0为待机唤醒引脚,进行WKUP唤醒初始化
void WKUP_init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	EXTI_InitTypeDef EXTI_InitStructure;
	
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);//使能GPIOA时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG,ENABLE);//使能SYSCFG时钟

	GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN;//输入模式
	GPIO_InitStructure.GPIO_OType=GPIO_OType_OD;//开漏输出模式
	GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;//PA0
	GPIO_InitStructure.GPIO_PuPd=GPIO_PuPd_DOWN;//内部下拉
	GPIO_InitStructure.GPIO_Speed=GPIO_Speed_100MHz;//100MHz
	//应用设置
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	if(Check_WKUP()==0)//查看是否正常开机
		WKUP_enter_standbymode();//未开机则进入待机模式
	
	SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOA,EXTI_PinSource0);//PA0连接到中断线0
	
	EXTI_InitStructure.EXTI_Line=EXTI_Line0;//中断线0
	EXTI_InitStructure.EXTI_Mode=EXTI_Mode_Interrupt;//中断事件
	EXTI_InitStructure.EXTI_Trigger=EXTI_Trigger_Rising;//上升沿触发
	EXTI_InitStructure.EXTI_LineCmd=ENABLE;//使能Line0
	//应用设置
	EXTI_Init(&EXTI_InitStructure);
	
	NVIC_InitStructure.NVIC_IRQChannel=EXTI0_IRQn;//外部中断0
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0x02;//抢占优先级2
	NVIC_InitStructure.NVIC_IRQChannelSubPriority=0x02;//子优先级2
	NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;//使能外部中断通道
	//应用设置
	NVIC_Init(&NVIC_InitStructure);
}

//中断线0-PA0中断服务函数
//上升沿触发,用于正常运行状态进入待机模式
void EXTI0_IRQHandler(void)
{
	EXTI_ClearITPendingBit(EXTI_Line0);//清除中断标志位
	if(Check_WKUP())//判断是否有待机按键按下
		WKUP_enter_standbymode();//进入待机模式
}

main.c

此程序为待机唤醒的演示

如果用了上面的“长按3s进入待机模式”代码,则main文件内应当放入日常执行的程序

int main(void)
{
	while(1)
	{
		if(KEY_scan(0)==KEY0_PRES)//如果KEY0按下,进入待机模式
		{
			RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);//使能电源配置时钟
			PWR_BackupAccessCmd(ENABLE);//使能备份域时钟
			
			RTC_ITConfig(RTC_IT_TS|RTC_IT_WUT|RTC_IT_ALRA|RTC_IT_ALRB,DISABLE);//关闭RTC相关中断
			RTC_ClearITPendingBit(RTC_IT_TS|RTC_IT_WUT|RTC_IT_ALRA|RTC_IT_ALRB);//清除中断标志位
			
			PWR_WakeUpPinCmd(ENABLE);//使能待机唤醒功能
			PWR_EnterSTANDBYMode();
		}
	}
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1设计要求 要求系统按如下方式进入和退出睡眠模式: 在系统启动2秒后,将RTC在3秒钟之后配置为产生一个报警事件,接着通过WFI指令使系统进入停机模式。 如果要唤醒系统到正常模式,可通过按Key按钮;否则,在3秒钟后,会产生RTC报警中断自动将系统唤醒。 一旦退出停机模式,系统时钟被配置成先前的状态(在停机模式下,外部高速振荡器HSE和PLL是不可用的)。 经过一段延时之后,系统将再次进入停机状态,并可按上述操作无限重复。 2 硬件电路设计 硬件电路采用与7.1小节应用实例一样硬件电路,可见图7-10。其中Key按钮用于通过PB9产生一个外部中断, LED1、LED2、LED3、LED4则用于显示处理器所处的模式和中断触发情况。 3 软件程序设计 根据任务要求,程序内容主要包括: (1) 配置GPIOB口,配置RTC,配置外部中断; (2) 配置PB口第9个引脚作为外部中断,下降延触发;配置RTC报警中断,上升沿触发; (3) 两个中断服务子程序的内容分别是:切换LED2和LED3灯的状态; 整个工程包含3个源文件:STM32F10x.s、stm32f10x_it.c和main.c,其中STM32F10x.s为启动代码,所有中断 服务子程序均在stm32f10x_it.c中,其它函数则在main.c中。下面分别介绍相关的函数,具体程序清单见参考程序。 函数SYSCLKConfig_STOP用于当处理器从停机模式唤醒之后,配置系统时钟、使能HSE和PLL,并以PLL作为系统时钟源。当处理器处理停机模式的时候,HSE、PLL是不可用的。 函数GPIO_Configuration用于配置GPIO的PC6、PC7、PC8、PC9和PB9。 函数EXTI_Configuration用于配置外部中断线9(PB9)和17(RTC报警)。 函数NVIC_Configuration配置NVIC及中断向量表,这里主要是配置外部中断线9和17。 函数EXTI9_5_IRQHandler处理按钮Key(PB9)所触发的中断,其主要作用是将LED2灯的状态翻转一次。 函数RTCAlarm_IRQHandler处理RTC报警所触发的中断,其主要作用事将LED3 灯的状态翻转一次,如果设置了唤醒标志则清除之。 运行过程: (1) 使用Keil uVision3 通过ULINK 2仿真器连接EduKit-M3实验平台,打开实验例程目录PWR_TEST子目录下的PWR.Uv2 例程,编译链接工程; (2) 选择软件调试模式,点击MDK 的Debug菜单,选择Start/Stop Debug Session项或Ctrl+F5键,在逻 辑分析仪中添加GPIOC_ODR.6、GPIOC_ODR.7、GPIOC_ODR.8、GPIOC_ODR.9,点击Run按钮即可在逻辑分析 仪中看到如图7-14,还可用Peripherals-General Port-GPIOB来模拟KEY按钮的动作; (3) 选择硬件调试模式,选择Start/Stop Debug Session项或Ctrl+F5键,下载程序并运行,观察LED灯 的变化情况。注意,当目标系统进入停机模式之后,将无法使用仿真器进行调试了; (4) 退出Debug模式,打开Flash菜单>Download,将程序下载到EduKit-M3实验平台的Flash中,按RESET键复位,观察 LED灯的情况,正常情况应为:系统处于运行模式时LED1亮、LED4灭;系统处于停机状态时LED1灭、LED4亮; 当按下KEY按钮时LED2灯状态发生反转;当发生RTC报警时LED3状态发生反转。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值