freeRTOS中文实用教程1

2023-05-16

资料转载出处 https://www.cnblogs.com/smartjourneys/p/7073450.html

1.前言
FreeRTOS是小型多任务嵌入式操作系统,硬实时性。本章主要讲述任务相关特性及调度相关的知识。

  1. 任务的总体特点
    任务的状态
    (1)任务有两个状态,运行态和非运行态

(2)任务由非运行态转入运行态为切入,相反为切出

设置优先级
(1)最高优先级在FreeRTOSConfig.h 中 设 定 的 编 译 时 配 置 常 量configMAX_PRIORITIES中进行设置。
注:FreeRTOS 本身并没有限定这个常量的最大值,但这个值越大,则内核花销的内存空间就越多。建议将此常量设为能够用到的最小值

(2)任意数量的任务可以共享同一个优先级也可以为每个任务唯一指定一个优先级

(3)低优先级号表示任务的优先级低
(4)有效的优先级号范围从 0 到(configMAX_PRIORITES – 1)
(5)调度器保证总是在所有可运行的任务中选择具有最高优先级的任务,并使其进入运行态
(6)如果被选中的优先级上具有不止一个任务,调度器会让这些任务轮流执行

(7)调度器总是选择所有能够进入运行态的任务中具有最高优先级的任务

改变任务优先级
可以通过vTaskPrioritySet() 来改变任务的优先级

任务时间片
(1)两个任务被创建在同一个优先级上,并且一直是可运行的。所以每个任务都执行一个”时间片”,任务在时间片起始时刻进入运行态,在时间片结束时刻又退出运行态

(2)调度器需要在每个时间片结束时能够调度自己, tick中断可以完成此目的

(3)tick中断频率可以通过FreeRTOSConfig.h 中的编译时配置常量configTICK_RATE_HZ 进行配置,如设为100HZ,则时间片长度为10ms

3.任务状态机
3.1 任务状态机
在这里插入图片描述

图 完整的任务状态机

阻塞态(blocked)
(1)如果一个任务正在等待某个事件,则称这个任务处于”阻塞态(blocked)”。阻塞态是非运行态的一个子状态。

(2)任务进入阻塞态一般等待以下两种事件:定时事件和同步事件

(3)任务可以在进入阻塞态等待同步事件时指定一个等待超时时间,这样可以有效地实现阻塞状态下同时等待两种类型的事件,

比如说,某个任务可以等待队列中有数据到来,但最多只等10ms。如果10ms 内有数据到来,或是10ms 过去了还没有数据到来,这两种情况下该任务都将退出阻塞态。

挂起状态(suspended)
(1)处于挂起状态的任务对调度器而言是不可见的

(2)让一个任务进入挂起状态的唯一办法就是调用vTaskSuspend() API 函数

(3)把一个挂起状态的任务唤醒的唯一途径就是调用vTaskResume() 或TaskResumeFromISR() API 函数

(4)大多数应用程序中都不会用到挂起状态

就绪状态(ready)
(1)如果任务处于非运行状态,但既没有阻塞也没有挂起,则这个任务处于就绪(ready,准备或就绪)状态

(2)处于就绪态的任务能够被运行,但只是”准备(ready)”运行,而当前尚未运行

3.2.利用阻塞态实现延迟
1 void vTaskDelay( portTickType xTicksToDelay );
调用上面的函数的任务会进入阻塞态,xTicksToDelay表示delay的tick周期数,portTICK_RATE_MS代表每毫秒有多少个tick周期

  1. 空闲任务
    空闲任务的特点
    (1)在调度器启动的时候vTaskStartScheduler会自动创建一个空闲任务

(2)空闲任务具有最低的优先级0,保证不会影响其它更高优先级的任务进入就绪态

(3)运行在最低优先级可以保证一旦有更高优先级的任务进入就绪态,空闲任务就会立即切出运行态

(4)空间任务可以获得的执行时间量,是系统处理能力裕量的一个度量指

空闲任务钩子函数
(1)通过空闲任务钩子函数(或称回调,hook, or call-back),可以直接在空闲任务中添加应用程序相关的功能,空闲任务钩子函数会被空闲任务每循环一次就自动调用一次

(2)空闲任务钩子函数被用于:

—执行低优先级,后台或需要不停处理的功能代码。

