0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

探究STM32、FreeRTOS低功耗设计思路和原理

strongerHuang 来源:嵌入式专栏 作者:strongerHuang 2021-05-07 17:20 次阅读

如今电池供电产品很多,电池供电通常设计到一个问题,那就是低功耗。 本文为大家讲讲基于STM32FreeRTOS实现低功耗思想和原理。

低功耗设计常规思路应用中使用的 RTOS 一般采用基于时间片轮转的抢占式任务调度机制,一般的低功耗设计思路如下:1. 当 Idle 任务运行时,进入低功耗模式;2. 在适当的条件下,通过中断或者外部事件唤醒 MCU

但是, 从第二点可以看出,每次当 OS 系统定时器产生中断时,也会将 MCU 从低功耗模式中唤醒,而频繁的进入低功耗模式/从低功耗模式中唤醒会使得 MCU 无法进入深度睡眠,对低功耗设计而言也是不合理的。 在 FreeRTOS 中给出了一种低功耗设计模式 ——Tickless Idle Mode, 这个方法可以让 MCU 更长时间的处于低功耗模式。

二Tickless Idle Mode原理及实现

1. 情景分析

FreeRTOS各任务情况:

o4YBAGCVBouAJJekAACzil3V4-g800.png

上图是任务调度示意图,横轴是时间轴, T1, T2, T3, T4 是 RTOS 的时间片基准,有四个任务分别是 TaskA,B,C,D。

Task A:周期性任务

Task B:周期性任务

Task C:突发性任务

Task D:周期性任务

从图中可以看出在四个任务进行调度之间,会有四次空闲期间(此时 RTOS 会调度 Idle 任务运行, 软件设计的目标应该是尽可能使 MCU 在 Idle 任务运行时处于低功耗模式) 。

Idle1: Idle 任务运行期间,会产生一次系统时钟滴答,此时会唤醒 MCU,唤醒后 MCU 又会进入低功耗模式, 这次唤醒是无意义的。期望使 MCU 在 Idle1 期间一直处于低功耗模式, 因此适当调整系统定时器中断使得 T1 时不触发系统时钟中断, 中断触发点设置为 Task B 到来时;

Idle2:Task C 在系统滴答到达前唤醒 MCU(外部事件) , MCU 可以在 Idle2 中可以一直处于低功耗模式;

Idle3: 与 Idle2 情况相同,但 Idle3 时间很短,如果这个时间很短,那么进入低功耗模式的意义并不大,因此在进入低功耗模式时软件应该添加策略;

Idle4: 与 Idle1 情况相同。

2. Tickless Idle Mode 的软件设计原理

Tickless Idle Mode 的设计思想在于尽可能得在 MCU 空闲时使其进入低功耗模式。从上述情景中可以看出软件设计需要解决的问题有:

a. 合理的进入低功耗模式(避免频繁使 MCU 在低功耗模式和运行模式下进行不必要的切换) ;

RTOS 的系统时钟源于硬件的某个周期性定时器(Cortex-M 系列内核多数采用 SysTick) ,RTOS 的任务调度器可以预期到下一个周期性任务(或者定时器任务) 的触发时间,如上文所述,调整系统时钟定时器中断触发时间,可以避免 RTOS 进入不必要的时间中断,从而更长的时间停留在低功耗模式中,此时 RTOS 的时钟不再是周期的而是动态的(在原有的时钟基准时将不再产生中断,即 Tickless) ;

b. 当 MCU 被唤醒时,通过某种方式提供为系统时钟提供补偿。

MCU 可能被两种情况所唤醒, 动态调整过的系统时钟中断或者突发性的外部事件,无论是哪一种情况,都可以通过运行在低功耗模式下的某种定时器来计算出 MCU 处于低功耗模式下的时间,在 MCU 唤醒后对系统时间进行软件补偿;

c. 软件实现时,要根据具体的应用情景和 MCU 低功耗特性来处理问题。

尤其是 MCU 的低功耗特性, 不同 MCU 处于不同的低功耗模式下所能使用的外设(主要是定时器) 是不同的, RTOS 的系统时钟可以进行适当的调整。

3. Tickless Idle Mode 的实现

这里以 STM32F407 系列的 MCU 为例, 首先需要明确的是 MCU 的低功耗模式, F407 有 3 种低功耗模式:Sleep、Stop、 Standby。

o4YBAGCVBouADdfQAAPbeX2DTGI339.png

在 RTOS 平台时, SRAM寄存器的数据不应丢失, 此外需要一个定时器为 RTOS 提供系统时钟, 这里选择 Sleep 模式下进行实现。 使能Tickless Idle:

#define configUSE_TICKLESS_IDLE 1

RTOS空闲任务(空闲时自动调用)实现:

