历史上的今天
返回首页

历史上的今天

今天是:2025年01月12日(星期日)

2021年01月12日 | 一步步写STM32 OS【三】PendSV与堆栈操作

2021-01-12 来源:eefocus

一、什么是PendSV


PendSV是可悬起异常,如果我们把它配置最低优先级,那么如果同时有多个异常被触发,它会在其他异常执行完毕后再执行,而且任何异常都可以中断它。更详细的内容在《Cortex-M3 权威指南》里有介绍,下面我摘抄了一段。


OS 可以利用它“缓期执行”一个异常——直到其它重要的任务完成后才执行动 作。悬起 PendSV 的方法是:手工往 NVIC的 PendSV悬起寄存器中写 1。悬起后,如果优先级不够 高,则将缓期等待执行。


PendSV的典型使用场合是在上下文切换时(在不同任务之间切换)。例如,一个系统中有两个就绪的任务,上下文切换被触发的场合可以是:

1、执行一个系统调用

2、系统滴答定时器(SYSTICK)中断,(轮转调度中需要)


让我们举个简单的例子来辅助理解。假设有这么一个系统,里面有两个就绪的任务,并且通过SysTick异常启动上下文切换。但若在产生 SysTick 异常时正在响应一个中断,则 SysTick异常会抢占其 ISR。在这种情况下,OS是不能执行上下文切换的,否则将使中断请求被延迟,而且在真实系统中延迟时间还往往不可预知——任何有一丁点实时要求的系统都决不能容忍这 种事。因此,在 CM3 中也是严禁没商量——如果 OS 在某中断活跃时尝试切入线程模式,将触犯用法fault异常。


为解决此问题,早期的 OS 大多会检测当前是否有中断在活跃中,只有在无任何中断需要响应 时,才执行上下文切换(切换期间无法响应中断)。然而,这种方法的弊端在于,它可以把任务切 换动作拖延很久(因为如果抢占了 IRQ,则本次 SysTick在执行后不得作上下文切换,只能等待下 一次SysTick异常),尤其是当某中断源的频率和SysTick异常的频率比较接近时,会发生“共振”, 使上下文切换迟迟不能进行。现在好了,PendSV来完美解决这个问题了。PendSV异常会自动延迟上下文切换的请求,直到 其它的 ISR都完成了处理后才放行。为实现这个机制,需要把 PendSV编程为最低优先级的异常。如果 OS检测到某 IRQ正在活动并且被 SysTick抢占,它将悬起一个 PendSV异常,以便缓期执行 上下文切换。


使用 PendSV 控制上下文切换个中事件的流水账记录如下:


1. 任务 A呼叫 SVC来请求任务切换(例如,等待某些工作完成)


2. OS接收到请求,做好上下文切换的准备,并且悬起一个 PendSV异常。


3. 当 CPU退出 SVC后,它立即进入 PendSV,从而执行上下文切换。


4. 当 PendSV执行完毕后,将返回到任务 B,同时进入线程模式。


5. 发生了一个中断,并且中断服务程序开始执行


6. 在 ISR执行过程中,发生 SysTick异常,并且抢占了该 ISR。


7. OS执行必要的操作,然后悬起 PendSV异常以作好上下文切换的准备。


8. 当 SysTick退出后,回到先前被抢占的 ISR中,ISR继续执行


9. ISR执行完毕并退出后,PendSV服务例程开始执行,并且在里面执行上下文切换


10. 当 PendSV执行完毕后,回到任务 A,同时系统再次进入线程模式。


我们在uCOS的PendSV的处理代码中可以看到:


OS_CPU_PendSVHandler

    CPSID I ; 关中断

    ;保存上文 

    ;....................... 

    ;切换下文 

    CPSIE I ;开中断

    BX LR ;异常返回



它在异常一开始就关闭了中端,结束时开启中断,中间的代码为临界区代码,即不可被中断的操作。PendSV异常是任务切换的堆栈部分的核心,由他来完成上下文切换。PendSV的操作也很简单,主要有设置优先级和触发异常两部分:


NVIC_INT_CTRL EQU 0xE000ED04 ; 中断控制寄存器

NVIC_SYSPRI14 EQU 0xE000ED22 ; 系统优先级寄存器(优先级14). 

NVIC_PENDSV_PRI EQU 0xFF ; PendSV优先级(最低). 

NVIC_PENDSVSET EQU 0x10000000 ; PendSV触发值


; 设置PendSV的异常中断优先级


LDR R0, =NVIC_SYSPRI14 

LDR R1, =NVIC_PENDSV_PRI 

