今天学习了uC/OS II的任务切换,知道要实现任务的切换,要将原先任务的寄存器压入任务堆栈,再将新任务中任务堆栈的寄存器内容弹出到CPU的寄存器,其中的CS、IP寄存器没有出栈和入栈指令,所以只能引发一次中断,自动将CS、IP寄存器压入堆栈,再利用中断返回,将新任务的任务断点指针弹出到CPU的CS、IP寄存器中,实现任务切换。虽然明白个大概,但是其中的细节却有点模糊,为什么调用IRET中断返回指令后,弹入CPU的CS、IP寄存器的断点指针是新任务的断点指针,而不是当前任务的,UCOS II是如何实现这个过程的?网上的文章没有提及任务切换的细节,然而这些细节却是理解的重点,所以才有了今天这一篇文章!

    
    在uC/OS II中,任务的调度由任务调度器完成,其主要的工作有两项:
    1.在任务就绪表中查找最高优先级的就绪任务,该任务将是未来切换执行的任务;
    2.实现任务的切换。
    
    任务级的任务调度器,由OS_Sched()实现。
     //!!!为简单说明问题,OS_Sched中特意删减了一些语句
void OS_Sched (void)
{
INT8U y;
OS_ENTER_CRITICAL();
y =OSUnMapTbl[OSRdyGrp];
//得到最高优先级任务,OSPrioHighRdy从此为就绪表中最高优先级任务的优先级别
OSPrioHighRdy=(INT8U)((y <<)+OSUnMapTbl[OSRdyTbl[y]]);
//判断当前任务是否为就绪表中最高优先级任务
if(OSPrioHighRdy!=OSPrioCur)
{
//得到新任务的任务控制块的指针
OSTCBHighRdy=OSTCBPrioTbl[OSPrioHighRdy];
//统计任务切换次数的计数器加1
OSCtxSwCtr++;
//此句关键!使用软件中断切换任务,OS_TASK_SW()其实为INT 0x80,128号中断
//中断产生后,CPU依次将PSW状态寄存器、断点指针段地址:偏移地址压入当前任务的堆栈中
OS_TASK_SW();
}
OS_EXIT_CRITICAL();
}
    从上述代码,可以知道,OS_Sched的任务切换步骤分为两步:
    1.获得最高优先级任务的任务控制块指针;
    2.利用中断实现断点数据的切换。(当前任务中止运行时,CPU寄存器CS、IP、PSW和通用寄存器等中的数据就称为断点数据)
   
     在获得了新任务的任务控制块指针,相当于获得了任务控制块中包含的任务堆栈,而任务堆栈中又包含了该任务运行时,CPU寄存器的初始内容,即断点数据。前面已经提到过 “要实现任务的切换,要将原先任务的寄存器压入任务堆栈,再将新任务中任务堆栈的寄存器内容弹出到CPU的寄存器”,除了CS、IP寄存器,其他寄存器只需要利用出栈和入栈指令即可,而CS、IP寄存器的出栈、入栈要利用中断INT 0x80和IRET。好吧,那问题来了,uC/OS是如何利用中断实现断点指针CS、IP的切换?
    
     产生INT 0x80后,CPU做了什么,当然是去执行和INT 0x80相关的中断服务程序去了(说明:每个中断都有一个对应好的中断服务程序,每当产生该中断,则由CPU自动调用该对应的中断服务程序)。
    
    在uC/OS II中,INT 0x80对应的中断服务程序是
     _OSCtxSw PROC FAR
