待机模式中唤醒

代码源自正点原子
文档源自STM32F1XX参考手册中文版
参考[https://www.cnblogs.com/alvis-jing/p/3712106.html#!comments]

一、理论知识

待机模式可实现系统的最低功耗。该模式是在Cortex-M3深睡眠模式时关闭电压调节器。整个1.8V供电区域被断电。 PLL、 HSI和HSE振荡器也被断电。 SRAM和寄存器内容丢失。只有备份的寄存器和待机电路维持供电。
从待机模式唤醒后的代码执行等同于复位后的执行
在这里插入图片描述
进入待机模式用库函数写代码流程:
1、开启电源时钟
因为要配置电源控制寄存器,所以必须先使能电源时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE); //使能 PWR 外设时钟
2、设置 WK_UP 引脚作为唤醒源
PWR_WakeUpPinCmd(ENABLE); //使能唤醒管脚功能
3、进入待机模式
设置 SLEEPDEEP 位, 设置 PDDS 位,执行 WFI 指令,进入待机模式。
void PWR_EnterSTANDBYMode(void);
在这里插入图片描述
在这里插入图片描述

二、代码

void Sys_Stanby(void)
{
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);//使能PWR外设时钟
	PWR_WakeUpPinCmd(ENABLE); //使能唤醒管脚功能
    PWR_EnterSTANDBYMode();//设置 SLEEPDEEP 位,设置 PDDS 位,执行 WFI 指令,进入待机模式
}

/*进入待机模式*/
void Sys_Enter_Standby(void)
{
	RCC_APB2PeriphResetCmd(0X01FC,DISABLE);//在 void Sys_Enter_Standby(void)函数里面,我们要在进入待机模式前把所有开启的外设全部关闭,我们这里仅仅复位了所有的 IO 口,使得 IO 口全部为浮空输入
	Sys_Stanby();
}

/*
检测WKUP脚的信号
返回值1:连续按下3s以上
	  0:误触发
*/
unsigned char Check_WKUP(void)
{
	unsigned char t=0;//记录按下时间
	GPIO_WriteBit(GPIOB,GPIO_Pin_5,Bit_RESET);
	while(1)
	{
		if(WKUP_KD)//按下
		{
			t++;
			delay_ms(30);
			if(t>=100)//超过3s
			{
				GPIO_WriteBit(GPIOB,GPIO_Pin_5,Bit_RESET);
				return 1;
			}
		}
		else
		{
			GPIO_WriteBit(GPIOB,GPIO_Pin_5,Bit_SET);
			return 0;
		}
	}
}

void WKUP_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	EXTI_InitTypeDef EXTI_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO,ENABLE);//使能GPIOA和复用功能时钟
	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;//上拉输入
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//PA0
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);

	//使用外部中断方式
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource0);
	
	EXTI_InitStructure.EXTI_Line = EXTI_Line0;//设置按键所有的外部线路
	EXTI_InitStructure.EXTI_LineCmd = ENABLE ;//外部中断模式
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt ;//上升沿触发
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;
	EXTI_Init(&EXTI_InitStructure);	// 初始化外部中断

	
	NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;//使能按键所在的外部中断通道
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能外部中断通道
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//先占优先级2级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;//从优先级2级
	NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器

	if(Check_WKUP()==0)  /*如果没有长按,则Check_WKUP()=0,,Sys_Enter_Standby 函数,直接进入待机模式了。如果长按则返回1那就会跳过Sys_Enter_Standby就会直接开机*/
	{
		Sys_Enter_Standby();//进入待机模式
	}
}

/*中断,检测到PA0脚的一个上升沿.	 
中断线0线上的中断检测 关机
待机模式下中断服务函数不会运行,因为MCU不工作
*/
void EXTI0_IRQHandler(void)
{
	EXTI_ClearITPendingBit(EXTI_Line0);// 清除LINE10上的中断标志位	
	if(Check_WKUP()==1)//关机
	{
		Sys_Enter_Standby();
	}
}

