【STM32 Nucleo-64测评】+测试低功耗睡眠模式和唤醒功能

STM32单片机具有睡眠、停止和待机三种低功耗方式。在睡眠模式下,只有STM32的内核时钟被关闭,而外设时钟仍然保持运行,所有的I/O引脚都保持它们在运行模式时的状态,WFI模式下可以通过任意中断唤醒,WFE模式下可以任意唤醒事件唤醒。唤醒后,程序继续在之前睡眠的位置往下运行。在以WFI方式进入睡眠模式时,CPU/CLK 关闭对其它时钟或模拟时钟源无影响。
进入睡眠模式,可以使用库函数HAL_PWR_EnterSLEEPMode。退出睡眠模式的操作也很简单,只要有任何中断发生,就自动退出。并且退出后,在之前进入睡眠模式的地方继续执行原来的程序。感觉很像在响应某个中断后,没完成中处理程序时的样子。
需要注意的是,要想测试睡眠状态,需要把系统的定时器、滴答器等自动周期产生中断的设备的中断处理禁止了,否则这些州段会理解解除睡眠状态。
测试程序很简单,把主要代码粘贴过来:
复制
// 用户按钮中断

/**

 * [url=home.php?mod=space&uid=247401]@brief[/url]  Configures EXTI line 13 (connected to PC.13 pin) in interrupt mode

 * @param  None

 * @retval None

 */

static void EXTI13_IRQHandler_Config(void) {

    GPIO_InitTypeDef GPIO_InitStructure;



    /* Enable GPIOC clock */

    __HAL_RCC_GPIOC_CLK_ENABLE();



    /* Configure PC.13 pin as input floating */

    GPIO_InitStructure.Mode = GPIO_MODE_IT_RISING_FALLING; //GPIO_MODE_IT_FALLING;



    GPIO_InitStructure.Pull = GPIO_NOPULL;

    GPIO_InitStructure.Pin = BUTTON_USER_PIN;

    HAL_GPIO_Init(GPIOC, &GPIO_InitStructure);



    /* Enable and set line 13 Interrupt to the lowest priority */

    HAL_NVIC_SetPriority(EXTI13_IRQn, 2, 0);

    HAL_NVIC_EnableIRQ(EXTI13_IRQn);

}



void HAL_GPIO_EXTI_Rising_Callback(uint16_t GPIO_Pin) {

    // 判断是不是来自PC13的

    if (GPIO_Pin == BUTTON_USER_PIN) {

        /* Toggle LED2 */

        //BSP_LED_On(LED2);

    }



    if (lpFlag == 0) {

        // 允许启动低功耗模式

        lpFlag = 1;



        // 清除计数变量

        cnt = 0;

    }

}



void HAL_GPIO_EXTI_Falling_Callback(uint16_t GPIO_Pin) {

    // 判断是不是来自PC13的

    if (GPIO_Pin == BUTTON_USER_PIN) {

        /* Toggle LED2 */

        //BSP_LED_Off(LED2);

    }

}



// 进入睡眠模式的处理

void enter_sleep_mode(void){

    HAL_SuspendTick();//关闭系统systick中断,防止睡眠被systick中断打断

    HAL_NVIC_DisableIRQ(TIM2_IRQn);    // 禁止定时器中断



    // 进入睡眠模式之前,点亮用户LED

    BSP_LED_On(LED2);



    HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);//进入WFI睡眠模式

}



// 主程序循环计数、显示处理

    while (1) {

        // 检查是否有I3C设备要求接入

        //checkI3CDevice();

        // 计数加1

        cnt++;



        GUI_ShowNum(80, 8, cnt, 3, 8, 1);



        if (lpFlag==1) {

            // 输出启动低功耗处理表示串

            GUI_ShowString(0, 8, "Wait Sleep ", 8, 1);

            // 延迟5秒后,进入低功耗模式

            if (cnt == 5) {

                // 时间到,改变标志,防止下次用户按钮启动低功耗模式

                lpFlag = 2;



                GUI_ShowString(0, 8, "Start Sleep", 8, 1);



                // 启动低功耗(睡眠模式)

                enter_sleep_mode();



            }

        }



        if (lpFlag==2) {

            GUI_ShowString(0, 8, "Exit Sleep ", 8, 1);



            // 关闭用户LED

            BSP_LED_Off(LED2);



            // 启动中断

            HAL_ResumeTick();//打开系统systic中断

            HAL_NVIC_EnableIRQ(TIM2_IRQn);    // 禁止定时器中断

        }



        HAL_Delay(1000);

    }

整体测试逻辑为:
1、在主程序中,循环计数一个变量,并在OLED上显示计数结果。
2、在用户按钮中断中,第一次按下后,建立标志
3、在主程序中检测到这个标志,延时5秒后,进入睡眠模式
4、进入睡眠模式时,关闭Sistick中断和定时器中断,保留用户按钮中断(为了唤醒用)
5、在第二次用户按钮中断时,自动退出睡眠模式,启动Sistick和sing时期中断,在主程序中继续计数并显示。
运行效果用三个图片展示:
1、进入睡眠之前,正常计数并显示

2、进入睡眠后,计数停止,显示不变,同时用户的绿色LED保持亮的状态

3、退出睡眠状态后,计数继续并显示。

可以看到,在睡眠程序之前,主程序的循环处理一直在正常执行。而在进入睡眠模式后,IO口仍然保持着之前的状态,测试里使用的板载的用户LED来证明这一点的。但主程序确实停止运行了,OLED上始终显示着进入睡眠状态之前计数值。在用户第二次触发按钮后,退出了睡眠状态,主程序恢复运行,继续完成计数并显示。

因为没有合适的方式测试进入睡眠前后的电流变化,后面有时间的话,会补上这个测试结果,看看在睡眠状态下的消耗情况。
---------------------
作者:suncat0504
链接:https://bbs.21ic.com/icview-3393606-1-1.html
来源:21ic.com
此文章已获得原创/原创奖标签,著作权归21ic所有,任何人未经允许禁止转载。

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

余额充值