RTOS 低功耗设计原理及实现

news2024/11/29 23:43:21

RTOS 低功耗设计原理及实现


在这里插入图片描述


文章目录

    • RTOS 低功耗设计原理及实现
    • 👨‍🏫前言
    • 👨‍🔬Tickless Idle Mode 的原理及实现
    • 👨‍🚀Tickless Idle Mode 的软件设计原理
    • 👨‍💻Tickless Idle Mode 的实现
    • 👨‍⚖️结尾


👨‍🏫前言


目前, 越来越多的嵌入式产品在开发中使用 RTOS 作为软件平台, 同时,开发中对低功耗的要求也越来越高, 本文会讨论一下如何在 RTOS 中处理微控制器的低功耗特性。

应用中使用的 RTOS 一般采用基于时间片轮转的抢占式任务调度机制,一般的低功耗设计思路如下:

  • 当 Idle 任务运行时,进入低功耗模式;
  • 在适当的条件下,通过中断或者外部事件唤醒 MCU

但是, 从第二点可以看出,每次当 OS 系统定时器产生中断时,也会将 MCU 从低功耗模式中唤醒,而频繁的进入低功耗模式/从低功耗模式中唤醒会使得 MCU 无法进入深度睡眠,对低功耗设计而言也是不合理的。

在 FreeRTOS 中给出了一种低功耗设计模式 —— Tickless Idle Mode, 这个方法可以让 MCU 更长的时间处于低功耗模式。

🌸🌸🌸🌷🌷🌷💐💐💐🌷🌷🌷🌸🌸🌸

👨‍🔬Tickless Idle Mode 的原理及实现


在这里插入图片描述

上图是任务调度示意图,横轴是时间轴, T1, T2, T3, T4 是 RTOS 的时间片基准,有四个任务分别是 TaskA,TaskB,TaskC,TaskD:

  • Task A,周期性任务
  • Task B, 周期性任务
  • Task C,突发性任务
  • Task D,周期性任务

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

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

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

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

4️⃣Idle4
与 Idle1 情况相同。

🐾🐾🐾🐾🐾🐾🐾🐾🐾🐾🐾🐾

👨‍🚀Tickless Idle Mode 的软件设计原理


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

  • 合理地进入低功耗模式(避免频繁使 MCU 在低功耗模式和运行模式下进行不必要的切换);RTOS 的系统时钟源于硬件的某个周期性定时器(Cortex-M 系列内核多数采用 SysTick),RTOS 的任务调度器可以预期到下一个周期性任务(或者定时器任务) 的触发时间,如上文所述,调整系统时钟定时器中断触发时间,可以避免 RTOS 进入不必要的时间中断,从而更长的时间停留在低功耗模式中,此时 RTOS 的时钟不再是周期的而是动态的(在原有的时钟基准时将不再产生中断,即 Tickless)。
  • 当 MCU 被唤醒时,通过某种方式为系统时钟提供补偿。MCU 可能被两种情况所唤醒,动态调整过的系统时钟中断或者突发性的外部事件,无论是哪一种情况,都可以通过运行在低功耗模式下的某种定时器来计算出 MCU 处于低功耗模式下的时间,在 MCU 唤醒后对系统时间进行软件补偿;
  • 软件实现时,要根据具体的应用情景和 MCU 低功耗特性来处理问题。尤其是 MCU 的低功耗特性,不同 MCU 处于不同的低功耗模式下所能使用的外设(主要是定时器) 是不同的, RTOS 的系统时钟可以进行适当的调整。
🌻🌻🌻🌼🌼🌼🌺🌺🌺🌼🌼🌼🌻🌻🌻

👨‍💻Tickless Idle Mode 的实现


这里以 STM32F407 系列的 MCU 为例, 首先需要明确的是 MCU 的低功耗模式, F407 有 3 种低功耗模式:Sleep,Stop, Standby, 在 RTOS 平台时, SRAM 和寄存器的数据不应丢失, 此外需要一个定时器为 RTOS 提供系统时钟, 这里选择 Sleep 模式下进行实现。

在这里插入图片描述

1. 使能

#define configUSE_TICKLESS_IDLE    1

