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

    文章

    17123

    浏览量

    350986
  • 电路设计
    +关注

    关注

    6673

    文章

    2451

    浏览量

    204155
  • RTOS
    +关注

    关注

    22

    文章

    811

    浏览量

    119593
  • 电池
    +关注

    关注

    84

    文章

    10560

    浏览量

    129470

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

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

收藏 人收藏

    评论

    相关推荐

    stm32低功耗设计技巧

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

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

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

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

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

    国产蓝牙模组 | 低功耗蓝牙应用

    蓝牙技术联盟于2010年推出了蓝牙4.0规范,其中低功耗蓝牙的出现满足了小型电池供电设备进行低功耗无线连接的需求,因此得到广泛应用。本文章将带你深入了解低功耗蓝牙的应用。低功耗蓝牙简介
    的头像 发表于 06-14 08:25 510次阅读
    国产蓝牙模组 | <b class='flag-5'>低功耗</b>蓝牙应用

    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

    无法从低功耗模式中正确唤醒STM32L552怎么办?

    STM32L552无法从低功耗模式中正确唤醒
    发表于 03-20 08:07

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

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

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

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

    stm32f030K4t6 低功耗设置

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