Chinaunix首页 | 论坛 | 博客
  • 博客访问: 145998
  • 博文数量: 29
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 265
  • 用 户 组: 普通用户
  • 注册时间: 2014-01-04 13:11
文章分类

全部博文(29)

文章存档

2015年(2)

2014年(27)

我的朋友

分类: 嵌入式

2014-02-27 10:09:42

UCOS优先级调度算法

经过一个上午时间,终于明白UCOS系统的按优先级调度就绪任务的算法,以个人见解展示给各位,希望各位提出意见uponzw630@163.com。

首先说明,解释两个表格的含义。

说明

OSMapTbl就是0-7这8个数值用二进制的BIT位来表示。

OSUnMapTbl就是将0x00-0xff每个数据中最低位为1的位号全部列举出来

INT8U  const  OSMapTbl[]   = {0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80};

INT8U  const  OSUnMapTbl[] = {

    0, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x00 to 0x0F*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x10 to 0x1F*/

    5, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x20 to 0x2F*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x30 to 0x3F*/

    6, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x40 to 0x4F*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x50 to 0x5F*/

    5, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x60 to 0x6F*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x70 to 0x7F*/

    7, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x80 to 0x8F*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0x90 to 0x9F*/

    5, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0xA0 to 0xAF*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0xB0 to 0xBF*/

    6, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0xC0 to 0xCF*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0xD0 to 0xDF*/

    5, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0,       /* 0xE0 to 0xEF*/

    4, 0, 1, 0, 2, 0, 1, 0, 3, 0, 1, 0, 2, 0, 1, 0        /* 0xF0 to 0xFF*/

};

 

优先级变量prio,可以表示为8个比特,如下表:

0

0

Y

Y

Y

X

X

X

 

Y处于BIT5-BIT3,X处于BIT2-BIT0。

我们知道优先级prio的值越小,prio的优先级越高。

X和Y的范围均为000-111,十进制数里0-7,更8个级别,为了判断X或Y的大小,我们就引入了表格OSMapTbl[]= {0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80};分别用来表示0-7优先级的值,这是为了方便找到最高优先级。

 

在使用中,OSRdyGrp表示y,OSRdyTbl[y]表示X。

OSRdyGrp   按BIT位可以表示为第0组-第7组优先级

OSRdyTbl[n] 可以表示为第n组的8个优先级

 

任务就绪算法:

OSRdyGrp        |= OSMapTbl[prio>>3];

OSRdyTbl[prio>>3] |= OSMapTbl[prio&0x07];

例如:prio为5,二进制为0b0000 0101,得到Y为0,X为5,再把这两个数据经过查表OSMapTbl[],最终得到OSRdyGrp为0x01,OSRdyTbl[0] 为0x20;

 

查询就绪算法:

Y=OSUnMapTbl[OSRdyGrp];

X=OSUnMapTbl[OSRdyTbl[y]];

Prio = (y<<3 )+ x;

验算:如OSRdyGrp为0x01,OSRdyTbl[0] 为0x20,经过查询就绪算法后

Y=0;X=5;

最终prio = (0<<3)+5=5;这样就优先级反向算法就完成了。

各位看官可以自己举例,进行验证。

 

多个优先级如何查找最高优先级

OSRdyGrp和OSRdyTbl[]中存了许多优先级,但是我们现在要做的是找出其中的最高优先级,我们先找出里面对应存了多少优先级。

如现在就绪的优先级有以下6个,prio = 1,3,4,13,16,20这6个优先级存于OSRdyGrp和OSRdyTbl[]中具体数值分别为多少。

Prio为1,Y=0,x=1经查表OSMapTbl[],得OSRdyGrp为0x01,OSRdyTbl[0] 为0x02;

Prio为3,Y=0,x=3经查表OSMapTbl[],得OSRdyGrp为0x01,OSRdyTbl[0] 为0x08;

Prio为4,Y=0,x=4经查表OSMapTbl[],得OSRdyGrp为0x01,OSRdyTbl[0] 为0x10;

Prio为16,Y=1,x=5经查表OSMapTbl[],得OSRdyGrp为0x02,OSRdyTbl[2] 为0x20;

Prio为16,Y=2,x=0经查表OSMapTbl[],得OSRdyGrp为0x04,OSRdyTbl[2] 为0x01;

Prio为20,Y=2,x=4经查表OSMapTbl[],得OSRdyGrp为0x04,OSRdyTbl[2] 为0x10;

用运算符“或”结果为

OSRdyGrp  = 0x07;//0b00000111

OSRdyTbl[0] = 0x1a;//0b00011010

OSRdyTbl[1] = 0x20;//0b00100000

OSRdyTbl[2] = 0x11;//0b00010001

 

再通过上面任务就绪算法,显然OSRdyGrp为1的最低位为bit0,所以最高优先级在第0组(优先级实际数值0-7),我们就要查OSRdyTbl[0]为1的最低位为bit1,所以就可以得到其中最高优先级为优先级1。

 

众所周知,时间对于一个系统而言是非常重要的,UCOS为了节约在任务调度时,按优先级算法的时间,只有短短几行指令移位和查表,就可以完成最高级任务的就绪和调度,挺佩服写出这个算法作者。

 

阅读(1847) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~