—测试系统处理裕量

—将处理器配置到低功耗模式——提供一种自动省电方法,使得在没有任何应用功能需要处理的时候,系统自动进入省电模式。

空闲任务钩子函数必须遵从以下规则
(1)绝不能阻塞或挂起。

空闲任务只会在其它任务都不运行时才会被执行(除非有应用任务共享空闲任务优先级)。

以任何方式阻塞空闲任务都可能导致没有任务能够进入运行态!
(2)如果应用程序用到了vTaskDelete() 函数,则空闲钩子函数必须能够尽快返回。

因为在任务被删除后,空闲任务负责回收内核资源。如果空闲任务一直运行在钩子函数中,则无法进行回收工作。

5.任务的调度
优先级抢占式调度
每个任务都被赋予了一个优先级,这个优先级不能被内核本身改变(只能被任务修改)。

”抢占式”是指当任务进入就绪态或是优先级被改变时,如果处于运行态的任务优先级更高,则该任务总是抢占当前运行的任务

单调速率调度

单调速率调度(Rate Monotonic Scheduling, RMS)是一种常用的优先级分配技术。其根据任务周期性执行的速率来分配一个唯一的优先级。

具有最高周期执行频率的任务赋予高最优先级;具有最低周期执行频率的任务赋予最低优先级。这种优先级分配方式被证明了可以最大化整个应用程序的可调度性(schedulability),但是运行时间不定以及并非所有任务都具有周期性,会使得对这种方式的全面计算变得相当复杂

协作式调度

只可能在运行态任务进入阻塞态或是运行态任务显式调用 taskYIELD()时,才会进行上下文切换。

任务永远不会被抢占,而具有相同优先级的任务也不会自动共享处理器时间。协作式调度的这作工作方式虽然比较简单,但可能会导致系统响应不够快

混合调度
这需要在中断服务例程中显式地进行上下文切换,从而允许同步事件产生抢占行为,但时间事件却不行。

这样做的结果是得到了一个没有时间片机制的抢占式系统。或许这正是所期望的,因为获得了效率,并且这也是一种常用的调度器配置

6.任务调度举例
两个优先级相同的任务时间片执行图如下:
在这里插入图片描述
图 优先级相同的两个任务的时间片执行序列

加入了tick中断及tick中断调度器调度下个任务图示如下:
在这里插入图片描述图 对执行流程进行扩展以显示tick中断的执行

由于task2比task1优先级高,因此task1永远得不到执行:
在这里插入图片描述图 Task2比Task1优先级高时的任务运行序列

用vTaskDelay()代替空循环后的执行流程,task1得到了执行,如下图:
在这里插入图片描述

图 用vTaskDelay()代替空循环后的执行流程

continuous 1与continuous 2具有优先级1,periodic具有优先级2,periodic使用vTaskDelayUntil,调度图如下(注意心跳中断在t5时刻实际是时间片到时调度器执行):
在这里插入图片描述

开始任务1具有最高优先级,任务1结束前将任务2设为最高优先级,任务2在结束时将自身设为最低优先级,执行序列如下:
在这里插入图片描述

idle在任务1进入阻塞态后,会执行对任务2的内存清除工作
在这里插入图片描述

三个任务(事件唤醒)的抢占执行过程
在这里插入图片描述

  1. 一些重要的API
    API 含义 重要参数说明 返回值
    void ATaskFunction( void *pvParameters ); 任务的原型 参数必须是void * 返回值必须是void
    portBASE_TYPE xTaskCreate( pdTASK_CODE pvTaskCode,
      const signed portCHAR * const pcName,
      unsigned portSHORT usStackDepth,
      void *pvParameters,
      unsigned portBASE_TYPE uxPriority,
      xTaskHandle *pxCreatedTask );
    创建任务
    uxPriority
    表示任务的优先级,最好设置成实际任务需要的最小值,以避免浪费

void vTaskStartScheduler( void )
开始调度一个任务执行,第一次调度时会自动创建一个空闲任务

void vTaskDelete( TaskHandle_t xTaskToDelete )
删除一个任务
xTaskToDelete
要删除任务的句柄

void vTaskDelay( const TickType_t xTicksToDelay )
让当前任务进入阻塞态,并保持多少个tick
xTicksToDelay
用来指定任务在调用vTaskDelay()到切出阻塞态整个过程包含多少个tick