/* Idle 任务 */void prvIdleTask( void *pvParameters ){ for( ; ; ) { //。。.#if(configUSE_TICKLESS_IDLE != 0) { TickType_t xExpectedIdleTime; /* 用户策略以决定是否需要进入 Tickless Mode */ xExpectedIdleTime = prvGetExpectedIdleTime(); if( xExpectedIdleTime 》= configEXPECTED_IDLE_TIME_BEFORE_SLEEP ) { vTaskSuspendAll();

// 挂起调度器 { configASSERT( xNextTaskUnblockTime 》= xTickCount ); xExpectedIdleTime = prvGetExpectedIdleTime(); if( xExpectedIdleTime 》= configEXPECTED_IDLE_TIME_BEFORE_SLEEP ) { /* 用户函数接口 */ /* 1. 进入低功耗模式和如何退出低功耗模式 */ /* 2. 系统时间补偿 */ portSUPPRESS_TICKS_AND_SLEEP( xExpectedIdleTime ); } } (void) xTaskResumeAll(); // 恢复调度器 } }#endif /* configUSE_TICKLESS_IDLE */ //。。。 }}

然后,低功耗模式处理(根据 MCU 的低功耗模式编写代码, 代码有点长……)

void vPortSuppressTicksAndSleep( portTickType xExpectedIdleTime ){ unsigned long ulReloadValue, ulCompleteTickPeriods, ulCompletedSysTickDecrements; portTickType xModifiableIdleTime; /*

最长睡眠时间不可以超过定时器的最大定时值 */ /* 通过调整定时器的时间基准可以获得更理想的最大定时值 */ if( xExpectedIdleTime 》 xMaximumPossibleSuppressedTicks ) { xExpectedIdleTime = xMaximumPossibleSuppressedTicks; } /* 停止 SysTick */ portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT; /*

计算唤醒时的系统时间,用于唤醒后的系统时间补偿 */ ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) ); if( ulReloadValue 》 ulStoppedTimerCompensation ) { ulReloadValue -= ulStoppedTimerCompensation; } __disable_interrupt(); /*

确认下是否可以进入低功耗模式 */ if( eTaskConfirmSleepModeStatus() == eAbortSleep ) { /* 不可以,重新启动系统定时器 */ portNVIC_SYSTICK_LOAD_REG = portNVIC_SYSTICK_CURRENT_VALUE_REG; portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT; portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL; __enable_interrupt(); } else { /

* 可以进入低功耗模式 */ /* 保存时间补偿,重启系统定时器 */ portNVIC_SYSTICK_LOAD_REG = ulReloadValue; portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL;portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT; /* 进入低功耗模式,可以通过 configPRE_SLEEP_PROCESSING 函数进行低功耗模式下 时钟及外设的配置*/ xModifiableIdleTime = xExpectedIdleTime; configPRE_SLEEP_PROCESSING( xModifiableIdleTime ); if( xModifiableIdleTime 》 0 ) { __DSB(); __WFI(); __ISB(); } /

* 退出低功耗模式 */ configPOST_SLEEP_PROCESSING( xExpectedIdleTime ); portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT; __disable_interrupt() __enable_interrupt(); /

*唤醒有两种情况:系统定时器或者外部事件(中断) */ if((portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT) != 0) { /* 系统定时器唤醒,时间补偿 */ unsigned long ulCalculatedLoadValue; ulCalculatedLoadValue = ( ulTimerCountsForOneTick - 1UL ) – ( ulReloadValue - portNVIC_SYSTICK_CURRENT_VALUE_REG ); if( ( ulCalculatedLoadValue 《 ulStoppedTimerCompensation ) || ( ulCalculatedLoadValue 》 ulTimerCountsForOneTick ) ) { ulCalculatedLoadValue = (ulTimerCountsForOneTick - 1UL); } portNVIC_SYSTICK_LOAD_REG = ulCalculatedLoadValue; ulCompleteTickPeriods = xExpectedIdleTime - 1UL; } else { /

* 外部事件(中断)唤醒 */ ulCompletedSysTickDecrements = ( xExpectedIdleTime * ulTimerCountsForOneTick ) - portNVIC_SYSTICK_CURRENT_VALUE_REG; ulCompleteTickPeriods = ulCompletedSysTickDecrements / ulTimerCountsForOneTick;portNVIC_SYSTICK_LOAD_REG = ( ( ulCompleteTickPeriods + 1 ) * ulTimerCountsForOneTick ) - ulCompletedSysTickDecrements; }

/* 重启 Systick,调整系统定时器中断为正常值 */ portNVIC_SYSTICK_CURRENT_VALUE_REG = 0UL; portENTER_CRITICAL(); { portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT | portNVIC_SYSTICK_ENABLE_BIT; vTaskStepTick( ulCompleteTickPeriods ); portNVIC_SYSTICK_LOAD_REG = ulTimerCountsForOneTick - 1UL; } portEXIT_CRITICAL(); }}

三、最后

低功耗的设计存在很多影响功耗的因素,比如电路设计、IO引脚配置等。

MCU实现低功耗的方法和种类有很多,设计时需要注意一些低功耗细节问题。

最后,以上方法仅供学习参考,具体请按照实际项目选择合理的低功耗设计方案。

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • mcu
    mcu
    +关注

    关注

    146

    文章

    16958

    浏览量

    350158
  • 电路设计
    +关注

    关注

    6663

    文章

    2426

    浏览量

    203140
  • RTOS
    +关注

    关注

    21

    文章

    809

    浏览量

    119404
  • 电池
    +关注

    关注

    84

    文章

    10436

    浏览量

    128847

