STM32F1开发指南笔记24----待机唤醒实验

待机模式下SRAM和寄存器的数据都丢失了,因此,待机模式唤醒后,程序将从头开始执行,类似于复位操作。

STM32F1开发指南笔记24----待机唤醒实验_第1张图片

STM32F1开发指南笔记24----待机唤醒实验_第2张图片

STM32F1开发指南笔记24----待机唤醒实验_第3张图片

STM32F1开发指南笔记24----待机唤醒实验_第4张图片

STM32F1开发指南笔记24----待机唤醒实验_第5张图片

STM32F1开发指南笔记24----待机唤醒实验_第6张图片

STM32F1开发指南笔记24----待机唤醒实验_第7张图片

代码流程

系统上电时默认进入待机模式,程序卡在唤醒键初始化。当按住唤醒键后系统会重新从main函数开始执行程序,当再次判断Check_WKUP()函数的返回值时,如果按住3S则返回1,不会再进入待机模式,而是进入了正常模式,接着向下执行程序。

正常模式->待机模式:(先执行中断服务函数)
正常模式下,当再次按住唤醒键时,会触发外部中断,中断服务函数中:如果按住没超过3S,则直接清中断标志然后退出。如果超过3S,则调用Sys_Enter_Standby()函数,再次进入待机模式。

待机模式->正常模式:(先执行复位操作)
待机模式下,按住唤醒键,相当于复位操作,程序会重新从main函数开始执行,当执行到判断唤醒键是否按住超过3S时,如果超过了3S,则跳过进入待机模式的函数,进入正常模式,接着向下执行程序。

STM32F1开发指南笔记24----待机唤醒实验_第8张图片

STM32F1开发指南笔记24----待机唤醒实验_第9张图片

STM32F1开发指南笔记24----待机唤醒实验_第10张图片

STM32F1开发指南笔记24----待机唤醒实验_第11张图片

STM32F1开发指南笔记24----待机唤醒实验_第12张图片

STM32F1开发指南笔记24----待机唤醒实验_第13张图片

STM32F1开发指南笔记24----待机唤醒实验_第14张图片

main函数

#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "lcd.h"
#include "usart.h"	 
#include "wkup.h"

 int main(void)
 {	 
	
	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	 	//串口初始化为115200
 	LED_Init();			     //LED端口初始化	
	 
	printf(" before wkup_init ");
		
	WKUP_Init(); //待机唤醒初始化
	//系统上电时默认进入待机模式,当按住唤醒键后系统会重新从main函数开始执行程序,
	//当再次判断Check_WKUP()函数的返回值时,如果按住3S则返回1,不会再进入待机模式,
	//而是进入了正常模式,接着向下执行程序。
	//当再次按住唤醒键时,会触发外部中断,如果按住没超过3S,则直接清中断标志然后退出。
	//如果超过3S,则调用Sys_Enter_Standby()函数,再次进入待机模式。
	 
	printf(" after wkup_init ");
	 
	LCD_Init();	 //LCD初始化
	POINT_COLOR=RED;
	 
	LCD_ShowString(30,50,200,16,16,"Warship STM32");	
	LCD_ShowString(30,70,200,16,16,"WKUP TEST");	
	LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");
	LCD_ShowString(30,110,200,16,16,"2014/1/14");
	 
	while(1)
	{
		LED0=!LED0;
		delay_ms(250);
	}
 }

wkup.c函数

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

#define WKUP_KD PAin(0)			//PA0 检测是否外部WK_UP按键按下
	 
void Sys_Standby(void)
{  
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);	//使能PWR外设时钟
	PWR_WakeUpPinCmd(ENABLE);  //使能唤醒管脚功能
	PWR_EnterSTANDBYMode();	  //进入待命(STANDBY)模式 		 
}


//系统进入待机模式
void Sys_Enter_Standby(void)
{			 
	RCC_APB2PeriphResetCmd(0X01FC,DISABLE);	//复位所有IO口
	Sys_Standby();
}


//检测WKUP脚的信号
//返回值1:连续按下3s以上
//      0:错误的触发	
u8 Check_WKUP(void) 
{
	u8 t=0;	//记录按下的时间
	LED0=0; //亮灯DS0 
	while(1)
	{
		if(WKUP_KD)
		{
			t++;			//已经按下了 
			delay_ms(30);
			if(t>=100)		//按下超过3秒钟
			{
				LED0=0;	 	//点亮DS0 
				return 1; 	//按下3s以上了
			}
		}else 
		{ 
			LED0=1;
			return 0; //按下不足3秒
		}
	}
} 


//中断,检测到PA0脚的一个上升沿.	  
//中断线0线上的中断检测
void EXTI0_IRQHandler(void)
{ 		    		    				     		    
	EXTI_ClearITPendingBit(EXTI_Line0); // 清除LINE10上的中断标志位		  
	if(Check_WKUP())//关机?
	{	
		Sys_Enter_Standby();  
	}
} 


//PA0 WKUP唤醒初始化
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_Pin =GPIO_Pin_0;	 //PA.0
	GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPD;//上拉输入
	GPIO_Init(GPIOA, &GPIO_InitStructure);	//初始化IO
    
	//使用外部中断方式
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0);	//中断线0连接GPIOA.0

	EXTI_InitStructure.EXTI_Line = EXTI_Line0;	//设置按键所有的外部线路
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;			//设外外部中断模式:EXTI线路为中断请求
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;  //上升沿触发
 	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
	EXTI_Init(&EXTI_InitStructure);	// 初始化外部中断

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

	if(Check_WKUP()==0) Sys_Standby();    //不是开机,进入待机模式  
}

你可能感兴趣的:(笔记)