freertos任务切换xPortPendSVHandler

freertos任务切换xPortPendSVHandler

2018年11月27日 11:59:22 heima6056 阅读数:37

FreeRTOS版本:FreeRTOS V8.2.3

 
  1. void xPortPendSVHandler( void )

  2. {

  3. /* This is a naked function. */

  4. /*__asm__(汇编语句模板: 输出部分: 输入部分: 破坏描述部分)

  5. /*共四个部分:汇编语句模板,输出部分,输入部分,破坏描述部分,各部分使用":"格开,

  6. /*汇编语句模板必不可少,其他三部分可选,如果使用了后面的部分,而前面部分为空,

  7. /*也需要用":"格开,相应部分内容为空. __asm volatile("...." : : "i"());破坏描述部分没

  8. /*用,i表示立即数,官方文档网址

  9.      /*https://gcc.gnu.org/onlinedocs/gcc/Extended-Asm.html

  10. */

  11. __asm volatile

  12. (

  13. " mrs r0, psp \n" //将psp值放到r0,此时sp得值为msp

  14. " isb \n"

  15. " \n"

  16. " ldr r3, pxCurrentTCBConst \n" /* Get the location of the current TCB. */

  17. //将pxCurrentTCBConst标签(指针变量pxCurrentTCB的内存地址为&pxCurrentTCB或者到map看)放到r3,此时r3=&pxCurrentTCB

  18. " ldr r2, [r3] \n" //将r3寄存器值作为指针取内容存到r2,此时r2保存的为任务控制块首地址

  19. " \n"

  20. " stmdb r0!, {r4-r11} \n" /* Save the remaining registers. */

  21. //r0寄存器地址减去4在将r11-r4依次入栈r0(此时r0为程序栈psp地址)-4地址后

  22. " str r0, [r2] \n" /* Save the new top of stack into the first member of the TCB. */

  23. //将r0(psp)值放到r2内容(pxCurrentTCBConst)所指的地址

  24. " \n"

  25. " stmdb sp!, {r3, r14} \n" //sp寄存器地址减去4在将r14,r3依次入栈sp-4地址后

  26. " mov r0, %0 \n" //获取中断优先级到r0,%0对应后面的configMAX_SYSCALL_INTERRUPT_PRIORITY

  27. " msr basepri, r0 \n" //屏蔽低于优先级的中断

  28. " bl vTaskSwitchContext \n" //跳到任务切换c函数,找到下一个任务控制块

  29. " mov r0, #0 \n"

  30. " msr basepri, r0 \n" //开中断

  31. " dmia sp!, {r3, r14} \n" //从sp中取出r3,r14

  32. " \n" /* Restore the context, including the critical nesting count. */

  33. " ldr r1, [r3] \n" //将r3(存放的是任务控制块指针变量的地址&pxCurrentTCB)寄存器内容作为地址取内容放到r1(此时r1为新的任务控制块地址)

  34. " ldr r0, [r1] \n" /* The first item in pxCurrentTCB is the task top of stack. */

  35. //将r1寄存器内容作为地址取内容放到r0(此时r0为新任务的pxTopOfStack)

  36. " ldmia r0!, {r4-r11} \n" /* Pop the registers. */

  37. //将新的任务堆栈顶指针出堆栈到r4-r11

  38. " msr psp, r0 \n" //将新的任务堆栈顶指针放到psp

  39. " isb \n"

  40. " bx r14 \n" //之后硬件会自动把PC指针出堆栈(因为此时psp为新任务的堆栈顶指针所以出堆栈也是新任务的寄存器)

  41. " \n"

  42. " .align 2 \n"

  43. "pxCurrentTCBConst: .word pxCurrentTCB \n"

  44. ::"i"(configMAX_SYSCALL_INTERRUPT_PRIORITY)

  45. );

  46. }

              图1 两个任务切换时堆栈存储分布情况

               图2 由任务2(task2)切换到任务1(idle)寄存器及内存变化

其中memory1列的是栈顶指针开始往上增长对应图1就是压栈方向存储情况,结合图1可以看出xPSR、PC指针的值,切换的过程是先将任务2的堆栈保存,找到要切换的任务1的任务控制块,将堆栈指针更新到psp中,然后出栈时,硬件会自动出栈R0-R3、R12、LR、PC、xPSR寄存器(此时出栈寄存器psp内容已经是任务1的内容了)任务切换完成后出栈PC指针就更新为切换后的任务函数值。

  • 7
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值