2. 空闲任务(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 */
   ...
  }
}

3. 低功耗模式处理(根据 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();
  }
}

👨‍⚖️结尾


STM32 家族中拥有不同的系列,特别是专为低功耗应用设计的 L 系列,为其设计 RTOS 低功耗特性实现时可以有更多的实现方式(例,某种模式下内核停止运行, 此时可以使用外部定时器或者 RTC 来代替 Systick 作为系统定时器)。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/792721.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

超过443万人次观看|央媒聚焦全球吉商大会,实在智能携国产大模型TARS出席

据吉林新闻联播近期报道&#xff0c;第八届全球吉商大会、第二届吉林省校友人才促进吉林振兴发展大会在长春开幕。省委书记景俊海出席开幕式并讲话。省委副书记、省长胡玉亭主持开幕式。省政协主席朱国贤为“吉商突出贡献人物”颁奖。全国工商联副主席汪鸿雁致辞。开幕式全程进…

3.2.20:DTP与Datepicker实现日期的输入

【分享成果&#xff0c;随喜正能量】人生艰难自不必去回避&#xff0c;人生艰难说多了也是白说&#xff0c;为什么&#xff0c;解决不了问题&#xff0c;说了也还是那么难。。 我给VBA的定义&#xff1a;VBA是个人小型自动化处理的有效工具。利用好了&#xff0c;可以大大提高…

PostgreSQL——sql文件导入

Windows方式&#xff1a; 进入PostgreSQL安装目录的bin&#xff0c;进入cmd 执行命令&#xff1a; psql -d 数据库名 -h localhost -p 5432 -U 用户名 -f 文件目录 SQL Shell: 执行命令&#xff1a; \i 文件目录(Windows下要加引号和双斜线)

第五章 聚合函数与内置函数

第五章 聚合函数与内置函数 一、聚合函数1、常用聚合函数2、示例&#xff08;1&#xff09;count&#xff08;2&#xff09;sum&#xff08;3&#xff09;avg&#xff08;4&#xff09;max&#xff08;5&#xff09;min 二、内置函数1、日期函数&#xff08;1&#xff09;总览&…

【Java基础学习打卡18】运算符(上)

目录 前言一、运算符和表达式1.运算符2.表达式 二、算术运算符1.加法运算符2.减法运算符3.乘法运算符4.除法运算符5.取余运算符6.表达式类型自动提升 总结 前言 本文主要介绍运算符和表达式&#xff0c;及运算符中的算术运算符。在 Java 编程中&#xff0c;运算符起着非常重要…

Java获取调用当前方法的方法名和行数(亲测可行)

有时候一个方法被很多方法调用了&#xff0c;但是在调试应用程序的时候&#xff0c;需要知道是哪个方法调用它的&#xff0c;方便定位bug问题。否者&#xff0c;比较难以理清和解决一些bug问题。 适用&#xff1a;任何适用java语言编程的地方&#xff0c;java后端和android端。…

【BOOST程序库】时间日期库

基本概念这里不再浪费时间介绍了&#xff0c;这里给出时间日期库的常见使用方法&#xff1a; #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #include <boost/version.hpp> #include <boost/config.hpp>//时间库&#xff1…

AI绘画 | 黄金时代的铠甲女王

我是赤辰。本栏目专程向大家分享由SD制作的令人惊叹的AI绘画作品。这些作品以高品质、纯净背景、完美形象和直爽风格脱颖而出。数字化时代的艺术创新&#xff0c;接下来让我们一同领略这些作品带来的视觉盛宴&#xff0c;让艺术点亮生活&#xff01; 参考提示词&#xff1a; 非…

mysql的日期类型的数据转换为年或者月类型的统计

SELECT CONCAT(YEAR(DATE), if (MONTH(DATE)<10,CONCAT(0,MONTH(DATE)),MONTH(DATE))) AS date , round(SUM(capacity),2) AS ca_dsoc FROM dianchi4 where date > 20211231 GROUP BY YEAR(DATE), MONTH(DATE) 月度的跨年处理就是第一个