int main(void) //部分代码
{
	LED_Init();		
	WKUP_Init();
	while(1)
	{
		led灯闪烁
	}
}

在 void Sys_Enter_Standby(void)函数里面,我们要在进入待机模式前把所有开启的外设全部关闭,我们这里仅仅复位了所有的 IO 口,使得 IO 口全部为浮空输入。RCC_APB2PeriphResetCmd(0X01FC,DISABLE); 0x01FC就是000111111100就是操作所有IO口。
在这里插入图片描述
在main函数里,WKUP_Init()和led_Init()初始化。
下载代码后,如果不按WKUP按键3s,则会在WKUP_Init()里运行到if(Check_WKUP()==0)满足的时候进入待机模式。
在待机模式下,长按WKUP,由于我们PWR_WakeUpPinCmd(ENABLE); 所以按下WKUP触发到WKUP引脚的上升沿,会唤醒CPU,而唤醒后的代码执行等同于复位后的执行,所以代码从上往下跑,又来到了WKUP_Init()这时如果我们是长按的话if(Check_WKUP()==0)不满足,就不会进入待机模式,那就是正常运行模式惹!

在正常运行模式下,按下WKUP会触发中断服务函数,如果是长按,则在EXTI0_IRQHandler()里满足if(Check_WKUP()==1)进入待机模式。

所以现象是:下载代码之后,如果啥都不干,那啥反应都没有。长按WKUP之后,有一个灯会闪,再长按WKUP,进入待机模式。

/分割线****/
一个疑问是为啥这个WKUP知道现在长按要关机了待会知道长按要开机了?

1 、如果是正常运行,按下WKUP,先执行中断程序,判断是否3秒,决定是否待机。
2 、如果是待机状态,按下WKUP,则先复位并初始化,判断是否3秒,决定是否开机。

因为在待机模式下,不会执行中断服务函数。从待机模式切换到正常运行模式 按住wkup键的那一瞬间会有一个上升沿,而这个上升沿执行了唤醒功能。但是没有执行外部中断0的中断服务函数,因为待机模式下cpu是不工作的。从待机模式唤醒后的代码执行等同于复位后的执行所以程序又会从头开始执行,然后又会执行到 Check_up()函数检测,如果按住按键的时间没有超过3s ,还是会处于待机模式。
所以结论就是外部中断0的中断服务程序在待机模式切换到正常运行模式的时候从始至终是不会运行的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32待机模式是一种低功耗模式,可以帮助节省能源并延长电池寿命。在待机模式下,主要的系统时钟会被关闭,但是一些特定的外设和断仍然可以唤醒处理器运行。 其,RTC(实时时钟)模块可以在待机模式下使用来唤醒STM32处理器。RTC模块有多种功能,其之一就是可以配置为定时器模式,在指定的时间唤醒处理器。 为了使用RTC唤醒STM32处理器,首先需要进行一些初始化操作。这包括设置RTC时钟和配置RTC断。 设置RTC时钟:通过使能外设电源时钟和RTC时钟,我们可以启用RTC模块并使其工作。 配置RTC断:在初始化过程,我们可以配置RTC模块的断源和优先级,以及RTC唤醒断使能。这样,在预定的时间到达时,RTC模块会发出一个断信号,从而唤醒处理器。 在配置好RTC模块后,我们可以进入待机模式。在待机模式下,处理器会暂停运行,只有RTC模块会继续工作。 当预定的时间到达时,RTC模块会触发一个断信号,处理器会从待机模式唤醒,并执行相应的断服务函数(ISR)。在断服务函数,我们可以进行一些必要的操作,例如处理数据、更新状态等。 总结来说,通过配置RTC模块和设置相应的断,我们可以在待机模式下使用RTC来唤醒STM32处理器。这可以大大降低系统功耗,并延长电池寿命,适用于一些对低功耗要求较高的应用场景。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值