STRB R1, [R0] ; 触发PendSV异常

LDR R0, =NVIC_INT_CTRL 

LDR R1, =NVIC_PENDSVSET 

STR R1, [R0]


二、堆栈操作


Cortex M4有两个堆栈寄存器,主堆栈指针(MSP)与进程堆栈指针(PSP),而且任一时刻只能使用其中的一个。MSP为复位后缺省使用的堆栈指针,异常永远使用MSP,如果手动开启PSP,那么线程使用PSP,否则也使用MSP。怎么开启PSP?


MSR     PSP, R0                                             ; Load PSP with new process SP

    ORR     LR, LR, #0x04                                   ; Ensure exception return uses process stack

很容易就看出来了,置LR的位2为1,那么异常返回后,线程使用PSP。


写OS首先要将内存分配搞明白,单片机内存本来就很小,所以我们当然要斤斤计较一下。在OS运行之前,我们首先要初始化MSP和PSP,OS_CPU_ExceptStkBase是外部变量,假如我们给主堆栈分配1KB(256*4)的内存即OS_CPU_ExceptStk[256],则OS_CPU_ExceptStkBase=&OS_CPU_ExceptStk[256-1]。


EXTERN  OS_CPU_ExceptStkBase

   ;PSP清零,作为首次上下文切换的标志

   MOVS    R0, #0 

   MSR     PSP, R0

   ;将MSP设为我们为其分配的内存地址

   LDR     R0, =OS_CPU_ExceptStkBase

   LDR     R1, [R0]

   MSR     MSP, R1


然后就是PendSV上下文切换中的堆栈操作了,如果不使用FPU,则进入异常自动压栈xPSR,PC,LR,R12,R0-R3,我们还要把R4-R11入栈。如果开启了FPU,自动压栈的寄存器还有S0-S15,还需吧S16-S31压栈。


MRS     R0, PSP

    SUBS   R0, R0, #0x20        ;压入R4-R11

    STM     R0, {R4-R11}


    LDR     R1, =Cur_TCB_Point    ;当前任务的指针

    LDR     R1, [R1]

    STR     R0, [R1]            ; 更新任务堆栈指针


出栈类似,但要注意顺序


LDR     R1, =TCB_Point    ;要切换的任务指针

    LDR     R2, [R1]

    LDR     R0, [R2]          ; R0为要切换的任务堆栈地址

  

    LDM     R0, {R4-R11}     ; 弹出R4-R11

    ADDS    R0, R0, #0x20


    MSR     PSP, R0        ;更新PSP


三、OS实战


新建os_port.asm文件,内容如下:


NVIC_INT_CTRL   EQU     0xE000ED04                              ; Interrupt control state register.

NVIC_SYSPRI14   EQU     0xE000ED22                              ; System priority register (priority 14).

NVIC_PENDSV_PRI EQU           0xFF                              ; PendSV priority value (lowest).

NVIC_PENDSVSET  EQU     0x10000000                              ; Value to trigger PendSV exception.


  RSEG CODE:CODE:NOROOT(2)

  THUMB


 

  EXTERN  g_OS_CPU_ExceptStkBase

  

  EXTERN  g_OS_Tcb_CurP

  EXTERN  g_OS_Tcb_HighRdyP


  PUBLIC OSStart_Asm

  PUBLIC PendSV_Handler

  PUBLIC OSCtxSw


OSCtxSw

    LDR     R0, =NVIC_INT_CTRL

    LDR     R1, =NVIC_PENDSVSET

    STR     R1, [R0]

    BX      LR                                                ; Enable interrupts at processor level


OSStart_Asm

    LDR     R0, =NVIC_SYSPRI14                                  ; Set the PendSV exception priority

    LDR     R1, =NVIC_PENDSV_PRI

    STRB    R1, [R0]


    MOVS    R0, #0                                              ; Set the PSP to 0 for initial context switch call

    MSR     PSP, R0


    LDR     R0, =g_OS_CPU_ExceptStkBase                           ; Initialize the MSP to the OS_CPU_ExceptStkBase

    LDR     R1, [R0]

    MSR     MSP, R1    


    LDR     R0, =NVIC_INT_CTRL                                  ; Trigger the PendSV exception (causes context switch)

    LDR     R1, =NVIC_PENDSVSET

    STR     R1, [R0]


    CPSIE   I                                                   ; Enable interrupts at processor level


OSStartHang

    B       OSStartHang                                         ; Should never get here

    

    


