0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

探究STM32、FreeRTOS低功耗設(shè)計思路和原理

strongerHuang ? 來源:嵌入式專欄 ? 作者:strongerHuang ? 2021-05-07 17:20 ? 次閱讀

如今電池供電產(chǎn)品很多,電池供電通常設(shè)計到一個問題,那就是低功耗。 本文為大家講講基于STM32FreeRTOS實現(xiàn)低功耗思想和原理。

低功耗設(shè)計常規(guī)思路應(yīng)用中使用的 RTOS 一般采用基于時間片輪轉(zhuǎn)的搶占式任務(wù)調(diào)度機制,一般的低功耗設(shè)計思路如下:1. 當 Idle 任務(wù)運行時,進入低功耗模式;2. 在適當?shù)臈l件下,通過中斷或者外部事件喚醒 MCU

但是, 從第二點可以看出,每次當 OS 系統(tǒng)定時器產(chǎn)生中斷時,也會將 MCU 從低功耗模式中喚醒,而頻繁的進入低功耗模式/從低功耗模式中喚醒會使得 MCU 無法進入深度睡眠,對低功耗設(shè)計而言也是不合理的。 在 FreeRTOS 中給出了一種低功耗設(shè)計模式 ——Tickless Idle Mode, 這個方法可以讓 MCU 更長時間的處于低功耗模式。

二Tickless Idle Mode原理及實現(xiàn)

1. 情景分析

FreeRTOS各任務(wù)情況:

o4YBAGCVBouAJJekAACzil3V4-g800.png

上圖是任務(wù)調(diào)度示意圖,橫軸是時間軸, T1, T2, T3, T4 是 RTOS 的時間片基準,有四個任務(wù)分別是 TaskA,B,C,D。

Task A:周期性任務(wù)

Task B:周期性任務(wù)

Task C:突發(fā)性任務(wù)

Task D:周期性任務(wù)

從圖中可以看出在四個任務(wù)進行調(diào)度之間,會有四次空閑期間(此時 RTOS 會調(diào)度 Idle 任務(wù)運行, 軟件設(shè)計的目標應(yīng)該是盡可能使 MCU 在 Idle 任務(wù)運行時處于低功耗模式) 。

Idle1: Idle 任務(wù)運行期間,會產(chǎn)生一次系統(tǒng)時鐘滴答,此時會喚醒 MCU,喚醒后 MCU 又會進入低功耗模式, 這次喚醒是無意義的。期望使 MCU 在 Idle1 期間一直處于低功耗模式, 因此適當調(diào)整系統(tǒng)定時器中斷使得 T1 時不觸發(fā)系統(tǒng)時鐘中斷, 中斷觸發(fā)點設(shè)置為 Task B 到來時;

Idle2:Task C 在系統(tǒng)滴答到達前喚醒 MCU(外部事件) , MCU 可以在 Idle2 中可以一直處于低功耗模式;

Idle3: 與 Idle2 情況相同,但 Idle3 時間很短,如果這個時間很短,那么進入低功耗模式的意義并不大,因此在進入低功耗模式時軟件應(yīng)該添加策略;

Idle4: 與 Idle1 情況相同。

2. Tickless Idle Mode 的軟件設(shè)計原理

Tickless Idle Mode 的設(shè)計思想在于盡可能得在 MCU 空閑時使其進入低功耗模式。從上述情景中可以看出軟件設(shè)計需要解決的問題有:

a. 合理的進入低功耗模式(避免頻繁使 MCU 在低功耗模式和運行模式下進行不必要的切換) ;

RTOS 的系統(tǒng)時鐘源于硬件的某個周期性定時器(Cortex-M 系列內(nèi)核多數(shù)采用 SysTick) ,RTOS 的任務(wù)調(diào)度器可以預(yù)期到下一個周期性任務(wù)(或者定時器任務(wù)) 的觸發(fā)時間,如上文所述,調(diào)整系統(tǒng)時鐘定時器中斷觸發(fā)時間,可以避免 RTOS 進入不必要的時間中斷,從而更長的時間停留在低功耗模式中,此時 RTOS 的時鐘不再是周期的而是動態(tài)的(在原有的時鐘基準時將不再產(chǎn)生中斷,即 Tickless) ;

b. 當 MCU 被喚醒時,通過某種方式提供為系統(tǒng)時鐘提供補償。

MCU 可能被兩種情況所喚醒, 動態(tài)調(diào)整過的系統(tǒng)時鐘中斷或者突發(fā)性的外部事件,無論是哪一種情況,都可以通過運行在低功耗模式下的某種定時器來計算出 MCU 處于低功耗模式下的時間,在 MCU 喚醒后對系統(tǒng)時間進行軟件補償;

c. 軟件實現(xiàn)時,要根據(jù)具體的應(yīng)用情景和 MCU 低功耗特性來處理問題。

