系统延时API详解
vTaskDelay()
vTaskDelayUntil()
xTaskGetTickCount()
相对延时与绝对延时的区别
系统延时函数实现原理
vTaskDelay
void vTaskDelay( const TickType_t xTicksToDelay )
{
//xAlreadyYielded:已经调度的状态
//初始赋值为0
BaseType_t xAlreadyYielded = pdFALSE;
/* 延时周期是否大于0,不大于0,就不应该调度 */
if( xTicksToDelay > ( TickType_t ) 0U )
{
configASSERT( uxSchedulerSuspended == 0 );
//1、挂起调度器
vTaskSuspendAll();
{
traceTASK_DELAY();
/* 1、添加任务到延时列表中
2、需要传入两个参数
2.1、xTicksToDelay:延时周期
2.2、pdFALSE:状态值为0
*/
prvAddCurrentTaskToDelayedList( xTicksToDelay, pdFALSE );
}
//恢复调度器,这个调度器是有返回值的,这返回值,表示在恢复调度器
的时候,是否已经进行了任务切换
xAlreadyYielded = xTaskResumeAll();
}
else
{
mtCOVERAGE_TEST_MARKER();
}
/* xAlreadyYielded 等于FALSE,代表在恢复调度器的时候,没有进行任务切换 */
if( xAlreadyYielded == pdFALSE )
{
//调用了任务切换:内部就是触发PendSV异常
portYIELD_WITHIN_API();
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
//添加任务到延时列表中
//传入两个参数
//xTicksToWait:延时周期
//xCanBlockIndefinitely :延时的确定状态
static void prvAddCurrentTaskToDelayedList( TickType_t xTicksToWait, const BaseType_t xCanBlockIndefinitely )
{
//延时周期----下次唤醒的时间
TickType_t xTimeToWake;
//xTickCount为系统节拍值 全局的,进行一次获取
const TickType_t xConstTickCount = xTickCount;
/* 把当前任务从就绪列表中移除 */
if( uxListRemove( &( pxCurrentTCB->xStateListItem ) ) == ( UBaseType_t ) 0 )
{
/* The current task must be in a ready list, so there is no need to
check, and the port reset macro can be called directly. */
portRESET_READY_PRIORITY( pxCurrentTCB->uxPriority, uxTopReadyPriority );
}
else
{
mtCOVERAGE_TEST_MARKER();
}
//是否使用了任务挂起的功能
#if ( INCLUDE_vTaskSuspend == 1 )
{
//portMAX_DELAY = 0XFFFFFFFF 表示延时是一直持续的,也就是让任务一直阻塞
if( ( xTicksToWait == portMAX_DELAY ) && ( xCanBlockIndefinitely != pdFALSE ) )
{
/* 把任务添加到,挂起列表中去*/
vListInsertEnd( &xSuspendedTaskList, &( pxCurrentTCB->xStateListItem ) );
}
else
{
/* 先去计算,下次唤醒的tick值 */
xTimeToWake = xConstTickCount + xTicksToWait;
/* 每个任务控制块里,状态列表都有一个延时值:value
这个value就是任务延时周期,在systick里面,进行比较,是否到达
*/
listSET_LIST_ITEM_VALUE( &( pxCurrentTCB->xStateListItem ), xTimeToWake );
//判断下次延时周期,是否小于系统节拍值,那就证明定时已经溢出
if( xTimeToWake < xConstTickCount )
{
/* 溢出,就把任务,添加到延时溢出列表里 */
vListInsert( pxOverflowDelayedTaskList, &( pxCurrentTCB->xStateListItem ) );
}
else
{
/* 没有溢出,把任务添加到延时列表中,让内核进行处理 */
vListInsert( pxDelayedTaskList, &( pxCurrentTCB->xStateListItem ) );
/*还要去更新系统时间片,因为系统时间片永远保存最小的延时周期 */
if( xTimeToWake < xNextTaskUnblockTime )
{
xNextTaskUnblockTime = xTimeToWake;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
}
}
#else /* INCLUDE_vTaskSuspend */
{
/* 计算下次唤醒的系统节拍值 */
xTimeToWake = xConstTickCount + xTicksToWait;
/* 赋值到任务控制块里 */
listSET_LIST_ITEM_VALUE( &( pxCurrentTCB->xStateListItem ), xTimeToWake );
if( xTimeToWake < xConstTickCount )
{
/* 溢出,添加到延时溢出列表中 */
vListInsert( pxOverflowDelayedTaskList, &( pxCurrentTCB->xStateListItem ) );
}
else
{
/* 没有溢出,添加到延时列表中 */
vListInsert( pxDelayedTaskList, &( pxCurrentTCB->xStateListItem ) );
/*更新时间片 */
if( xTimeToWake < xNextTaskUnblockTime )
{
xNextTaskUnblockTime = xTimeToWake;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
/* Avoid compiler warning when INCLUDE_vTaskSuspend is not 1. */
( void ) xCanBlockIndefinitely;
}
#endif /* INCLUDE_vTaskSuspend */
}
vTaskDelayUntil
void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime, const TickType_t xTimeIncrement )
{
//下次任务要唤醒的系统节拍值
TickType_t xTimeToWake;
//xAlreadyYielded:表示是否已经进行了任务切换
//xShouldDelay:表示是否需要进行延时处理
BaseType_t xAlreadyYielded, xShouldDelay = pdFALSE;
//挂起调度器
vTaskSuspendAll();
{
/* 获取系统节拍值 */
const TickType_t xConstTickCount = xTickCount;
/* 获取任务下次唤醒的系统节拍值 */
xTimeToWake = *pxPreviousWakeTime + xTimeIncrement;
//pxPreviousWakeTime指向上一次保存的任务的唤醒节拍值
//这个时候如果大于当前系统节拍值,无非两种可能
// 1、延时周期已经到达了
2、整个tick计数值,已经溢出了
if( xConstTickCount < *pxPreviousWakeTime )
{
/*1、下次要唤醒的系统节拍值小于上次要唤醒节拍值 ----- 这个就表示:系统节拍值计数溢出了
2、下次要唤醒的系统节拍值大于当前的系统节拍值 ----- 表示需要延时
*/
if( ( xTimeToWake < *pxPreviousWakeTime ) && ( xTimeToWake > xConstTickCount ) )
{
//标记需要延时
xShouldDelay = pdTRUE;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
else
{
/*
1、下次要唤醒的系统节拍值小于上次唤醒的系统节拍值 --- 证明系统节拍值溢出,需要进行延时
2、下次要唤醒的系统节拍值大于当前系统节拍值---------证明延时周期,在当前的时间之后
*/
if( ( xTimeToWake < *pxPreviousWakeTime ) || ( xTimeToWake > xConstTickCount ) )
{
//标记延时状态
xShouldDelay = pdTRUE;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
/* 保存下次唤醒的节拍值,所以传入pxPreviousWakeTime是指针类型 */
*pxPreviousWakeTime = xTimeToWake;
//判断是否需要延时
if( xShouldDelay != pdFALSE )
{
traceTASK_DELAY_UNTIL( xTimeToWake );
/* 添加任务到延时列表中去 */
prvAddCurrentTaskToDelayedList( xTimeToWake - xConstTickCount, pdFALSE );
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
//调度器恢复,如果调度器内部已经进行了任务切换,返回一个true
xAlreadyYielded = xTaskResumeAll();
/* 如果 调度器没有进行任务切换,那么要进行任务切换*/
if( xAlreadyYielded == pdFALSE )
{
//进行PendSV异常触发
portYIELD_WITHIN_API();
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
vTaskSuspendAll
void vTaskSuspendAll( void )
{
/* 调取记录值++
这个值是去让systick 中断产生了时候,不去遍历阻塞列表,进行任务恢复
*/
++uxSchedulerSuspended;
}
xTaskResumeAll
BaseType_t xTaskResumeAll( void )
{
TCB_t *pxTCB = NULL;
BaseType_t xAlreadyYielded = pdFALSE;
/* 进入临界段 */
taskENTER_CRITICAL();
{
//调度器记录值减一
--uxSchedulerSuspended;
//如果调度器需要恢复了
if( uxSchedulerSuspended == ( UBaseType_t ) pdFALSE )
{
//判断当前任务数量大于0
if( uxCurrentNumberOfTasks > ( UBaseType_t ) 0U )
{
/* 从挂起的就绪列表中遍历 */
while( listLIST_IS_EMPTY( &xPendingReadyList ) == pdFALSE )
{
//获取任务控制块
pxTCB = ( TCB_t * ) listGET_OWNER_OF_HEAD_ENTRY( ( &xPendingReadyList ) );
//移除 挂起就绪列表,移除事件列表
( void ) uxListRemove( &( pxTCB->xEventListItem ) );
( void ) uxListRemove( &( pxTCB->xStateListItem ) );
//添加到就绪列表中
prvAddTaskToReadyList( pxTCB );
/* 如果优先级大于当前任务优先级,则进行任务切换 */
if( pxTCB->uxPriority >= pxCurrentTCB->uxPriority )
{
xYieldPending = pdTRUE;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
//获取到任务控制块不为空
if( pxTCB != NULL )
{
/* 需要更新系统的时间片 */
prvResetNextTaskUnblockTime();
}
/* 获取 在调度器挂起时,systick挂起记录值 */
{
UBaseType_t uxPendedCounts = uxPendedTicks; /* Non-volatile copy. */
//如果记录值大于0
if( uxPendedCounts > ( UBaseType_t ) 0U )
{
do
{
//进行systick调度处理,其实就遍历阻塞列表,如果需要任务切换,返回true
if( xTaskIncrementTick() != pdFALSE )
{
//标记任务需要切换
xYieldPending = pdTRUE;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
--uxPendedCounts;
//一直遍历,直到uxPendedCounts = 0
} while( uxPendedCounts > ( UBaseType_t ) 0U );
//赋值为0
uxPendedTicks = 0;
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
//如果需要进行任务切换
if( xYieldPending != pdFALSE )
{
//判断是否内核是抢占式
#if( configUSE_PREEMPTION != 0 )
{
//标记已经调度的状态
xAlreadyYielded = pdTRUE;
}
#endif
//进行调度
taskYIELD_IF_USING_PREEMPTION();
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
}
else
{
mtCOVERAGE_TEST_MARKER();
}
}
//退出临界段
taskEXIT_CRITICAL();
//返回调度的状态值
return xAlreadyYielded;
}