代码分布pwr.h
用TFLCD的例程改主函数即可
STM32 待机模式简介
很多单片机都有低功耗模式,STM32 也不例外。在系统或电源复位以后,微控制器处于运 行状态。运行状态下的 HCLK 为 CPU 提供时钟,内核执行程序代码。当 CPU 不需继续运行时, 可以利用多个低功耗模式来节省功耗,例如等待某个外部事件时。用户需要根据最低电源消耗, 最快速启动时间和可用的唤醒源等条件,选定一个最佳的低功耗模式。
STM32的3种低功耗模式
①睡眠模式:内核停止,外设如NVIC,系统时钟Systick仍运行。
②停止模式:所有时钟都已停止。1.8V内核电源工作。PLL,HIS和HSE RC振荡器功能禁止。
寄存器和SRAM内容保留。
③待机模式:1.8V内核电源关闭。只有备份寄存器和待机电路维持供电。寄存器和SRAM内容全部丢失。实现最低功耗。
在运行模式下,可以通过下面方式降低功耗:
①降低系统时钟。
②关闭APB和AHB总线上未被使用的外设时钟。
进入待机模式寄存器由 void PWR_EnterSTANDBYMode(void)函数配置
待机模式下的输入/输出端口状态
在待机模式下,所有的1/O引脚处于高阻态,除了以下的引脚:
复位引脚(始终有效)
当被设置为防侵入或校准输出时的TAMPER引脚
被使能的唤醒引脚
待机模式理想状态下,只需要2uA电流。停机模式下典型电流为20uA。
配置步骤
1)使能电源时钟。
2) 设置 WK_UP 引脚作为唤醒源。
3)设置 SLEEPDEEP 位,设置 PDDS 位,执行 WFI 指令,进入待机模式。
4)最后编写 WK_UP 中断函数。
实验现象
下载后长按wkup开机,按wk_up 3s进入待机模式,再长按wk_up3s唤醒
进入主函数 判断是否按下 没有按直接进入待机模式 在待机模式下 按wkup开启中断 唤醒重新执行主函数 再按wkup进入待机模式
实验代码
wkup.c
#include "wkup.h"
#include "led.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_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 寄存器
if(Check_WKUP()==0) Sys_Standby(); //不是开机,进入待机模式
}
wkup.h
#ifndef __WKUP_H
#define __WKUP_H
#include "sys.h"
#define WKUP_KD PAin(0) //PA0 检测是否外部 WK_UP 按键按下
u8 Check_WKUP(void); //检测 WKUP 脚的信号
void WKUP_Init(void); //PA0 WKUP 唤醒初始化
void Sys_Enter_Standby(void); //系统进入待机模式
#endif
main.c
#include "led.h"
#include "sys.h"
#include "lcd.h"
#include "usart.h"
#include "delay.h"
#include "wkup.h"
int main(void)
{
delay_init(); //延时函数初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
uart_init(9600); //串口初始化为 9600
LED_Init(); //初始化与 LED 连接的硬件接口
WKUP_Init(); //初始化 WK_UP 按键,同时检测是否正常开机?
LCD_Init(); //初始化 LCD
POINT_COLOR=RED;
LCD_ShowString(30,50,200,16,16,"Mini 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/3/8");
while(1)
{
LED0=!LED0;
delay_ms(250);
}
}