「STM32」待机唤醒实验

stm32的3种低功耗模式

待机模式

低功耗涉及到的固件操作函数

待机唤醒配置步骤

待机唤醒实验

程序设计思路:

WKUP是一个按键,没有按下输出低电平,按钮按下输出高电平。

源码:

wkup.h

#ifndef __WKUP_H
#define __WKUP_H
#include "stm32f10x_gpio.h"

#define WKUP_KD GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) //PA0 检测是否外部WK_UP按键按下K

u8 Check_WKUP(void); //检测WKUP脚的信号
void WKUP_Init(void); //PA0 WKUP唤醒初始化
void Sys_Enter_Standby(void); //系统进入待机模式
#endif

wkup.c

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

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(); //不是开机,进入待机模式

}

main.c

/******************************************************************
*示例说明:stm32f103rbt6程序实例
2019.6.11 待机唤醒实验
*作者:小5
*****************************************************************/
/* Standard includes. */
#include <stdio.h>
/* Library includes.*/
#include "stm32f10x.h"
/* Hardware Library */
#include "usart.h"
#include "delay.h"
#include "led.h"
#include "key.h"
#include "wdg.h"
#include "timer.h"
#include "rtc.h"
#include "wkup.h"
int main(void)
{
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);

delay_init();
usart_config();
led_gpio_init();
WKUP_Init(); //待机唤醒初始化

while(1)
{
delay_ms(250);
}
}

示例程序现象及其简单的说明:(假如有一个显示屏)

下载程序,上电直接进入到待机模式,显示屏处于灭的状态。那是因为源文件里面的

if(Check_WKUP()==0) Sys_Standby(); 

这个语句起作用了。按钮没有被按下,输出的一个低电平,按钮按下,输出高电平,这个是按钮这个硬件在设计的时候决定的。

然后连续按按钮3s,芯片就被唤醒了,显示屏亮。这是因为当你按下的按钮的时候,芯片将从main()函数处依次向下运行,

if(Check_WKUP()==0) Sys_Standby(); 

还是这句话,它没有被运行,那么主函数继续运行后面的内容,那肯定就是正常的状态了,要是有显示屏的话,那么肯定显示屏就被点亮了,

再将按钮长按3s,这个是时候芯片进入待机模式,显示屏灭。这是因为在正常的情况下,这个按钮对应的PA0口也被设置为外部中断口,当有按钮被按下的时候,那么就去会执行我们设置的中断服务函数,进入了待机模式,显而易见了。

void EXTI0_IRQHandler(void)
{
EXTI_ClearITPendingBit(EXTI_Line0); // 清除LINE10上的中断标志位
if(Check_WKUP())//关机?
{
Sys_Enter_Standby();
}
}

喜欢我文章的朋友,欢迎关注、点赞、评论、交流。版权个人所有,转载请注明出处。

发表评论
留言与评论(共有 0 条评论)
   
验证码:

相关文章

推荐文章

'); })();