void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime,
const TickType_t xTimeIncrement )
用于实现一个固定执行周期的需求求(当你需要让你的任务以固定频率周期性执行的时候)
pxPreviousWakeTime
保存了任务上一次离开阻塞态(被唤醒)的时刻这个时刻被用作一个参考点来计算该任务下一次离开阻塞态的时刻。pxPreviousWakeTime 指向的变量值会在API 函数vTaskDelayUntil()调用过程中自动更新,应用程序除了该变量第一次初始化外(通过调用xTaskGetTickCount()初始化),通常都不要修改它的值

xTimeIncrement
此参数命名时同样是假定vTaskDelayUntil()用于实现某个任务以固定频率周期性执行 —— 这个频率就是由xTimeIncrement 指定的。xTimeIncrement 的单位是心跳周期, 可以使用常量

portTICK_RATE_MS 将毫秒转换为心跳周期。

void vApplicationIdleHook( void )
空闲钩子函数
void vTaskPrioritySet( xTaskHandle pxTask,
    unsigned portBASE_TYPE uxNewPriority )
改变任务优先级
pxTask
被修改优先级的任务句柄(即目标任务)

uxNewPriority
目标任务将被设置到哪个优先级上。如果设置的值超过了最大可用优
先级(configMAX_PRIORITIES – 1),则会被自动封顶为最大值。常
量 configMAX_PRIORITIES 是在 FreeRTOSConfig.h 头文件中设置
的一个编译时选项。

unsigned portBASE_TYPE uxTaskPriorityGet( xTaskHandle pxTask );
用于查询一个任务的优先级
pxTask
被查询任务的句柄(目标任务)

返回值 被查询任务的当前优先级
void vTaskDelete( xTaskHandle pxTaskToDelete )
删除任务
pxTaskToDelete
被删除任务的句柄(目标任务) 。任务可以通过传入 NULL 值来删除自己

taskYIELD
调用此函数的任务转入就绪态,同时另一个任务转入运行态
7.参考文档
[1] FreeRTOS中文实用教程

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

