STM32 待机唤醒

3.12.1 STM32待机模式简介

3.12.2 硬件设计

3.12.3 软件设计

3.12.4 下载与测试

3.12.1 STM32待机模式简介

 

很多单片机都有低功耗模式,STM32也不例外。在系统或电源复位以后,微控制器处于运行状态。运行状态下的HCLKCPU提供时钟,内核执行程序代码。当CPU不需继续运行时,可以利用多个低功耗模式来节省功耗,例如等待某个外部事件时。用户需要根据最低电源消耗,最快速启动时间和可用的唤醒源等条件,选定一个最佳的低功耗模式。STM323中低功耗模式我们在第二章也粗略介绍了一下,这里我们再回顾一下。

STM32的低功耗模式有3种:

1)睡眠模式(CM3内核停止,外设仍然运行)

2)停止模式(所有时钟都停止)

3)待机模式(1.8V内核电源关闭)

在运行模式下,我们也可以通过降低系统时钟关闭APBAHB总线上未被使用的外设的时钟来降低功耗。三种低功耗模式一览表:

                     3.12.1.1 STM32低功耗一览表

在这三种低功耗模式中,最低功耗的是待机模式,在此模式下,最低只需要2uA左右的电流。停机模式是次低功耗的,其典型的电流消耗在20uA左右。最后就是睡眠模式了。用户可以根据自己的需求来决定使用哪种低功耗模式。

这一节,我们就针对STM32的最低功耗模式-待机模式,来做介绍。待机模式可实现STM32的最低功耗。该模式是在CM3深睡眠模式时关闭电压调节器。整个1.8V供电区域被断电。PLLHSIHSE振荡器也被断电。SRAM和寄存器内容丢失。只有备份的寄存器和待机电路维持供电。

那么我们如何进入待机模式呢?其实很简单,只要按下表的步骤执行就可以了:

                            3.12.1.1 STM32 进入及退出待机模式条件

上表还列出了退出待机模式的操作,从上表可知,我们有4种方式可以退出待机模式,即当一个外部复位(NRST引脚)IWDG复位WKUP引脚(PA0脚)上的上升沿RTC闹钟事件发生时,微控制器从待机模式退出。从待机唤醒后,除了电源控制/状态寄存器(PWR_CSR),所有寄存器被复位

从待机模式唤醒后的代码执行等同于复位后的执行(采样启动模式引脚,读取复位向量等)。电源控制/状态寄存器(PWR_CSR)将会指示内核由待机状态退出。

在进入待机模式后,除了复位引脚以及被设置为防侵入或校准输出时的TAMPER引脚和被是能的唤醒引脚(WK_UP脚),其他的IO引脚都将处于高阻态。

上表已经清楚的说明了进入待机模式的通用步骤,其中涉及到2个寄存器,也就是电源控制寄存器(PWR_CR)和电源控制/状态寄存器(PWR_CSR)。下面我们介绍一下这两个寄存器:

电源控制寄存器(PWR_CR),该寄存器的各位描述如下:

                                       3.12.1.2 寄存器PWR_CR 各位描述

这里我们通过设置PWR_CRPDDS位,使CPU进入深度睡眠时进入待机模式,同时我们通过CWUF位,清除之前的唤醒位。电源控制/状态寄存器(PWR_CSR)的各位描述如下:

                                                3.12.1.3 寄存器PWR_ CSR 各位描述

这里,我们通过设置PWR_CSREWUP位,来使能WKUP引脚用于待机模式唤醒。我们还可以从WUF来检查是否发生了唤醒事件。不过这一节我们没有用到。

通过以上介绍,我们了解了进入待机模式的方法,以及设置WK_UP引脚用于把STM32从待机模式唤醒的方法。具体步骤如下:

1)设置SLEEPDEEP位。

该位在系统控制寄存器(SCB_SCR)的第二位(详见《CM3权威指南》,第182页表13.1),我们通过设置该位,作为进入待机模式的第一步。

2)使能电源时钟,设置WK_UP引脚作为唤醒源。

因为要配置电源控制寄存器,所以必须先使能电源时钟。然后再设置PWR_CSREWUP位,使能WK_UP用于将CPU从待机模式唤醒。

3)设置PDDS位,执行WFI指令,进入待机模式。

接着我们通过PWR_CR设置PDDS位,使得CPU进入深度睡眠时进入待机模式,最后执行WFI指令开始进入待机模式,并等待WK_UP中断的到来。

4)最后编写WK_UP中断函数。

因为我们通过WK_UP中断(PA0中断)来唤醒CPU,所以我们有必要设置一下该中断函数,同时我们也通过该函数里面进入待机模式。

通过以上几个步骤的设置,我们就可以使用STM32的待机模式了,并且可以通过WK_UP来唤醒CPU,我们最终要实现这样一个功能:通过长按(3秒)WK_UP按键开机,并且通过DS0的闪烁指示程序已经开始运行,再次长按该键,则进入待机模式,DS0关闭,程序停止运行。类似于手机的开关机。

 

 

3.12.2 硬件设计

 

这一节,我们使用了WK_UP按键用于唤醒和进入待机模式。然后通过DS0来指示程序是否在运行。因为DS0WK_UPMiniSTM32开发板上都是直接连在STM32IO口上的,不需要任何修改,这里我们就不在贴图了。大家可以参考第一章相关内容。

 

3.12.3 软件设计

 

找到上一节的工程,首先在HARDWARE文件夹下新建一个WKUP的文件夹。然后打开USER文件夹下的工程,新建一个wkup.c的文件和wkup.h的头文件,保存在WKUP文件夹下,并将WKUP文件夹加入头文件包含路径。

打开wkup.c,输入如下代码:

#include "wkup.h"

