FreeRTOS系列第29篇---FreeRTOS空闲任务分析
ID:技术让梦想更伟大
整理:李肖遥
当RTOS调度器开始工作后,为了保证至少有一个任务在运行,空闲任务被自动创建,占用最低优先级(0优先级)。
xReturn = xTaskCreate( prvIdleTask,
"IDLE",configMINIMAL_STACK_SIZE,
(void * ) NULL,
(tskIDLE_PRIORITY | portPRIVILEGE_BIT ),
&xIdleTaskHandle);
空闲任务是FreeRTOS不可缺少的任务,因为FreeRTOS设计要求必须至少有一个任务处于运行状态。我们来看一下空闲任务要做的工作。
1.释放内存
从V9.0版本开始,如果一个任务删除另外一个任务,被删除任务的堆栈和TCB立即释放。
如果一个任务删除自己,则任务的堆栈和TCB和以前一样,通过空闲任务删除。
所以空闲任务开始就会检查是否有任务删除了自己,如果有的话,空闲任务负责删除这个任务的TCB和堆栈空间。
2. 处理空闲优先级任务
当使用抢占式内核,相同优先级的任务使用时间片方式获得CPU权限。
如果有任务与空闲任务共享一个优先级,并且宏configIDLE_SHOULD_YIELD
设置为1,那么空闲任务不必等到时间片耗尽再进行任务切换。
所以空闲任务检查空闲优先级下的就绪列表中是否有多个任务,有的话则执行任务切换,让用户任务获得CPU权限。
宏configIDLE_SHOULD_YIELD
控制任务在空闲优先级中的行为。仅在满足下列条件后,才会起作用。
使用抢占式内核调度 用户任务使用空闲优先级。
通过时间片共享同一个优先级的多个任务,如果共享的优先级大于空闲优先级,并假设没有更高优先级任务,这些任务应该获得相同的处理器时间。
但如果共享空闲优先级时,情况会稍微有些不同。当configIDLE_SHOULD_YIELD
为1时,其它共享空闲优先级的用户任务就绪时,空闲任务立刻让出CPU,用户任务运行,这样确保了能最快响应用户任务。
处于这种模式下也会有不良效果(取决于你的程序需要),描述如下:
图中描述了四个处于空闲优先级的任务,任务A、B和C是用户任务,任务I是空闲任务。
上下文切换周期性的发生在T0、T1…T6时刻。
当用户任务运行时,空闲任务立刻让出CPU,但是,空闲任务已经消耗了当前时间片中的一定时间。
这样的结果就是空闲任务I和用户任务A共享一个时间片。
用户任务B和用户任务C因此获得了比用户任务A更多的处理器时间。
「可以通过下面方法避免:」
如果合适的话,将处于空闲优先级的各单独的任务放置到空闲钩子函数中; 创建的用户任务优先级大于空闲优先级; 设置 IDLE_SHOULD_YIELD
为0;
设置configIDLE_SHOULD_YIELD
为0将阻止空闲任务为用户任务让出CPU,直到空闲任务的时间片结束。
这确保所有处在空闲优先级的任务分配到相同多的处理器时间,但是,这是以分配给空闲任务更高比例的处理器时间为代价的。
3.执行空闲任务钩子函数
空闲任务钩子是一个函数,这个函数由用户来实现,RTOS规定了函数的名字和参数,这个函数在每个空闲任务周期都会被调用。
「要创建一个空闲钩子」:
设置 FreeRTOSConfig.h
文件中的configUSE_IDLE_HOOK
为1;定义一个函数,函数名和参数如下所示:
void vApplicationIdleHook(void );
这个钩子函数不可以调用会引起空闲任务阻塞的API函数(例如:vTaskDelay()
、带有阻塞时间的队列和信号量函数),在钩子函数内部使用协程是被允许的。
使用空闲钩子函数设置CPU进入省电模式是很常见的。
4.低功耗tickless模式
通常情况下,FreeRTOS回调空闲任务钩子函数(需要设计者自己实现),在空闲任务钩子函数中设置微处理器进入低功耗模式来达到省电的目的。
因为系统要响应系统节拍中断事件,因此使用这种方法会周期性的退出、再进入低功耗状态。
如果系统节拍中断频率过快,则大部分电能和CPU时间会消耗在进入和退出低功耗状态上。
FreeRTOS的tickless
空闲模式会在空闲周期时停止周期性系统节拍中断。
停止周期性系统节拍中断可以使微控制器长时间处于低功耗模式。
移植层需要配置外部唤醒中断,当唤醒事件到来时,将微控制器从低功耗模式唤醒。微控制器唤醒后,会重新使能系统节拍中断。
由于微控制器在进入低功耗后,系统节拍计数器是停止的,但我们又需要知道这段时间能折算成多少次系统节拍中断周期;
这就需要有一个不受低功耗影响的外部时钟源,即微处理器处于低功耗模式时它也在计时的;
这样在重启系统节拍中断时就可以根据这个外部计时器计算出一个调整值并写入RTOS 系统节拍计数器变量中。
空闲任务的源代码如下所示,其中宏portTASK_FUNCTION
翻译出来为:void prvIdleTask(void * pvParameters)
。
static portTASK_FUNCTION( prvIdleTask,pvParameters )
{
/*防止编译器警告 */
(void ) pvParameters;
for(;; )
{
/*检查是否有任务删除了自己,如果有的话,空闲任务负责删除这个任务的TCB和堆栈空间 */
prvCheckTasksWaitingTermination();
#if( configUSE_PREEMPTION == 0 )
{
/*如果我们没有使用抢占式调度,我们会强制任务切换,看看是否有其它任务变得有效.
如果使用抢占式调度,是不需要这样的,因为任务变得有效后会抢占空闲任务.*/
taskYIELD();
}
#endif/* configUSE_PREEMPTION */
#if( ( configUSE_PREEMPTION == 1 ) && ( configIDLE_SHOULD_YIELD == 1 ) )
{
/* 当使用抢占式内核,相同优先级的任务使用时间片方式获得CPU权限.如果有任务与空闲
任务共享一个优先级,那么空闲任务不必等到时间片耗尽再进行任务切换.
如果空闲优先级下的就绪列表中有多个任务,则执行用户任务*/
if(listCURRENT_LIST_LENGTH( &( pxReadyTasksLists[ tskIDLE_PRIORITY ] ) ) >( UBaseType_t ) 1 )
{
taskYIELD();
}
}
#endif/* ( ( configUSE_PREEMPTION == 1 ) && ( configIDLE_SHOULD_YIELD == 1 )) */
#if( configUSE_IDLE_HOOK == 1 )
{
externvoid vApplicationIdleHook( void );
/*调用用户定义函数.这样允许设计者在不增加任务开销的情况下实现后台功能
注意:这个函数中绝对不允许调用任务可能引起阻塞的函数.*/
vApplicationIdleHook();
}
#endif/* configUSE_IDLE_HOOK */
#if( configUSE_TICKLESS_IDLE != 0 )
{
TickType_txExpectedIdleTime;
/*如果每次执行空闲任务都挂起调度器,然后再解除调度器,这很难让人满意,因此这里
执行两次同样的比较(xExpectedIdleTime和configEXPECTED_IDLE_TIME_BEFORE_SLEEP),
第一次比较是测试一下是否达到预期的空闲时间,并不会挂起调度器.*/
xExpectedIdleTime= prvGetExpectedIdleTime();
if(xExpectedIdleTime >= configEXPECTED_IDLE_TIME_BEFORE_SLEEP )
{
vTaskSuspendAll();
{
/*现在调度器被挂起,需要再次采样空闲时间,这次空闲时间可以使用了*/
configASSERT(xNextTaskUnblockTime >= xTickCount );
xExpectedIdleTime= prvGetExpectedIdleTime();
if(xExpectedIdleTime >= configEXPECTED_IDLE_TIME_BEFORE_SLEEP )
{
portSUPPRESS_TICKS_AND_SLEEP(xExpectedIdleTime );
}
}
(void ) xTaskResumeAll();
}
}
#endif/* configUSE_TICKLESS_IDLE */
}
}
‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧ END ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧ 关注我的微信公众号,回复“加群”按规则加入技术交流群。
点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。