freeRTOS中文实用教程1 的相关文章

  • FreeRTOS例程4-串口DMA收发不定长数据

    FreeRTOS例程4 串口DMA收发不定长数据 知乎 zhihu com
  • FreeRTOS config开始的宏

    FreeRTOSConfig h系统配置文件中可以自定义 FreeRTOS h中定义默认值 configAPPLICATION ALLOCATED HEAP 默认情况下FreeRTOS的堆内存是由编译器来分配的 将宏configAPPLIC
  • STM32CubeMX+FreeRTOS学习笔记(一)

    嵌入式实时操作系统FreeRTOS 基本概述 在嵌入式领域当中 实时操作系统的应用越来越广泛了 目前嵌入式操作系统种类很多 例如 Clinux C OS II C OS III FreeRTOS RT Thread等等 这篇文章所记录的就是
  • FreeRTOS-内核控制函数

    FreeRTOS 内核控制函数 FreeRTOS中有一些内核函数 一般来说这些内核函数在应用层不会使用 但是内核控制函数是理解FreeRTOS中断的基础 接下来我们逐一分析这些内核函数 taskYIELD 该函数的作用是进行任务切换 这是一
  • 解决错误“ #error “include FreeRTOS.h“ must appear in source files before “include event_groups.““例子分享

    今天来给大家分享一下 关于之前自己在学习FreeRTOS过程中遇到的一个错误提示 话不多说 我们直接来看 错误分析 首先 我们看一下错误的提示 error 35 error directive include FreeRTOS h must
  • FreeRTOS学习笔记 6 - 互斥量

    目录 1 创建 2 获取 3 释放 4 测试 FreeRTOS不支持调度方式的设置 所以下面2个宏定义可以随意设置值 define RTOS IPC FLAG FIFO 0x00 define RTOS IPC FLAG PRIO 0x01
  • 一文教你学会keil软件仿真

    仿真在我们调试代码中是非常重要的 通过仿真 我们可以快速定位到错误代码 或者错误逻辑的地方 这里我就以上一篇博客为例 教大家如何软件仿真 软件仿真不需要单片机 直接通过keil软件进行代码调试 一 打开工具 二 选择软件仿真 三 开始仿真
  • freertos---软定时器

    一 软件定时器介绍 freeRTOS软件定时器的时基是基于系统时钟节拍实现的 可以创建很多个 在硬件定时器资源不充足的情况下非常有用 软件定时器一般用作周期性地执行函数 在创建软件定时器时指定软件定时器的回调函数 在回调函数中实现相应的功能
  • FreeRTOS+CubeMX系列第一篇——初识FreeRTOS

    文章目录 一 关于FreeRTOS 二 FreeRTOS的特点 三 如何在CubeMX上配置FreeRTOS 四 FreeRTOS文档资料 五 同系列博客 一 关于FreeRTOS 1 什么是FreeRTOS FreeRTOS是一个迷你的实
  • FreeRTOS临界区

    FreeRTOS临界区是指那些必须完整运行 不能被打断的代码段 比如有的外设的初始化需要严格的时序 初始化过程中不能被打断 FreeRTOS 在进入临界区代码的时候需要关闭中断 当处理完临界区代码以后再打开中断 FreeRTOS 系统本身就
  • 【FreeRTOS(三)】任务状态

    文章目录 任务状态 任务挂起 vTaskSuspend 取消任务挂起 vTaskResume 挂起任务调度器 vTaskSuspendAll 取消挂起任务调度器 xTaskResumeAll 代码示例 任务挂起 取消任务挂起 代码示例 挂起
  • FreeRTOS临界段和开关中断

    http blog sina com cn s blog 98ee3a930102wg5u html 本章教程为大家讲解两个重要的概念 FreeRTOS的临界段和开关中断 本章教程配套的例子含Cortex M3内核的STM32F103和Co
  • stm32f103zet6移植标准库的sdio驱动

    sdio移植 st官网给的标准库有给一个用于st出的评估板的sdio外设实现 但一是文件结构有点复杂 二是相比于国内正点原子和野火的板子也有点不同 因此还是需要移植下才能使用 当然也可以直接使用正点原子或野火提供的实例 但为了熟悉下sdio
  • FreeRTOS轻量级同步--任务通知

    1 简介 在FreeRTOS的配置参数中的configUSE TASK NOTIFICATIONS宏打开 一般RTOS会默认打开 如图1所示 图1 notify宏开关 RTOS在创建任务时 会创建一个32位的通知值ulNotifiedVal
  • FreeRTOS实时操作系统(三)任务挂起与恢复

    系列文章 FreeRTOS实时操作系统 一 RTOS的基本概念 FreeRTOS实时操作系统 二 任务创建与任务删除 HAL库 FreeRTOS实时操作系统 三 任务挂起与恢复 FreeRTOS实时操作系统 四 中断任务管理 FreeRTO
  • FreeRTOS多任务调度器基础

    Cortex M4中SysTick调度器核心 Cortex M4中的中断管理 Cortex M4中影子栈指针 Cortex M4中SVC和PendSV异常 1 Cortex M4中SysTick调度器核心 systick每一次中断都会触发内
  • 当一个任务写入变量而其他任务读取该变量时,我们是否需要信号量?

    我正在研究 freeRtos 并且我有一个名为 x 的变量 现在 每秒只有一个任务正在写入该变量 而其他任务正在读取该变量值 我需要用互斥锁来保护变量吗 如果变量为 32 位或更小 并且其值是独立的并且不与任何其他变量一起解释 则不需要互斥
  • 如何将 void* 转换为函数指针?

    我在 FreeRTOS 中使用 xTaskCreate 其第四个参数 void const 是传递给新线程调用的函数的参数 void connect to foo void const task params void on connect
  • FreeRTOS 匈牙利表示法 [重复]

    这个问题在这里已经有答案了 我是 RTOS 和 C 编程的新手 而且我仍在习惯 C 的良好实践 因此 我打开了一个使用 FreeRTOS 的项目 我注意到操作系统文件使用匈牙利表示法 我知道一点符号 但面临一些新的 标准 FreeRTOS
  • 小型 ARM 微控制器的 RTOS 内核之间的可量化差异 [关闭]

    Closed 这个问题是基于意见的 help closed questions 目前不接受答案 有许多不同的 RTOS 可用于微控制器 我专门寻找支持 ARM Cortex M 处理器的 RTOS 另外 我对闭源解决方案不感兴趣 试图从网站

随机推荐