FreeRTOS一般被用于硬件设计上RAM大小存在一定限制(成本、资源、性能功耗比等)的平台之中,它是目前运用的比较多的实时操作系统之一。选用FreeRTOS作为项目应用一般从以下几方面考虑:
基于以上几点特性,FreeRTOS广泛应用于对实时性有一定要求以及资源有一定的限制的系统中,在汽车行业、消费电子、网络服务等行业中的MCU芯片、DSP芯片以及定制化SOC芯片中被大量使用。
FreeRTOS上面的任务类似Linux上的进程,为实时操作系统上进行系统调度运行的单元。对于单核的CPU而言,任意FreeRTOS系统运行时刻,只有一个任务被执行。
FreeRTOS的任务主要分为以下几个状态:
任务正在使用处理器在执行相应的操作,此时任务被称作处于运行态。如果运行的硬件环境为单处理器环境,在一定的时间内只有一个任务在运行。
就绪任务为那些可以做被执行,但因有更高优先级或者同等优先级的任务正在执行,暂时未执行的任务。
如果任务正在等待定时周期运行或者外部事件、外部资源,则当前任务处于阻塞态。任务可以通过调用系统提供的vTaskDelay函数(或vTaskDelayUntil)进入阻塞状态,设置不一样的系统delay周期进行唤醒进入Ready态。另外任务也可以通过等待队列、信号量、事件组、通知或信号量事件等任务通信机制进入阻塞态,等待这些通信资源是可以设置自身的阻塞时间的,阻塞时间即可以以系统节拍作为最小单元设置timeout时间。Timeout超时时间过去,任务从Blocked态自动切换为Ready态。
任务的挂起态类似任务的阻塞态,与Blocked态不同的是,处于挂起态的任务不可以通过系统心跳周期唤醒运行,并且Suspended的任务不能以系统心跳作为最小单元设置挂起的超时时间,Suspended的任务不存在自动的从Suspended切换为ready态的情况,只能分别通过vTaskSuspend()和xTaskResume()API调用显式命令任务进入或退出挂起状态。等待任务通信资源的时候,若开启系统挂起功能(INCLUDE_vTaskSuspend 值为1)以及将等待时间设置为portMAX_DELAY,任务直接挂起直到资源满足进行唤醒。此外处于Ready态的任务是可以显示调用vTaskSuspend()让自身进入Suspended态,但是处于Ready的任务意味着等待资源满足,是不会直接切换为Block态。Blockd态一般是运行中的任务发现自身运行资源不满足进行等待资源或者主动进行系统delay才会进入的一个状态。
FreeRTOS任务间状态的联系可参考下图:
FreeRTOS任务调度方式采用的是抢占式调度与时间片轮转结合的方式。系统默认会将两种方式共同开启。不同优先级的任务采用的是抢占式调度,同等优先级的任务采用的是时间片轮转的方式调度。通过配置configUSE_TIME_SLICING的值,可以选择开启时间片轮转或者关闭时间片轮转。因此,FreeRTOS同时提供在开启时间片轮转时的有时间片的优先抢占式调度算法,在关闭时间片轮转时的无时间片的优先抢占式调度算法。此外,FreeRTOS提供taskYIELD()接口给在运行态任务进入阻塞态或运行态任务调用,进行请求重新调度。
FreeRTOS的任务调度过程主要为任务创建阶段、任务调度器的启动阶段以及任务调度器启动后的正常工作阶段,以下基于ARM的cortex-m4系列平台举例进行说明。
1. 任务创建阶段
FreeRTOS在芯片复位,BootLoader阶段加载完成代码,进入main函数,在main函数完成CPU资源、中断资源以及外设资源等配置之后,根据项目的任务规划创建相应的任务。FreeRTOS提供两种创建任务的方式与相应的API,为动态创建任务方式以及静态创建任务方式,一般而言,多数采用动态创建任务的方式进行任务的创建(以动态创建任务进行说明)。
参数:
返回值:
2. 任务调度开启阶段
FreeRTOS任务调度器开启依赖svc中断与vTaskStartScheduler函数,在cortex-m4系列芯片完成CPU、外设与中断等硬件资源初始化进入到主要函数main函数后,创建项目运行的所需任务,调用vTaskStartScheduler函数开启FreeRTOS任务调度(如下图所示)。
FreeRTOS在任务调度器的启动阶段,会默认自动创建idle任务,idle任务优先级默认为系统最低优先级,源码中会调用prvCheckTasksWaitingTermination函数进行删除任务的资源释放(TCB与任务栈)以及提供idle任务的钩子函数vApplicationIdleHook进行系统负荷的计算,或者使用者可在此钩子函数添加基于自身项目的监控以及feedback。Idle任务的任务入口为portTASK_FUNCTION,部分源码如下:
3. 任务调度运行阶段
基于ARM的cortex-m4系列平台芯片的FreeRTOS的系统中,systick中断、普通外部中断、SVC与pendSV中断与任务(线程)的优先级一般被设计如下:
systick中断 > 普通外部中断 > SVC与pendsv中断 > 任务(线程)
SVC中断多数在任务调度器启动阶段或者进行任务抢占时使用,作为启动任务调度器或者抢占任务。在任务调度器启动完成后,依赖systick中断进行周期性任务调度(时间片轮询或者抢占)以及pendSV中断进行任务的上下文切换与保存。下图以任务A与任务B在任务调度时,中间穿插systick中断以及普通外部中断进行实例说明。
FreeRTOS的任务堆栈总空间大小可以根据具体的项目资源进行配置,FreeRTOS根据具体项目中的RAM的大小规划任务堆栈的总空间,宏定义configTOTAL_HEAP_SIZE值决定总的任务堆的大小。每次创建一个新的任务都会在总的任务堆中申请对应创建的新任务的任务堆栈空间,任务总堆栈的使用如下所示。
同时配置configUSE_TRACE_FACILITY与configUSE_STATS_FORMATTING_FUNCTIONS的值为1会prvWriteNameToBuffer(),vTaskList()与vTaskGetRunTimeStats()三个函数,开启FreeRTOS的任务监控。此功能可以将当前FreeRTOS的存在的任务的任务状态、任务优先级以及每个任务的最大堆栈剩余空间进行反馈。FreeRTOS的源码中提供的是调用sprintf对这些数值的打印,部分源码如下:
官方的demo打印结果如下所示:
Name:任务名State:任务状态Priority:任务优先级Stack:剩余堆栈(可根据此值对总体的RAM空间进行优化)
FreeRTOS的任务机制使得它有很好的实时性,同时原生系统中的各个可裁剪的配置项很好的给不同的嵌入式提供因地制宜的适配,更加吸引各大厂商的是:系统商业免费开源。嵌入式应用中功耗、资源、成本等因素往往是各个厂商进行系统选择的重要依据。目前消费类电子的应用中,因考虑到功耗、资源、成本等,有时会采取大核CPU+小核CPU的架构。大核中往往会加载通用的操作系统(Linux、Android、IOS等),小核中越来越多的厂商考虑在上面搭载FreeRTOS,根据小核的不同应用场景进行裁剪适配使用。免费、开源、易裁剪以及占用资源少使得FreeRTOS有不输于其他RTOS的竞争力。
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8