1. 多任务启动流程
多任务启动流程如下表所示
启动后以下各函数由上至下依次执行 | 含义 |
---|
osKernelStart() | 启动内核 |
vTaskStartScheduler() | 启动任务调度器 |
xPortStartScheduler() | 启动调度器 |
prvStartFirstTask() | 启动第一个任务 |
SVC | 调用SVC中断 |
2. 源码分析
void vTaskStartScheduler( void ){
BaseType_t xReturn;
#if(configSUPPORT_STATIC_ALLOCATION == 1){
}
#else{
xReturn = xTaskCreate(prvIdleTask,
"IDLE", configMINIMAL_STACK_SIZE,
(void *) NULL,
(tskIDLE_PRIORITY|portPRIVILEGE_BIT),
&xIdleTaskHandle);
}
#endif
if(xReturn == pdPASS){
portDISABLE_INTERRUPTS();
#define portMAX_DELAY ( TickType_t ) 0xffffffffUL
xNextTaskUnblockTime = portMAX_DELAY;
xSchedulerRunning = pdTRUE;
xTickCount = ( TickType_t ) 0U;
if(xPortStartScheduler() != pdFALSE){
}
else{
}
}
else{
}
}
- 启动调度器:FreeRTOS系统时钟是由滴答定时器来提供,任务切换也会用到PendSV中断,这些硬件的初始化在这里完成
BaseType_t xPortStartScheduler( void ){
portNVIC_SYSPRI2_REG |= portNVIC_PENDSV_PRI;
portNVIC_SYSPRI2_REG |= portNVIC_SYSTICK_PRI;
vPortSetupTimerInterrupt();
uxCriticalNesting = 0;
prvStartFirstTask();
return 0;
}
- 启动第一个任务:用于启动第一个任务,是一个汇编函数
__asm void prvStartFirstTask( void ){
PRESERVE8
ldr r0, =0xE000ED08
ldr r0, [r0]
ldr r0, [r0]
msr msp, r0
cpsie i
cpsie f
dsb
isb
svc 0
nop
nop
}
- 调用SVC中断:在prvStartFirstTask()中通过调用SVC指令触发了SVC中断,而第一个任务的启动就是在SVC中断服务函数中完成的
__asm void vPortSVCHandler(void){
PRESERVE8
ldr r3, =pxCurrentTCB
ldr r1, [r3]
ldr r0, [r1]
ldmia r0!, {r4-r11, r14}
msr psp, r0
isb
mov r0, #0
msr basepri, r0
bx r14
}
3. 任务切换
3.1 任务切换场合
RTOS系统的核心是任务管理,而任务管理的核心是任务切换,任务切换决定了任务的执行顺序。上下文(任务)切换被触发的场合可以是
- 系统滴答定时器(SysTick)中断
- 执行一个系统调用
典型的嵌入式OS系统中,处理器被划分为多个时间片。若系统中只有两个任务,这两个任务会交替执行,任务切换都是在SysTick中断中执行,如下图示:
在一些OS设计中,为了解决SysTick和IRQ的冲突问题,PendSV异常将上下文切换请求延迟到所有其他IRQ处理都已经完成后,在PendSV异常内执行上下文切换。如下图示:
PendSV(可挂起的系统调用)异常对OS操作非常重要,其优先级可通过编程设置。可通过将中断控制和状态寄存器ICSR的bit28(挂起位)置1来触发PendSV中断。上面提到过上下文切换被触发的两个场合:SysTick中断和执行一个系统调用,其源码分析如下:
void SysTick_Handler(void){
if(xTaskGetSchedulerState()!=taskSCHEDULER_NOT_STARTED){
xPortSysTickHandler();
}
}
void xPortSysTickHandler( void ){
vPortRaiseBASEPRI();
{
if( xTaskIncrementTick() != pdFALSE ){
portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT;
}
}
vPortClearBASEPRIFromISR();
}
#define taskYIELD() portYIELD()
#define portYIELD()
{
portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT;
__dsb( portSY_FULL_READ_WRITE );
__isb( portSY_FULL_READ_WRITE );
}
3.2 PendSV中断服务函数
FreeRTOS任务切换的具体过程是在PendSV中断服务函数中完成的,下面分析PendSV中断服务函数源码,看看切换过程是如何进行的
__asm void xPortPendSVHandler( void ){
extern uxCriticalNesting;
extern pxCurrentTCB;
extern vTaskSwitchContext;
PRESERVE8
mrs r0, psp
isb
ldr r3, =pxCurrentTCB
ldr r2, [r3]
tst r14, #0x10
it eq
vstmdbeq r0!, {s16-s31}
stmdb r0!, {r4-r11, r14}
str r0, [r2]
stmdb sp!, {r3}
mov r0, #configMAX_SYSCALL_INTERRUPT_PRIORITY
msr basepri, r0
dsb
isb
bl vTaskSwitchContext
mov r0, #0
msr basepri, r0
ldmia sp!, {r3}
ldr r1, [r3]
ldr r0, [r1]
ldmia r0!, {r4-r11, r14}
tst r14, #0x10
it eq
vldmiaeq r0!, {s16-s31}
msr psp, r0
isb
bx r14
nop
nop
}
在PendSV中断服务函数中有调用函数vTaskSwitchContext来获取下一个要运行的任务,其源码如下
void vTaskSwitchContext( void ){
if( uxSchedulerSuspended != ( UBaseType_t ) pdFALSE ){
xYieldPending = pdTRUE;
}
else{
xYieldPending = pdFALSE;
taskCHECK_FOR_STACK_OVERFLOW();
taskSELECT_HIGHEST_PRIORITY_TASK();
}
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)