发新帖本帖赏金 70.00元(功能说明)我要提问
返回列表
打印
[APM32F1]

基于APM32F103 Stop模式关于WFE内核命令问题分析及解决

[复制链接]
894|4
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
本帖最后由 lzh12a3nf 于 2023-10-8 13:42 编辑

#申请原创#@21小跑堂
1、前言
     近日,在学习APM32开发板关于PMU模块的内容,看到很多内容都是调用WFI内核指令进入低功耗模式,于是自己想尝试调用WFE内核指令进入低功耗模式,但在APM32F10xx中,我运用按键中断,在中断调用PMU_EnterSTOPMode库函数,用WFE内核指令进入STOP模式是存在问题的,后经查验解决了问题,于是在此进行了内容记录。

2、相关知识介绍
2.1、低功耗模式概述
         当APM32在系统或者电源复位后,芯片处于运行状态,此时HCLK为CPU提供时钟,内核执行程序代码,当CPU不需要运行时,可以采用低功耗模式来降低芯片运行的电流。

2.2、低功耗模式
        低功耗模式可分为睡眠模式和深度睡眠模式,其中深度睡眠模式分别停止模式和待机模式。而本文的重点则在于讲解进入停止模式。

2.3、进入停止模式配置

        如上,进入停止模式需要将SCB->SCR->SLEEPDEEP置为1,同时PMU->CTRL->PDDSCFG置为0,同时要执行WFI/WFE指令进入停止模式。其中,两个内核指令的区别如下:


        如上,当调用WFI内核指令时,会直接进入睡眠/深度睡眠模式。当调用WFE指令时,会根据事件锁存器的值来判断能否直接进入睡眠/深度睡眠模式。如下,我做了一个流程图:



3、问题分析及解决
3.1、配置的关键代码
int main(void)
{
    RCM_EnableAPB1PeriphClock((RCM_APB1_PERIPH_T)(RCM_APB1_PERIPH_PMU | RCM_APB1_PERIPH_BAKR));

    APM_MINI_LEDInit(LED2);
    APM_MINI_LEDInit(LED3);
        
    /* KEY1 \ KEY2 Set */
                //APM_MINI_PBInit(BUTTON_KEY1, BUTTON_MODE_GPIO);
    APM_MINI_PBInit(BUTTON_KEY1, BUTTON_MODE_EINT);
    APM_MINI_PBInit(BUTTON_KEY2, BUTTON_MODE_EINT);

    /* NVIC Priority Set */
    NVIC_ConfigPriorityGroup(NVIC_PRIORITY_GROUP_1);
    NVIC_EnableIRQRequest(EINT0_IRQn, 0, 1);
    NVIC_EnableIRQRequest(EINT1_IRQn, 1, 1);

    APM_MINI_LEDOn(LED2);
    APM_MINI_LEDOff(LED3);

    /* Enable PMU Periph Clock */
    RCM_EnableAPB1PeriphClock(RCM_APB1_PERIPH_PMU);
    PMU_Reset();

    while (1)
    {
        Delay(0x7FFFFF);

        APM_MINI_LEDToggle(LED2);
    }
}
void Eint1_Isr(void)
{
    if (EINT_ReadIntFlag(KEY1_BUTTON_EINT_LINE) != RESET)
    {
        APM_MINI_LEDOn(LED3);
        APM_MINI_LEDOff(LED2);

        /* Enter STOP Mode */
        PMU_EnterSTOPMode(PMU_REGULATOR_LOWPOWER, PMU_STOP_ENTRY_WFE);
        EINT_ClearIntFlag(KEY1_BUTTON_EINT_LINE);
    }
}

void Eint0_Isr(void)
{
    if (EINT_ReadIntFlag(KEY2_BUTTON_EINT_LINE) != RESET)
    {
        SystemInit();
        APM_MINI_LEDOff(LED3);
        /* Wait for system init */
        Delay(0xfffff);
        EINT_ClearIntFlag(KEY2_BUTTON_EINT_LINE);
    }
}
           如上代码,按下按键1后会进入睡眠模式,LED2灯灭,LED3常亮。按下按键2后会从睡眠模式中唤醒,LED2跳灯,LED3灯灭。但真实的现象便是按下按键1后,LED2仍处于跳灯状态,但LED3常亮,因此我初步判断第一次运用WFE指令时没有进入停止模式,但我从而验证我的判断?

3.2、PMU_EnterSTOPMode 函数
void PMU_EnterSTOPMode(PMU_REGULATOR_T regulator, PMU_STOP_ENTRY_T entry)
{
    /* Clear PDDSCFG and LPDSCFG bits */
    PMU->CTRL_B.PDDSCFG = 0x00;
    PMU->CTRL_B.LPDSCFG = 0x00;
    /* Set LPDSCFG bit according to regulator value */
    PMU->CTRL_B.LPDSCFG = regulator;
    /* Set Cortex System Control Register */
    SCB->SCR |= (uint32_t)0x04;
    /* Select STOP mode entry*/
    if (entry == PMU_STOP_ENTRY_WFI)
    {
        /* Request Wait For Interrupt */
        __WFI();
    }
    else
    {
        /* Request Wait For Event */
        __WFE();
    }

    /* Reset SLEEPDEEP bit of Cortex System Control Register */
    SCB->SCR &= (uint32_t)~((uint32_t)0x04);
}
           如下库API函数中,运用一次WFE内核指令,当我第一次看到这个函数时,并没有发现什么问题,于是,我照着手册深入我的问题探究。于是,我在《Cortex M3与M4权威指南》中找到如下内容:

             当我们运用WFE内核指令进入停止模式时,一般调用两次WFE内核指令,因为事件寄存器会因为中断事件的产生而置位。这时,在结合2.3中内容,我便知晓了问题的答案。因为在初始化的按键配置中,按键1和按键2连接了外部中断线,当我调用该库函数中,运用WFE指令进入停止模式时,第一次会因为有中断事件的产生,WFE的作用是运用于清除事件锁存器的值,而第二次才用于进入睡眠模式,因此在后面的Demo例程中,我给出了一种解决方法。

注:

1、在解决问题的过程中,我给出了第二种解决方法,便是不通过按键中断调用WFE内核指令进入停止模式,而是在主函数中直接对按键进行一个是否按键的判断,按下即进入睡眠模式。(这两种方法均已通过实验)。




4、基于APM32F103 PMU_Stop Demo - 运用WFE内核指令

PMU_Stop Demo.rar (264.08 KB)


本次分享到此结束,如有问题大家一起在评论区讨论,谢谢

使用特权

评论回复

打赏榜单

21小跑堂 打赏了 70.00 元 2023-10-19
理由:恭喜通过原创审核!期待您更多的原创作品~

评论
21小跑堂 2023-10-19 18:23 回复TA
解决APM32F103使用WFE指令进入停止模式失败的问题,从发现问题要猜测问题原因,再到解决问题,整个流程完整详细。其实二姨这里也可以给出一个关键技术点,或许可以与你一同验证猜想:为了顺利进入深度睡眠模式,所有 EXTI 线上的挂起状态和相关外设标志位必须被复位。否则,程序将直接跳过深度睡眠模式进入过程而继续执行下面的程序。 
forgot 2023-10-12 11:46 回复TA
赞,来学习一下! 
lzh12a3nf 2023-10-8 10:21 回复TA
修改部分内容表述 
沙发
lzh12a3nf|  楼主 | 2023-10-8 10:21 | 只看该作者
修改部分内容表述

使用特权

评论回复
发新帖 本帖赏金 70.00元(功能说明)我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

8

主题

37

帖子

0

粉丝