教师综合评价系统ssm学生班级课程选课教务评教管理jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目&#xff0c;Java EE JSP项目&#xff0c;在工作环境中基本使用不到&#xff0c;但是很多学校把这个当作编程入门的项目来做&#xff0c;故分享出本项目供初学者参考。 一、项目描述 基于Java的疫情社区人员流动管理系统springboot vue …

大数据时代个人信息安全保护小贴士

个人信息安全保护小贴士 1. 朋友圈“五不晒”2. 手机使用“四要”、“六不要”3. 电脑使用“七注意”4. 日常上网“七注意”5. 日常生活“五注意” 互联网就像公路&#xff0c;用户使用它&#xff0c;就会留下脚印。 每个人都在无时不刻的产生数据&#xff0c;在消费数据的同时…

error: /tmp/ccxy1wo0.o: multiple definition of ‘tgt_flow_thread_init‘

linux 项目使用Makefile 编译代码时&#xff0c;一直报错 从报错意思上看很明确&#xff0c;就是重复定义 tgt_flow_thread_init函数 但是我从全局搜索代码看根本不存在重复定义问题。 从网上看是说可能存在头文件有重复的定义或者头文件被重复的引用&#xff0c;但是我看了…

PM、PMZ、PDM、VPDM比例调压阀控制放大器

PMV、PDV、SPDB、VPDB比例溢流阀控制放大器 PM、PMZ、PDM、VPDM比例调压阀控制放大器 SE、SEH、PSR2、PSR3比例流量调节阀控制放大器 EDL、PSL、 PSV、PSLF、PSVF比例多路换向阀控制放大器 比例多路换向阀属于换向阀类。 它控制一个或同时操作的多个液压耗能器的运动方向和…

【C语言】9-三大结构之选择结构-3

1. 综合举例 1.1 例1 编写一个程序完成输入一个 1-7 中的数字,输出对应的是星期几 这里推荐使用 Visio 或者类似的软件来画一个流程图。在进行程序设计时,尤其是大型项目时,软件流程图可以帮助我们很好的分析程序的结构以及结构需求,跟着流程图来写程序可以让写出来的程序…

数据库触发器简介——插入数据的触发器

1.触发器 准备工作&#xff1a;日志表user_logs create table user_logs(id int(11) not null auto_increment,operation varchar(20) not null comment 操作类型, insert/update/delete,operate_time datetime not null comment 操作时间,operate_id int(11) not null commen…

TortoiseGit 入门指南17:使用子模块

如果你想在自己的代码仓库中嵌入其它仓库&#xff0c;这称为引入子模块&#xff08;Submodule&#xff09;。使用右键菜单TortoiseGit - Submodules Add 选项&#xff0c;弹出添加子模块对话框&#xff0c;可以将一个外部仓库嵌入到源代码树的专用子目录中。 Repository&#x…

Apache pulsar 技术系列-- 消息重推的几种方式

导语 Apache Pulsar 是一个多租户、高性能的服务间消息传输解决方案&#xff0c;支持多租户、低延时、读写分离、跨地域复制&#xff08;GEO replication&#xff09;、快速扩容、灵活容错等特性。在很多场景下&#xff0c;用户需要通过 MQ 实现消息的重新推送能力&#xff0c…

若依框架@DataScop不生效问题

主要原因没有在方法传参中method直接变成对象&#xff0c;而还是用String来进行接受&#xff0c;导致切面没有获取到参数 若依框架DataScop不生效问题

neo4j教程-Cypher操作

Cypher基础操作 Cypher是图形存储数据库Neo4j的查询语言&#xff0c;Cypher是通过模式匹配Neo4j数据库中的节点和关系&#xff0c;从而对数据库Neo4j中的节点和关系进行一系列的相关操作。 下面&#xff0c;通过一张表来介绍一下常用的Neo4j操作命令及相关说明&#xff0c;具…

产业大数据应用:洞察企业全维数据,提升企业监、管、服水平

​在数字经济时代&#xff0c;数据已经成为重要的生产要素&#xff0c;数字化改革风生水起&#xff0c;在新一代科技革命、产业革命的背景下&#xff0c;产业大数据服务应运而生&#xff0c;为区域产业发展主导部门提供了企业洞察、监测、评估工具。能够助力区域全面了解企业经…