;
;保存当前任务寄存器内容,将它们压入当前任务堆栈,AX、CX、DX、BX、SP(原始值)、BP、SI 及 DI
PUSHA ;
PUSH ES ;
PUSH DS ;
;
;将当前任务的任务控制块的指针段地址赋值给DS寄存器
MOV AX, SEG _OSTCBCur;Reload DS in case it was altered
MOV DS, AX ;
;
; LES指令作用,取得OSTCBCur变量内容(因为任务控制块的第一个成员为任务堆栈指针,所以ES:[BX+0]表示的地址不仅是任务控制块的首地址,也是任务堆栈指针的地址),低字存放于BX,高字存放于ES,有关LES详细查看另一篇文章!
;同时由于任务控制块的第一成员为任务堆栈指针变量,所以任务堆栈指针成员变量的地址和任务控制块的地址一样
LES BX, DWORD PTR DS:_OSTCBCur; OSTCBCur->OSTCBStkPtr = SS:SP!!!
MOV ES:[BX+], SS ;将当前SS(栈的基地址)寄存器值存放至当前任务控制块的2,3内存单元
MOV ES:[BX+], SP ;将当前SP(栈顶的偏移量)存放至当前任务控制块的0,1内存单元
;
CALL FAR PTR _OSTaskSwHook;Call user defined task switch hook
;
;存储器寻址,DS:_OSTCBHighRdy+2存放的是新任务的栈基址SS,DS:_OSTCBHighRdy存放的是栈顶偏移量SP
MOV AX, WORD PTR DS:_OSTCBHighRdy+;将OSTCBHighRdy赋值给OSTCBCur,使OSTCBCur指向新的任务控制块
MOV DX, WORD PTR DS:_OSTCBHighRdy;OSTCBCur=OSTCBHighRdy
MOV WORD PTR DS:_OSTCBCur+, AX ;
MOV WORD PTR DS:_OSTCBCur, DX ;
;
MOV AL, BYTE PTR DS:_OSPrioHighRdy;OSPrioCur=OSPrioHighRdy
MOV BYTE PTR DS:_OSPrioCur, AL ;
;
;将最高级优先级任务的任务控制块包含的SS和SP寄存器值,CPU的SS和SP指向了新的任务堆栈
LES BX, DWORD PTR DS:_OSTCBHighRdy; SS:SP =OSTCBHighRdy->OSTCBStkPtr
MOV SS, ES:[BX+];
MOV SP, ES:[BX];
;
;注意:这里弹出的是新任务的堆栈,而不是旧的,因为CPU的SS和SP已经指向了新的任务堆栈
POP DS ;Loadnew task's context
POP ES ;
POPA ;
;
;当新任务的堆栈都弹出的时候,只剩下新任务的CS、IP指针,刚好运行IRET弹到CPU的CS、IP寄存器,开始新任务的运行
IRET ;Return to new task
;
_OSCtxSw ENDP
    请仔细将上面代码对照着注释看一遍,注释已经说得很清楚了,_OSCtxSw的堆栈操作,可以看下面两幅图
(0)在调用_OSCtxSw中断服务程序前,CPU将PSW和任务断点指针CS、IP压入当前任务堆栈
(1)将CPU寄存器AX、CX、DX、BX、SP(原始值)、BP、SI 及 DI压入当前任务堆栈
(2)将当前CPU的堆栈指针SS和SP存入当前任务控制块的任务堆栈指针变量(FAR指针变量占4字节)
说明:至此,当前任务的断点数据已经全部保存,所以在每一个新任务运行之前,uC/OS都会将旧任务的断点数据全部按照顺序压入私有堆栈中,同时每一个新任务的任务堆栈都已保存初始化好的或以前任务中止时,CPU保存的断点数据。
(3)OSTCBCur = OSTCBHighRdy、OSPrioCur = OSPrioHighRdy
(4)使CPU的SS和SP堆栈指针指向新的任务堆栈
(5)将新的任务堆栈的内容弹出到CPU寄存器,顺序为DS、ES、DI、SI、BP、SP、BX、DX、CX、AX。
说明:注意是新任务堆栈在进行弹栈操作,而新任务堆栈在弹栈前,和旧任务的断点数据全部压栈后的结构一模一样,都是顺序为DS、ES、DI、SI、BP、SP、BX、DX、CX、AX、IP、CS、PSW。经过POP DS POP ES POPA,这3条指令之后,新任务堆栈便只剩下IP、CS、PSW,IP和CS这断点指针指向新任务的任务代码,只要将它们弹出到CPU的CS、IP寄存器,就可以实现任务切换。
(6)运行IRET,将新任务断点指针弹出到CPU的CS、IP指针寄存器,PSW弹出到CPU的PSW寄存器
(7)至此,uC/OS开始运行新任务
参考链接:
μC-OS-Ⅱ中通过中断返回指令实现任务切换  http://www.docin.com/p-223857136.html
我对OSCtxSW()任务级任务切换函数的理解(以图例的方式) http://blog.163.com/muren20062094%40yeah/blog/static/1618444162011727102557489/
uCOSii中断处理过程详解 http://www.doc88.com/p-1498746979512.html
uC/OS-II任务栈处理的一种改进方法 http://www.3edu.net/lw/qrs/lw_46692.html

本文链接:http://www.cnblogs.com/cposture/p/4291503.html

