1.前言
FreeRTOS是小型多任务嵌入式操作系统,硬实时性。本章主要讲述任务相关特性及调度相关的知识。
2. 任务的总体特点
- 任务的状态
(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周期
4. 空闲任务
- 空闲任务的特点
(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的内存清除工作
- 三个任务(事件唤醒)的抢占执行过程
6. 一些重要的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 ); |
创建任务 |
表示任务的优先级,最好设置成实际任务需要的最小值,以避免浪费 |
|
void vTaskStartScheduler( void ) |
开始调度一个任务执行,第一次调度时会自动创建一个空闲任务 |
|
|
void vTaskDelete( TaskHandle_t xTaskToDelete ) |
删除一个任务 |
要删除任务的句柄 |
|
void vTaskDelay( const TickType_t xTicksToDelay ) |
让当前任务进入阻塞态,并保持多少个tick |
用来指定任务在调用vTaskDelay()到切出阻塞态整个过程包含多少个tick |
|
void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime, |
用于实现一个固定执行周期的需求求(当你需要让你的任务以固定频率周期性执行的时候) |
保存了任务上一次离开阻塞态(被唤醒)的时刻这个时刻被用作一个参考点来计算该任务下一次离开阻塞态的时刻。pxPreviousWakeTime 指向的变量值会在API 函数vTaskDelayUntil()调用过程中自动更新,应用程序除了该变量第一次初始化外(通过调用xTaskGetTickCount()初始化),通常都不要修改它的值
此参数命名时同样是假定vTaskDelayUntil()用于实现某个任务以固定频率周期性执行 —— 这个频率就是由xTimeIncrement 指定的。xTimeIncrement 的单位是心跳周期, 可以使用常量 portTICK_RATE_MS 将毫秒转换为心跳周期。 |
|
void vApplicationIdleHook( void ) |
空闲钩子函数 | ||
void vTaskPrioritySet( xTaskHandle pxTask, |
改变任务优先级 |
被修改优先级的任务句柄(即目标任务)
目标任务将被设置到哪个优先级上。如果设置的值超过了最大可用优 |
|
unsigned portBASE_TYPE uxTaskPriorityGet( xTaskHandle pxTask ); |
用于查询一个任务的优先级 |
被查询任务的句柄(目标任务) |
返回值 被查询任务的当前优先级 |
void vTaskDelete( xTaskHandle pxTaskToDelete ) |
删除任务 |
被删除任务的句柄(目标任务) 。任务可以通过传入 NULL 值来删除自己 |
|
taskYIELD |
调用此函数的任务转入就绪态,同时另一个任务转入运行态 |
7.参考文档
[1] FreeRTOS中文实用教程