原文标题:基于STM32、FreeRTOS低功耗设计思路和原理

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    stm32低功耗设计技巧

    STM32低功耗设计是一个综合性的工作,需要在硬件设计和软件设计两个层面进行综合考虑和优化。以下是一些关键的STM32低功耗设计技巧: 一、硬件设计层面 选择
    的头像 发表于 11-19 15:52 56次阅读

    STM32WB0系列无线微控制器满足低功耗蓝牙应用需求

    蓝牙技术作为应用最为广泛的2.4GHz短距离通信技术,对无线微控制器芯片的集成度、功耗、性能、安全性等有较高要求。STM32WB0系列,是兼具高性价比和低功耗的无线微控制器,可充分满足无线蓝牙应用对芯片的要求。
    的头像 发表于 10-12 11:03 363次阅读
    <b class='flag-5'>STM32</b>WB0系列无线微控制器满足<b class='flag-5'>低功耗</b>蓝牙应用需求

    一款4644芯片低功耗设计思路解析

    输出并联使用。 功耗是衡量芯片性能的一个重要指标,功耗越低,电子设备的续航时间越长,减少了散热问题,降低能耗。ASP4644单通道工作时,通过拉低RUN引脚使芯片进入关断模式,此时芯片将会处于低功耗
    发表于 08-16 14:44

    STM32L051x8做项目遇到的功耗问题求解

    在用STM32L051x8做一个项目,在功耗方面要求很高。有几个问题想请教论坛的高手: 1. STM32L0x有6个低功耗模式,在freeRTOS
    发表于 04-26 06:53

    STM32F407 STOP模式低功耗的电流偏大怎么解决?

    大家好!最近有个项目用 STM32F407VGT6(100pin , 内1MBFLASH)低功耗的电流偏大 , 程序 测试说明如下: 1. STM32F407VGT6 只接了8M的高频晶振和32K
    发表于 04-15 07:00

    STM32L151RCT6单片机下载完程序不断电重启进不了低功耗怎么解决?

    最近遇到一个很奇怪的问题,使用的是stm32L151RCT6单片机,固件库用的是STM32Cube_FW_L1_V1.8.1,程序中有低功耗的部分,当我下载完程序然后将设备彻底断电重启后,设备能正常
    发表于 04-12 07:07

    stm32L151配置低功耗,无法进入wakeup中断怎么解决?

    最近在搞lora低功耗,需要将stm32L151配置低功耗,但是遇到问题了,无法进入wakeup中断。。。
    发表于 04-10 06:13

    STM32L431低功耗进入异常的原因?

    各位大佬好,小弟在使用STM32L431单片机的时候遇到一个奇怪的问题,我用的是HAL库,当我下载完程序后如果不完全断电,单片机进不了低功耗,即使通过硬件复位现象也是一样,但是只要我断开电源重新上电
    发表于 04-09 07:03

    stm32g070怎么调试才能降低功耗

    stm32g070,单芯片的最小系统板使用官方例程进入standy 模式后电量70ua左右,和手册的1,2个ua不符,已经确定是芯片的功耗的,不知道该怎么调试才能降低功耗
    发表于 03-28 06:46

    STM32调试进入低功耗而没有退出,该如何解决?

    STM32调试遇到如图问题,大意是进入低功耗而没有退出,请教大神们该如何解决?
    发表于 03-22 07:52

    stm32G070串口设置外部中断触发退出低功耗后重新变为串口,串口发送后退出低功耗为什么会卡死?

    stm32G070低功耗STOP模式外部中断唤醒,串口设置外部中断触发退出低功耗后重新变为串口,串口发送后 退出低功耗不知道为什么会卡死
    发表于 03-20 07:24

    低功耗蓝牙技术的特点 低功耗蓝牙如何实现低功耗

    低功耗蓝牙技术是一种优化的蓝牙技术,专为满足低功耗需求而设计。它通过采用一系列节能措施和技术,实现了更低的功耗消耗,延长了设备的续航时间。
    的头像 发表于 02-07 16:49 1871次阅读

    stm32f030K4t6 低功耗设置

    低功耗是微控制器设计中非常重要的一个方面,尤其对于像STM32F030K4T6这样的微控制器来说,低功耗设置是提高电池寿命、减少能耗的关键。本文将详细介绍如何在STM32F030K4T
    的头像 发表于 01-04 10:41 1141次阅读

    【应用笔记】STM32WB 低功耗蓝牙无线接口

    低功耗蓝牙是由蓝牙技术联盟(Bluetooth SIG)设计和销售的无线个人局域网技术,该技术面向医疗保健、健身、信标、安全和家庭娱乐行业的创新应用。与标准蓝牙相比,低功耗蓝牙大大降低了功耗和成本
    发表于 12-12 16:30 1次下载

    嵌入式软件的低功耗驱动软件设计

    1 低功耗的意义 电子产品尤其是电池供电的都要求低功耗,究竟怎样才算低功耗?脱离应用场合的数值没有意义,低功耗是一种看情况而定、只可意会的标准。 2
    的头像 发表于 12-07 15:00 435次阅读