任务切换,又称上下文切换(context switch)是实现操作系统的基础。
基于CM3实现任务切换必须要有以下2点概念认识
- PendSV
PendSV叫做可悬起系统调用,与之相对的叫做系统服务调用(SVC)。
两者的区别是SVC异常必须在执行SVC指令后立即得到响应,如果因为优先级或者其他原因无法立即响应,就会上访成硬fault。而PendSV则可以像普通中断一样被悬起(pending),从而等待其他优先级更高的中断被响应完毕后执行。
在实现任务切换的操作系统中,往往会把PendSV的中断优先级设为最低。这样在进行任务切换时,只需要将PendSV悬起寄存器置位即可。PendSV的中断处理子程序会在所有高优先级中断响应完后开始进行任务切换。
在ucos中任务切换代码十分简单:
OSCtxSw LDR R0, =NVIC_INT_CTRL LDR R1, =NVIC_PENDSVSET //trigger PendSV STR R1, [R0] BX LR
相对应的PendSV处理函数:
OS_PendSV_Handler CPSID I /* 寄存器保存和恢复 */ Save & Restore registers CPSIE I BX LR
- 堆栈操作
任务切换的本质是实现寄存器的保存和恢复。在CM3中通过堆栈操作能快速的进行任务切换,为OS节约下不小的开销。
1. 中断入栈
当CM3开始响应一个中断时,会硬件自动把xPSR、PC、LR、R12、R3 - R0压入堆栈中
2. 手动入栈
除了自动入栈的寄存器外,在任务切换中断子程序(PendSV ISR)中,我们还需要手动把R4 - R11压入堆栈中,从而实现完整现场保护。
旧SP(N - 0) 原先已压栈的内容(N - 4) xPSR(N - 8) PC(N - 12) LR(N - 16) R12(N - 20) R3(N - 24) R2(N - 28) R1(N - 32) R0--------------------------- 手动入栈(N - 36) R11(N - 40) R10(N - 44) R9(N - 48) R8(N - 52) R7(N - 56) R6(N - 60) R5(N - 64) R4 ---> 新SP
在ucos可以看到响应的实现
OS_PendSV_Handler: .......... //save SUBS R0, R0, #0x20 ;8 registers STM R0, {R4-R11} .......... //restore LDM R0, {R4-R11} ADDS R0, R0, #0x20