FreeRTOS实现调度器

1,实现调度器


在task.c中定义调度器启动函数:

这里面涉及任务优先级之类的,目前还未涉及优先级,可以自己手写一个手动指定任务:

这个Task1TCB是后面定义第一个任务的TCB,现在还没定义。这个任务调度器启动函数使用了xPortStartScheduler函数,具体定义如下;

这个函数功能就是把cortex m内核的SCB中SHPR3寄存器设置Pendsv和systick中断优先级为最低,然后就调用prvStartFirstTask启动第一个任务:

该函数首先将当前栈按8字节对齐,然后初始化MSP主栈的地址(从SCB_VTOR寄存器取得,该寄存器存放的是向量表起始地址,即让MSP指向向量表起始地址)然后用CPS指令开启全局中断,CPS指令操作的是内核中PRIMASK和FAULTMASK的中断屏蔽寄存器,此外内核还有一个BASEPRI,有关这三个寄存器的功能具体参考cortex权威指南。接下来调用SVC产生SVC中断,在SVC中断服务函数里面真正的启动我们第一个函数。

 此函数将pxCurrentTCB指向的将要执行的任务栈中信息传入相应寄存器,然后更新psp指针指向r0,在SVC中断服务函数退出前向r14寄存器或上0x0d,让中断返回时后使用PSP栈指针(在中断退出前,内核会检查r14最后四位来判断中断返回后使用那个指针,通常情况下默认使用MSP),然后bx r14跳转,此时会自动根据栈指针(此时使用的是PSP)将r0—xPSR寄存器值更新,最后PSP指向任务栈栈底。此时,任务开始运行了。(当进入中断的时候,r14就不是保存返回的地址,而是保存异常中断模式将要返回的地址,包含了进入中断时使用的MSP还是PSP栈指针)。

2,任务切换


该函数触发Pendsv中断进入中断服务函数

该函数将当前运行的函数信息保存在栈中,然后进入临界段调用 vTaskSwitchContext切换pxCurrentTCB指向下一个将要运行函数,然后退出临界段将将要运行函数栈信息出栈恢复到相应寄存器,随后就会执行任务2(注:在进入vTaskSwitchContext时将当前r14和r3寄存器压入中断函数栈中,因为调用vTaskSwitchContext时会更新r14寄存器,而r3不确定会不会作为中间变量)

该函数是自己写的手动指向任务。 

总结,任务启动在SVC中断启动,任务切换在pendsv中断启动,任务切换要注意MSP和PSP指针的切换使用。

  • 5
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值