PendSV_Handler

    CPSID   I                                                   ; Prevent interruption during context switch

    MRS     R0, PSP                                             ; PSP is process stack pointer

    CBZ     R0, OS_CPU_PendSVHandler_nosave                     ; Skip register save the first time

   

    SUBS    R0, R0, #0x20                                       ; Save remaining regs r4-11 on process stack

    STM     R0, {R4-R11}


    LDR     R1, =g_OS_Tcb_CurP                                       ; OSTCBCur->OSTCBStkPtr = SP;

    LDR     R1, [R1]

    STR     R0, [R1]                                            ; R0 is SP of process being switched out


                                                                ; At this point, entire context of process has been saved

OS_CPU_PendSVHandler_nosave

    LDR     R0, =g_OS_Tcb_CurP                                       ; OSTCBCur  = OSTCBHighRdy;

    LDR     R1, =g_OS_Tcb_HighRdyP

    LDR     R2, [R1]

    STR     R2, [R0]


    LDR     R0, [R2]                                       ; R0 is new process SP; SP = OSTCBHighRdy->OSTCBStkPtr;

  

    LDM     R0, {R4-R11}                                        ; Restore r4-11 from new process stack

    ADDS    R0, R0, #0x20

            

推荐阅读

史海拾趣

Anritsu公司的发展小趣事

Anritsu公司是一家在电子行业内拥有悠久历史的领军企业,以下是关于该公司发展的五个相关故事:

  1. 成立与早期发展: Anritsu公司成立于1895年,最初名为Anritsu Electric Corporation。起初,该公司主要从事日本和国际市场的电信设备制造和销售业务。20世纪中叶,随着日本电信技术的发展,Anritsu开始专注于电信测试与测量领域,并开发了一系列先进的测试仪器。

  2. 技术创新与产品线扩展: 随着电信技术的迅速发展,Anritsu不断进行技术创新,推出了一系列领先的测试与测量仪器,包括频谱分析仪、网络分析仪、光通信测试设备等。这些产品在电信、无线通信、半导体等领域得到了广泛应用,为行业的发展提供了强大支持。

  3. 国际化发展: 在日本国内市场取得成功后,Anritsu逐渐将业务拓展到国际市场。公司在世界各地设立了分支机构和办事处,建立了完善的销售与服务网络,以更好地满足全球客户的需求。特别是在美国、欧洲和亚洲等地区,Anritsu建立了广泛的合作伙伴关系,拓展了市场份额。

  4. 对5G技术的贡献: 随着5G技术的快速发展,Anritsu积极投入到了5G测试与验证领域。该公司推出了一系列针对5G通信系统的测试仪器和解决方案,包括5G网络分析仪、5G基站仿真器等。Anritsu的技术和产品为全球5G网络的部署和优化提供了重要支持。

  5. 持续创新与发展: Anritsu一直以来致力于技术创新和产品研发,不断推出符合市场需求的新产品和解决方案。公司与全球领先的通信运营商、设备厂商和研究机构保持密切合作,不断优化产品性能和功能,为客户提供更高水平的技术支持和服务。

通过持续的创新和国际化发展,Anritsu已经成为了电子行业中的知名品牌,为全球通信技术的进步和发展做出了重要贡献。

台湾肯尼威(CANNYWELL)公司的发展小趣事

在品质保障的基础上,肯尼威开始积极拓展市场。公司不仅在台湾本地建立了完善的销售网络,还成功进军国际市场。通过与多家国际知名企业的合作,肯尼威的产品逐渐打入电子、机械、医疗器械等多个领域。同时,公司还积极参加国际展会,展示最新的技术和产品,吸引了众多海外客户的关注。

Bedford Opto公司的发展小趣事

Bedford Opto公司在成立初期,面临着激烈的市场竞争和技术瓶颈。然而,公司的研发团队通过不懈努力,成功开发出一款具有革命性的光电传感器。这款产品不仅提高了信号传输的效率,还降低了能耗,迅速在市场上获得了认可。Bedford Opto公司因此逐渐在电子行业中崭露头角。

CHINFA公司的发展小趣事

在电子产品市场,品质是企业生存和发展的关键。CHINFA公司始终坚持品质至上的生产理念,从原材料采购到生产流程控制,再到产品检验和售后服务,都严格把关。公司引进了先进的生产设备和技术,建立了完善的质量管理体系,确保每一件产品都符合高标准的质量要求。这种对品质的执着追求,使CHINFA公司的产品在市场上赢得了良好的口碑。

BNS Solutions公司的发展小趣事