【原创】uC/OS II 任务切换原理的更多相关文章

  1. uC/OS II原理分析及源码阅读(一)

    uC/OS II(Micro Control Operation System Two)是一个可以基于ROM运行的.可裁减的.抢占式.实时多任务内核,具有高度可移植性,特别适合于微处理器和控制器,是和 ...

  2. 【小梅哥SOPC学习笔记】NIOS II处理器运行UC/OS II

    SOPC开发流程之NIOS II 处理器运行 UC/OS II 这里以在芯航线FPGA学习套件的核心板上搭建 NIOS II 软核并运行 UCOS II操作系统为例介绍SOPC的开发流程. 第一步:建 ...

  3. uC/OS II 函数说明 之–OSTaskCreate()与OSTaskCreateExt()

    1. OSTaskCreate()    OSTaskCreate()建立一个新任务,能够在多任务环境启动之前,或者执行任务中建立任务.注意,ISR中禁止建立任务,一个任务必须为无限循环结构.    ...

  4. uc/os iii移植到STM32F4---IAR开发环境

    也许是先入为主的原因,时钟用不惯Keil环境,大多数的教程都是拿keil写的,尝试将官方的uc/os iii 移植到IAR环境. 1.首先尝试从官网上下载的官方移植的代码,编译通过,但是执行会报堆栈溢 ...

  5. 【原创】uC/OS 中LES BX,DWORD PTR DS:_OSTCBCur的作用及原理

    LES BX, DWORD PTR DS:_OSTCBCur ;OSTCBCur->OSTCBStkPtr = SS:SP!!! ], SS ;将当前SS(栈的基地址)寄存器值存放至当前任务控制 ...

  6. 在STM32F401上移植uC/OS的一个小问题 [原创]

    STM32F401xx是意法半导体新推出的Cortex-M4内核的MCU,相较于已经非常流行的STM32F407xx和STM32F427xx等相同内核的MCU而言,其特点是功耗仅为128uA/MHz, ...

  7. 关于uC/OS的简单学习(转)

    1.微内核 与Linux的首要区别是,它是一个微内核,内核所实现的功能非常简单,主要包括: 一些通用函数,如TaskCreate(),OSMutexPend(),OSQPost()等. 中断处理函数, ...

  8. uc/os任务创建

    问题描述:      uc/os中任务创建 问题解决: 创建一个任务,任务从无到有.任务创建函数分两种, 一种是基本的创建函数OSTaskCreate, 另一种是扩展的任务创建函数OSTaskCrea ...

  9. uC/OS 的任务调度解析 (转)

    uC/OS 的任务调度解析 1.任务调度器启动之后(初始化,主要是TCB的初始化),就可以创建任务,开始任务调度了,实际上第一个任务准确的说不是进行任务切换,而是进行启动当前最高优先级任务.uC/OS ...

随机推荐

  1. 洛谷 质因子分 p2043

    #include <iostream>#include <algorithm>#include <cstring>using namespace std; cons ...

  2. php面试题--并列排名问题

    给定一个二维数组: <?php $data = [ ['name' =>'j1', 'score' => '80'], ['name' =>'j2', 'score' => ...

  3. mvc开发网站打开慢总结

    开始学习mvc开发网站的时候,看了传智博客的视频教程,其中学习了一个和牛逼的框架,开始激动的深入学习,学完后却发现其实那套框架太重并不适合一些中小型的网站开发,并且也使用导航属性关联外键,导致打开网站 ...

  4. cookie httpOnly 打勾

    在 chrome dev tools 中,通过 Application -> Cookies 可以查看 cookie. 其中有一列表头为`HTTP`,如果其下打勾了,则表明 httpOnly. ...

  5. jdango

    1.jdango的下载 命令行: pip install django ==1.11.18 pip install django ==1.11.18 -i https://pypi.douban.co ...

  6. memcache启动报错:memcached: error while loading shared libraries: libevent-XXXXX5: cannot 。。。。

    创建连接 ln -s /usr/lib/libevent-2.1.so.6  /usr/lib/libevent-2.1.so.6 如果还不行就下面解决 执行下面语句查看链接地址 LD_DEBUG=l ...

  7. Maven和Gradle的区别

    转自:http://www.infoq.com/cn/news/2011/04/xxb-maven-6-gradle Maven面临的挑战 软件行业新旧交替的速度之快往往令人咂舌,不用多少时间,你就会 ...

  8. XML学习总结二——DTD

    主要用处是约束XML. 1.DTD分为内部DTD与外部DTD两类: 内部DTD:将DTD定义在XML文档的内部 <!DOCTYPE  根元素名  [    元素描述]> 外部DTD < ...

  9. swarm集群日常部分操作

    docker swarm: 1)查看集群使用docker info 或 docker node ls 查看集群中的相关信息 2)swarm集群中node的availability状态可以为 activ ...

  10. 28.实现 strStr() 函数

    28.实现 strStr() 函数 给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始).如果不存在, ...