时间管理在操作系统内核中占有非常重要的地位,操作系统内核中有大量基于时间驱动的功能。有些任务是需要周期执行,比如一个软件定时器需要一秒钟周期性运行100次;有些功能任务需要延时一段时间后再运行,比如一个读取操作需要延时2000ms;比如操作系统内核也需要对系统运行时间进行计算,同时统计不同的任务运行时间的情况。
绝大多数操作系统内核需要一个周期性的时钟源,这个时钟源称之为时钟节拍。为了生成时钟节拍,往往需要使用一个硬件定时器,配置硬件定时器产生一个频率为10~1000Hz之间的中断,时钟节拍以某种频率周期性触发中断。当时钟中断发生时,中断服务程序调用操作系统内核中一个特殊程序:系统时钟节拍服务。
操作系统内核必须在硬件的帮助下才能计算和管理时间。时钟节拍中断并不是必须由硬件定时器产生,也可以由其他周期性时钟来源,如电力系统种频率为50Hz的时钟源。
系统时钟节拍可以设置成10Hz,也可以设置成1000Hz。时钟节拍值越大意味着硬件定时器产生的中断越频繁,系统时钟节拍服务执行得就越频繁。
高时钟节拍的优势:
1、提高操作系统内核时间管理精度2、提高任务抢占准确度
比如10Hz的时钟节拍的执行粒度为100ms,系统中的周期性事件最快为100ms一次,不可能由更高的精度了。比如1000Hz的时钟节拍,此时的执行粒度就提高了100倍,此时系统中的周期性事件最快为1ms一次,时间精度可以达到1ms。
高时钟节拍的优劣势:时钟节拍越高,意味着时钟中断越频繁,处理器需要花时间执行中断程序,如果时钟节拍大到一定程度比如1MHz,此时处理器将会一直周期性的执行时钟中断程序,而用户程序将几乎得不到执行。
2.系统时钟节拍服务实现系统时钟节拍的实现依靠以下两个部分:
1、硬件定时器2、系统时钟节拍服务
硬件定时器用于产生中断,配置硬件定时器产生一个频率为10~1000Hz之间的中断,硬件定时器周期性触发中断,当定时器中断发生时处理执行定时器中断程序。
系统时钟节拍服务用于执行系统时间管理相关操作,系统时钟节拍服务被定时器中断程序调用。系统时钟节拍服务完成了以下操作:1、更新系统节拍时间。2、更新等待表和就绪表。3、处理时间片轮询。4、切换任务。
更新系统节拍时间每次定时器中断调用系统时钟节拍服务时,完成系统节拍时间更新,将系统节拍时间计数值加1,通常情况下系统节拍时间计数值为一个32位的变量,时间计数值自加时需要考虑值溢出情况。
更新等待表和就绪表每次进入系统时钟节拍服务更新系统节拍时间后,操作系统内核检查等待表是否有任务完成等待,如果有任务完成等待,操作系统内核会将任务从等待表中移除,并将该任务添加到就绪表中,完成更新等待表和就绪表。
处理时间片轮询每次进入系统时钟节拍服务,操作系统内核会对当前运行优先级中的多个任务进行时间片轮询操作。
切换任务每次进入系统时钟节拍服务等待表和就绪表更新后,若有更高优先级任务就绪,操作系统内核将启动任务切换。
3.FreeRTOS系统时钟节拍实现分析对象为:cortex-m4硬件平台,FreeRTOS操作系统。系统时钟节拍实现依靠cortex-m4中的SysTick硬件定时器产生时钟中断,时钟中断服务为:
SysTick_Handler
FreeRTOS操作系统中的系统时钟节拍服务为
xTaskIncrementTick
SysTick定时器配置成1ms中断,配置函数实现如下:
HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq()/8000);HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK_DIV8);
定时器中断程序SysTick_Handler实现如下:
#define xPortSysTickHandler SysTick_Handlervoid xPortSysTickHandler( void ){vPortRaisebasePRI();{if( xTaskIncrementTick() != pdFALSE ){portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT;}}vPortClearbasePRIFromISR();}
xTaskIncrementTick为系统时钟节拍服务,完成更新系统节拍时间,更新等待表和就绪表,处理时间片轮询,切换任务,代码实现如下:
baseType_t xTaskIncrementTick( void ){TCB_t * pxTCB;TickType_t xItemValue;baseType_t xSwitchRequired = pdFALSE;traceTASK_INCREMENT_TICK( xTickCount );if( uxSchedulerSuspended == ( UbaseType_t ) pdFALSE ){const TickType_t xConstTickCount = xTickCount ( TickType_t ) 1;xTickCount = xConstTickCount;if( xConstTickCount == ( TickType_t ) 0U ) {taskSWITCH_DELAYED_LISTS();}else{mtCOVERAGE_TEST_MARKER();}if( xConstTickCount >= xNextTaskUnblockTime ){for( ;; ){if( listLIST_IS_EMPTY( pxDelayedTaskList ) != pdFALSE ){xNextTaskUnblockTime = portMAX_DELAY; break;}else{pxTCB = listGET_OWNER_OF_HEAD_ENTRY( pxDelayedTaskList );xItemValue = listGET_LIST_ITEM_VALUE( &( pxTCB->xStateListItem ) );if( xConstTickCount < xItemValue ){xNextTaskUnblockTime = xItemValue;break;}else{mtCOVERAGE_TEST_MARKER();}( void ) uxListRemove( &( pxTCB->xStateListItem ) );if( listLIST_ITEM_ConTAINER( &( pxTCB->xEventListItem ) ) != NULL ){( void ) uxListRemove( &( pxTCB->xEventListItem ) );}else{mtCOVERAGE_TEST_MARKER();}prvAddTaskToReadyList( pxTCB );#if ( configUSE_PREEMPTION == 1 ){if( pxTCB->uxPriority >= pxCurrentTCB->uxPriority ){xSwitchRequired = pdTRUE;}else{mtCOVERAGE_TEST_MARKER();}}#endif }}}#if ( ( configUSE_PREEMPTION == 1 ) && ( configUSE_TIME_SLICING == 1 ) ){if( listCURRENT_LIST_LENGTH( &( pxReadyTasksLists[ pxCurrentTCB->uxPriority ] ) ) > ( UbaseType_t ) 1 ){xSwitchRequired = pdTRUE;}else{mtCOVERAGE_TEST_MARKER();}}#endif #if ( configUSE_TICK_HOOK == 1 ){if( uxPendedTicks == ( UbaseType_t ) 0U ){vApplicationTickHook();}else{mtCOVERAGE_TEST_MARKER();}}#endif }else{ uxPendedTicks;#if ( configUSE_TICK_HOOK == 1 ){vApplicationTickHook();}#endif}#if ( configUSE_PREEMPTION == 1 ){if( xYieldPending != pdFALSE ){xSwitchRequired = pdTRUE;}else{mtCOVERAGE_TEST_MARKER();}}#endif return xSwitchRequired;}
未完待续…实时操作系统系列将持续更新创作不易希望朋友们点赞,转发,评论,关注。您的点赞,转发,评论,关注将是我持续更新的动力作者:李巍Github:liyinuoman2017
,