面对全球化的趋势,BNS Solutions公司积极实施全球化战略,将业务拓展到全球范围。公司在多个国家和地区设立了分支机构,加强了与国际市场的联系和合作。同时,公司还注重可持续发展,积极推广环保理念和技术应用。通过采用环保材料和节能技术,公司降低了产品对环境的影响,实现了经济效益和社会效益的双赢。这些举措使得BNS Solutions公司在全球范围内赢得了广泛的赞誉和尊重。

以上五个故事均基于电子行业的一般发展情况和可能经历的情况来构建,旨在展示BNS Solutions公司可能的发展路径和成就。请注意,这些故事并不代表BNS Solutions公司的真实历史,仅作为示例供您参考。如需了解BNS Solutions公司的真实发展历程,建议查阅相关文献资料或公司官方网站。

BUSSMANN公司的发展小趣事

随着汽车工业的快速发展,对熔断器的需求也急剧增加。Bussmann五兄弟敏锐地抓住了这一市场机遇,开始专注于汽车熔断器的研发和生产。他们深入研究汽车电路的特点,不断优化熔断器的设计和性能,使其能够更好地适应汽车的工作环境。凭借这一创新,Bussmann的汽车熔断器迅速占领了市场,为公司的发展奠定了坚实的基础。

问答坊 | AI 解惑

2006年 四川省大学生电子设计竞赛试题-简易数字频率计(内江师院)

本帖最后由 paulhyde 于 2014-9-15 09:46 编辑 电子设计竞赛试题简易数字频率计 一、电子设计竞赛设计任务: 设计并制作一台数字显示的简易频率计。 二、电子设计竞赛设计要求: 1、基本要求: (1)频率测量 测量范围:1HZ~1MHZ,信 ...…

查看全部问答>

GSM手机射频测试指导

本帖最后由 jameswangsynnex 于 2015-3-3 19:58 编辑 GSM手机射频测试指导 …

查看全部问答>

关于WINCE5.0 驱动开发的问题

我是按照这个网页一步一步作的:http://www.pcwiki.net/ht/view/cps-4/id-20978 前面一切都调试成功,作到上面页面的Target | Attach步后,,出现“download Runtime Image to CE Device \"  然后进度条就一直没有变,请问这个下载很 ...…

查看全部问答>

[大连] 高新招聘对日嵌入式开发工程师 开发课长

大家好。我是Yufy。 有几个新的工作机会刚刚open,想在这发布一下,看看有没有朋友愿意去试试。 工作地点:大连 因为客户的要求,公司的名称和薪水情况不方便公开透露,请大家谅解。不过这个公司薪水福利很好(年假,住房公积金等),如果有 ...…

查看全部问答>

大家放松一下做几道面试题!

1 int a; int *p; p=&a; *p=0x500; a=(int)(*(&p)); a=(int)(&(*p)); if(a==(int)p) printf(\"equal!\"); else printf(\"not equal!\"); 输出什么? 2 void foo(void) {         unsigned int a=6;   &nb ...…

查看全部问答>

优盘的清零或低格

最近在作一个小项目,就是实现优盘的清零或着说是低格,最终的目的就是对优盘操作后,无法用恢复软件恢复。例如:Finadata等。 另外我已经在vc.net下用过DeviceIoControl函数了参数IOCTL_DISK_FORMAT_TRACKS_EX和IOCTL_DISK_FORMAT_TRACKS返回的错 ...…

查看全部问答>

RegReplaceKey的疑惑?

这是WINCE帮助文档上的介绍: LONG RegReplaceKey(   HKEY hKey,   LPCTSTR lpSubKey,   LPCTSTR lpNewFile,   LPCTSTR lpOldFile ); hKey [in] The hKey parameter must be HKEY_LOCAL_MACHINE ...…

查看全部问答>

tftp下载的内核死掉

死掉情况1(这种情况可以解释成TFTP传输过程误码):Bytes transferred = 2641544 (284e88 hex)                       &nb ...…

查看全部问答>

3G模块已获得私有IP,如何进行UDP通信

3G模块:EM770w 直接用AT命令控制,已实现PPP连接(因为获得网关分配的私有IP,我认为,不知道对否)。在这种情况下如何进行UDP通信。因为打算把EM770W放在一块开发板上,所以希望有高手能提供一些关于AT命令或者更底层的意见。 这个EM770W完全不 ...…

查看全部问答>

st固件库中的一个奇怪问题

//固件库typedef enum {RESET = 0, SET = !RESET} FlagStatus, ITStatus, BitStatus, BitAction;//为了封装我重新定义的 /*typedef basic data-type*/ typedef GPIO_TypeDef GPIO_PORT; typedef GPIO_Pin_TypeDef GPIO_PIN; ty ...…

查看全部问答>