历史上的今天
今天是:2025年01月12日(星期日)
2021年01月12日 | 一步步写STM32 OS【一】 序言
2021-01-12 来源:eefocus
一直想写个类似uCOS的OS,近段时间考研复习之余忙里偷闲,总算有点成果了。言归正传,我觉得OS最难的部分首先便是上下文切换的问题,他和MCU的架构有关,所以对于不同的MCU,这部分需要移植。一旦这个问题解决了,整个OS算是成功了一半了,当然,是对于简单的OS。
好了,要写一个OS,首先需要一个开发板和仿真器。我的开发板是STM32F4DISCOVERY,自带ST-LINK V2仿真器,板载MCU为STM32F407VGT6,支持FPU,32位ARM Cortex-M4F核,1024KB FLASH,192 KB RAM,总之很强大。对STM32其他系列,本OS几乎不需修改修改就可使用。开发环境为IAR for ARM 6.5,如果是MDK的话,也是大同小异,汇编部分需要修改。
研究了一下UCOS-II的Cortex-M4的Port部分,觉得很好,就直接拿来用了,修改的很少。首先我们来看一下这一部分几个比较重要的函数,打开os_cpu_a.asm文件,定位到下面的地方,注释我改成中文了。当OS初始化完毕后,执行OSStart,OSStart最后调用OSStartHighRdy函数,注意在此之前的线程模式和异常模式的堆栈都是MSP,在此之后线程模式的堆栈是PSP,异常模式的堆栈仍是MSP。
OSStartHighRdy
LDR R0, =NVIC_SYSPRI14 ; 设置PendSV的异常中断优先级
LDR R1, =NVIC_PENDSV_PRI
STRB R1, [R0]
MOVS R0, #0 ; 初始化PSP=0
MSR PSP, R0
LDR R0, =OS_CPU_ExceptStkBase ; 初始化异常堆栈MSP地址
LDR R1, [R0]
MSR MSP, R1
LDR R0, =OSRunning ; 置OSRunning = TRUE
MOVS R1, #1
STRB R1, [R0]
LDR R0, =NVIC_INT_CTRL ; 触发PendSV异常 (引起上下文切换)
LDR R1, =NVIC_PENDSVSET
STR R1, [R0]
CPSIE I ; 开启中断,于是进入PendSV异常
OSStartHang
B OSStartHang ; 正常情况下,不应运行到这
其中最核心的函数应该是OS_CPU_PendSVHandler了,它处理PendSV中断,完成上下文切换。
OS_CPU_PendSVHandler
CPSID I ; 关中断
MRS R0, PSP ; 获得PSP
CBZ R0, OS_CPU_PendSVHandler_nosave; PSP为0跳到OS_CPU_PendSVHandler_nosave,即不保存上文,直接进入下文。
; 问什么呢,因为首次调用,是没有上文的。
; 保存上文
SUBS R0, R0, #0x20 ; 因为寄存器是32位的,4字节对齐,自动压栈的寄存器有8个,所以偏移为8*0x04=0x20
STM R0, {R4-R11} ; 除去自动压栈的寄存器外,需手动将R4-R11压栈
LDR R1, =OSTCBCur ; 保存上文的SP指针 OSTCBCur->OSTCBStkPtr = SP;
LDR R1, [R1]
STR R0, [R1]
OS_CPU_PendSVHandler_nosave ; 切换下文
PUSH {R14} ; LR压栈,下面要调用C函数
LDR R0, =OSTaskSwHook ; 调用OSTaskSwHook();
BLX R0
POP {R14}
LDR R0, =OSPrioCur ; 置OSPrioCur = OSPrioHighRdy;
LDR R1, =OSPrioHighRdy
LDRB R2, [R1]
STRB R2, [R0]
LDR R0, =OSTCBCur ; 置OSTCBCur = OSTCBHighRdy;
LDR R1, =OSTCBHighRdy
LDR R2, [R1]
STR R2, [R0]
LDR R0, [R2] ; R0中的值为新任务的SP; SP = OSTCBHighRdy->OSTCBStkPtr;
LDM R0, {R4-R11} ; 手动弹出 R4-R11
ADDS R0, R0, #0x20
MSR PSP, R0 ; PSP = 新任务SP
ORR LR, LR, #0x04 ; 确保异常返回后使用PSP
CPSIE I
BX LR ; 退出异常,从PSP弹出xPSR,PC,LR,R0-R3,进入新任务运行
之后我们在此基础上构建自己的OS,首先完成两个任务互相调用,然后是加入SysTick的任务调度,最后加入信号量,邮箱等功能。
下一篇:stm32 fsmc 功能讲解
史海拾趣
|
作者:从11月初回到深圳,就开始着手找工作,拖到05年1月,80天的奔波和忐忑不安的心理终于告一段落。期间面试了好几家公司,有惊喜有失望有抱怨甚至对自己的怀疑,但是我顽固的执着终于有了回报,得到了想要的工作。其中的甘苦与大家分享、共勉, ...… 查看全部问答> |
|
能否让Nk.Nb0的大小自动变化? 现在我的NK.NB0大小都一定会等于config.bib文件中定义的大小。 里边的ROMSIZE等于多大,NK.NB0就一定是多大。 请教… 查看全部问答> |
|
最近写了个应用程序对数据总线上的数据进行操作,关于物理地址和虚拟地址的映射我知道 假设*vi_adress为映射好的虚拟地址 先是写操作 *vi_adress=tmp (tmp是数据) 这个操作是成功的 现在我想读取外面从过来的数据 tmp=*vi_adress& ...… 查看全部问答> |
|
Wince 5.0 如何实现 AlphaBlend? Requirements OS Versions: Microsoft? Windows CE? 5.0 and later. Header: Windows.h. Link Library: Coredll.lib. 加入以上 lib后,编译仍出错。 应如何处理,找了些算法来代替,但速度太慢。 谢谢。… 查看全部问答> |
|
偶菜鸟, 问一个傻子问题,硬件中断如何跳转到软件代码 例如,我有一个键盘,按一下, CPU是如何通过中断服务程序跳转到程序中的代码中? … 查看全部问答> |