#include "led.h"

#include "delay.h"

//Mini STM32开发板

//待机唤醒 驱动代码                                 

//正点原子@ALIENTEK

//2010/6/7

//THUMB指令不支持汇编内联

//采用如下方法实现执行汇编指令EFI

__asm void WFI_SET(void)

{

WFI;

}

void Sys_Standby(void)쨔ᅵᄉ￧

{

SCB->SCR|=1<<2; //使能SLEEPDEEP位 (SYS->CTRL)

RCC->APB1ENR|=1<<28; //使能电源时钟

PWR->CSR|=1<<8; //设置WKUP用于唤醒

PWR->CR|=1<<2;  //清除Wake-up标志

PWR->CR|=1<<1; //PDDS置位

WFI_SET();

}ᅯᄡᅧ뮈ᅮ  

//系统进入待机模式

void Sys_Enter_Standby(void)

{                          

     //关闭所有外设(根据实际情况写)

     RCC->APB2RSTR|=0X01FC;//复位所有IO

     Sys_Standby();//进入待机模式

}

//检测WKUP脚的信号

//返回值1:连续按下3s以上

//     0:错误的触发

u8 Check_WKUP(void)

{

     u8 t=0;

     u8 tx=0;//记录松开的次数

     LED0=0; //亮灯DS0

     while(1)

     {

                 if(WKUP_KD)//已经按下了

                 {

                             t++;

                             tx=0;

                 }else

                 {

                             tx++; //超过300ms内没有WKUP信号

                             if(tx>3)

                             {

                                         LED0=1;

                                         return 0;//错误的按键,按下次数不够

                             }

                 }

                 delay_ms(30);

                 if(t>=100)//按下超过3秒钟

                 {

                             LED0=0;            //点亮DS0

                             return 1; //按下3s以上了

                 }

     }

//中断,检测到PA0脚的一个上升沿.          

//中断线0线上的中断检测

void EXTI0_IRQHandler(void)

{                                                                                                                   

     EXTI->PR=1<<0;  //清除LINE10上的中断标志位                       

     if(Check_WKUP())//关机?

     {                       

                 Sys_Enter_Standby(); 

     }

}

//PA0 WKUP唤醒初始化

void WKUP_Init(void)

{                             

     RCC->APB2ENR|=1<<2;     //先使能外设IO PORTA时钟   

     RCC->APB2ENR|=1<<0;     //开启辅助时钟                    

    

     GPIOA->CRL&=0XFFFFFFF0;//PA0设置成输入             

     GPIOA->CRL|=0X00000008; 

     Ex_NVIC_Config(GPIO_A0RTIR);//PA0上升沿触发              

                                                                                                                                                                    

     //(检查是否是正常开)                         

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

     MY_NVIC_Init(22EXTI0_IRQChannel2);//抢占2,子优先级2,组2

}

这里我们要删除exti.c,因为该函数里面也有void EXTI0_IRQHandler(void)函数,如果不删除,MDK就会报错。该部分代码比较简单,我们在这里说明2点:1,在void Sys_Enter_Standby(void)函数里面,我们要在进入待机模式前把所有开启的外设全部关闭,我们这里仅仅复位了所有的IO口,使得IO口全部为浮空输入。其他外设(比如ADC等),大家根据自己所开启的情况进行一一关闭就可,这样才能达到最低功耗!2,在void WKUP_Init(void)函数里面,我们要先判断WK_UP是否按下了3秒钟,来决定要不要开机,如果没有按下3秒钟,程序直接就进入了待机模式。所以在下载完代码的时候,是看不到任何反应的。我们必须先按WK_UP按键3秒钟以开机,才能看到DS0闪烁。

保存wkup.c,并加入到HARDWARE组下,然后我们在wkup.h里面加入如下代码:

#ifndef __WKUP_H

#define __WKUP_H           

#include "sys.h"

//Mini STM32开发板

//待机唤醒 驱动代码                                 

//正点原子@ALIENTEK

//2010/6/7                                                                   

#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

该部分代码,也很简单,我们就不多说了。最后我们在test.c里面修改main函数如下:

int main(void)

{              

     Stm32_Clock_Init(9);//系统时钟设置

     delay_init(72);               //延时初始化

     uart_init(72,9600); //串口1初始化  

     LED_Init();                                   

     WKUP_Init();

     while(1)

     {

                 LED0=!LED0;

                 delay_ms(250);

     }

}

这里我们先初始化LEDWK_UP按键(通过WKUP_Init()函数初始化),在死循环里面等待WK_UP中断的到来,在得到中断后,判断WK_UP按下的时间长短,来决定是否进入待机模式。在WKUP_Init函数里面,我们有检测WK_UP是否按下3秒来决定是否开机,这点在前面已经介绍了。大家在下载完代码的时候要注意一下。

 

 

3.12.4 下载与测试

    

在代码编译成功之后,我们通过USB串口线下载代码到ALIENTEK MiniSTM32开发板上,这里不能像前面的代码一样,下载完就可以测试了,这里我们必须先把B0通过跳线帽连接到GND,然后再按复位键,才能开始测试。这是因为:虽然我们在mcuisp软件里面设置了编程后执行,但是代码运行的时候,我们并没有长按WK_UP按键3秒,所以程序就进入了待机模式。然后由于BOOT0的设置是串口下载模式,我们不能通过按复位键来重新启动程序。所以必须把BOOT0接到GND,使得复位键可以唤醒CPU。然后才能开始测试。

B0接到GND后,我们按复位键,然后再长按WK_UP按键3秒钟左右,可以看到DS0开始闪烁。然后再长按WK_UPDS0会灭掉,程序再次进入待机模式。

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、付费专栏及课程。

余额充值