尤其是 MCU 的低功耗特性, 不同 MCU 處于不同的低功耗模式下所能使用的外設(shè)(主要是定時器) 是不同的, RTOS 的系統(tǒng)時鐘可以進行適當?shù)恼{(diào)整。

3. Tickless Idle Mode 的實現(xiàn)

這里以 STM32F407 系列的 MCU 為例, 首先需要明確的是 MCU 的低功耗模式, F407 有 3 種低功耗模式:Sleep、Stop、 Standby。

o4YBAGCVBouADdfQAAPbeX2DTGI339.png

在 RTOS 平臺時, SRAM寄存器的數(shù)據(jù)不應(yīng)丟失, 此外需要一個定時器為 RTOS 提供系統(tǒng)時鐘, 這里選擇 Sleep 模式下進行實現(xiàn)。 使能Tickless Idle:

#define configUSE_TICKLESS_IDLE 1

RTOS空閑任務(wù)(空閑時自動調(diào)用)實現(xiàn):

/* Idle 任務(wù) */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();

// 掛起調(diào)度器 { configASSERT( xNextTaskUnblockTime 》= xTickCount ); xExpectedIdleTime = prvGetExpectedIdleTime(); if( xExpectedIdleTime 》= configEXPECTED_IDLE_TIME_BEFORE_SLEEP ) { /* 用戶函數(shù)接口 */ /* 1. 進入低功耗模式和如何退出低功耗模式 */ /* 2. 系統(tǒng)時間補償 */ portSUPPRESS_TICKS_AND_SLEEP( xExpectedIdleTime ); } } (void) xTaskResumeAll(); // 恢復(fù)調(diào)度器 } }#endif /* configUSE_TICKLESS_IDLE */ //。。。 }}

然后,低功耗模式處理(根據(jù) MCU 的低功耗模式編寫代碼, 代碼有點長……)

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

最長睡眠時間不可以超過定時器的最大定時值 */ /* 通過調(diào)整定時器的時間基準可以獲得更理想的最大定時值 */ if( xExpectedIdleTime 》 xMaximumPossibleSuppressedTicks ) { xExpectedIdleTime = xMaximumPossibleSuppressedTicks; } /* 停止 SysTick */ portNVIC_SYSTICK_CTRL_REG = portNVIC_SYSTICK_CLK_BIT | portNVIC_SYSTICK_INT_BIT; /*

計算喚醒時的系統(tǒng)時間,用于喚醒后的系統(tǒng)時間補償 */ ulReloadValue = portNVIC_SYSTICK_CURRENT_VALUE_REG + ( ulTimerCountsForOneTick * ( xExpectedIdleTime - 1UL ) ); if( ulReloadValue 》 ulStoppedTimerCompensation ) { ulReloadValue -= ulStoppedTimerCompensation; } __disable_interrupt(); /*

確認下是否可以進入低功耗模式 */ if( eTaskConfirmSleepModeStatus() == eAbortSleep ) { /* 不可以,重新啟動系統(tǒng)定時器 */ 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 { /

* 可以進入低功耗模式 */ /* 保存時間補償,重啟系統(tǒng)定時器 */ 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 函數(shù)進行低功耗模式下 時鐘及外設(shè)的配置*/ 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(); /

*喚醒有兩種情況:系統(tǒng)定時器或者外部事件(中斷) */ if((portNVIC_SYSTICK_CTRL_REG & portNVIC_SYSTICK_COUNT_FLAG_BIT) != 0) { /* 系統(tǒng)定時器喚醒,時間補償 */ 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,調(diào)整系統(tǒng)定時器中斷為正常值 */ 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(); }}

三、最后

低功耗的設(shè)計存在很多影響功耗的因素,比如電路設(shè)計、IO引腳配置等。

MCU實現(xiàn)低功耗的方法和種類有很多,設(shè)計時需要注意一些低功耗細節(jié)問題。

最后,以上方法僅供學(xué)習(xí)參考,具體請按照實際項目選擇合理的低功耗設(shè)計方案。

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • mcu
    mcu
    +關(guān)注

    關(guān)注

    146

    文章

    17307

    瀏覽量

    352172
  • 電路設(shè)計
    +關(guān)注

    關(guān)注

    6678

    文章

    2462

    瀏覽量

    205056
  • RTOS
    +關(guān)注

    關(guān)注

    22

    文章

    819

    瀏覽量

    119806
  • 電池
    +關(guān)注

    關(guān)注

    84

    文章

    10669

    瀏覽量

    130797

原文標題:基于STM32、FreeRTOS低功耗設(shè)計思路和原理

文章出處:【微信號:strongerHuang,微信公眾號:strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    stm32低功耗設(shè)計技巧

    STM32低功耗設(shè)計是一個綜合性的工作,需要在硬件設(shè)計和軟件設(shè)計兩個層面進行綜合考慮和優(yōu)化。以下是一些關(guān)鍵的STM32低功耗設(shè)計技巧: 一、硬件設(shè)計層面 選擇
    的頭像 發(fā)表于 11-19 15:52 ?671次閱讀

    STM32WB0系列無線微控制器滿足低功耗藍牙應(yīng)用需求

    藍牙技術(shù)作為應(yīng)用最為廣泛的2.4GHz短距離通信技術(shù),對無線微控制器芯片的集成度、功耗、性能、安全性等有較高要求。STM32WB0系列,是兼具高性價比和低功耗的無線微控制器,可充分滿足無線藍牙應(yīng)用對芯片的要求。
    的頭像 發(fā)表于 10-12 11:03 ?556次閱讀
    <b class='flag-5'>STM32</b>WB0系列無線微控制器滿足<b class='flag-5'>低功耗</b>藍牙應(yīng)用需求

    一款4644芯片低功耗設(shè)計思路解析

    輸出并聯(lián)使用。 功耗是衡量芯片性能的一個重要指標,功耗越低,電子設(shè)備的續(xù)航時間越長,減少了散熱問題,降低能耗。ASP4644單通道工作時,通過拉低RUN引腳使芯片進入關(guān)斷模式,此時芯片將會處于低功耗
    發(fā)表于 08-16 14:44

    國產(chǎn)藍牙模組 | 低功耗藍牙應(yīng)用

    藍牙技術(shù)聯(lián)盟于2010年推出了藍牙4.0規(guī)范,其中低功耗藍牙的出現(xiàn)滿足了小型電池供電設(shè)備進行低功耗無線連接的需求,因此得到廣泛應(yīng)用。本文章將帶你深入了解低功耗藍牙的應(yīng)用。低功耗藍牙簡介
    的頭像 發(fā)表于 06-14 08:25 ?577次閱讀
    國產(chǎn)藍牙模組 | <b class='flag-5'>低功耗</b>藍牙應(yīng)用

    STM32低功耗控制一般都是采用哪種模式?

    STM32中,低功耗控制有三種,休眠模式,停機模式,待機模式,想知道大家都常用哪個模式?都能控制到什么程度?多少微安?
    發(fā)表于 04-30 08:03

    STM32L051x8做項目遇到的功耗問題求解

    在用STM32L051x8做一個項目,在功耗方面要求很高。有幾個問題想請教論壇的高手: 1. STM32L0x有6個低功耗模式,在freeRTOS
    發(fā)表于 04-26 06:53

    STM32F407 STOP模式低功耗的電流偏大怎么解決?

    大家好!最近有個項目用 STM32F407VGT6(100pin , 內(nèi)1MBFLASH)低功耗的電流偏大 , 程序 測試說明如下: 1. STM32F407VGT6 只接了8M的高頻晶振和32K
    發(fā)表于 04-15 07:00

    STM32L151RCT6單片機下載完程序不斷電重啟進不了低功耗怎么解決?

    最近遇到一個很奇怪的問題,使用的是stm32L151RCT6單片機,固件庫用的是STM32Cube_FW_L1_V1.8.1,程序中有低功耗的部分,當我下載完程序然后將設(shè)備徹底斷電重啟后,設(shè)備能正常
    發(fā)表于 04-12 07:07

    stm32L151配置低功耗,無法進入wakeup中斷怎么解決?

    最近在搞lora低功耗,需要將stm32L151配置低功耗,但是遇到問題了,無法進入wakeup中斷。。。
    發(fā)表于 04-10 06:13

    STM32L431低功耗進入異常的原因?

    各位大佬好,小弟在使用STM32L431單片機的時候遇到一個奇怪的問題,我用的是HAL庫,當我下載完程序后如果不完全斷電,單片機進不了低功耗,即使通過硬件復(fù)位現(xiàn)象也是一樣,但是只要我斷開電源重新上電
    發(fā)表于 04-09 07:03

    stm32g070怎么調(diào)試才能降低功耗?

    stm32g070,單芯片的最小系統(tǒng)板使用官方例程進入standy 模式后電量70ua左右,和手冊的1,2個ua不符,已經(jīng)確定是芯片的功耗的,不知道該怎么調(diào)試才能降低功耗
    發(fā)表于 03-28 06:46

    STM32調(diào)試進入低功耗而沒有退出,該如何解決?

    STM32調(diào)試遇到如圖問題,大意是進入低功耗而沒有退出,請教大神們該如何解決?
    發(fā)表于 03-22 07:52

    無法從低功耗模式中正確喚醒STM32L552怎么辦?

    STM32L552無法從低功耗模式中正確喚醒
    發(fā)表于 03-20 08:07

    stm32G070串口設(shè)置外部中斷觸發(fā)退出低功耗后重新變?yōu)榇?,串口發(fā)送后退出低功耗為什么會卡死?

    stm32G070低功耗STOP模式外部中斷喚醒,串口設(shè)置外部中斷觸發(fā)退出低功耗后重新變?yōu)榇冢诎l(fā)送后 退出低功耗不知道為什么會卡死
    發(fā)表于 03-20 07:24

    低功耗藍牙技術(shù)的特點 低功耗藍牙如何實現(xiàn)低功耗?

    低功耗藍牙技術(shù)是一種優(yōu)化的藍牙技術(shù),專為滿足低功耗需求而設(shè)計。它通過采用一系列節(jié)能措施和技術(shù),實現(xiàn)了更低的功耗消耗,延長了設(shè)備的續(xù)航時間。
    的頭像 發(fā)表于 02-07 16:49 ?2239次閱讀