找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5492|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

uC/OS-III 任務(wù)就緒表學(xué)習(xí)記錄

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:82083 發(fā)表于 2015-6-6 02:54 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
       

這段時間利用上下班的時間在看 《嵌入式實時操作系統(tǒng)uC/OS-III》這本書,里面有很多很有意思的思想。如今看到第九章,想將之前的一些心得記錄下來,本想今晚從第三章開始寫筆記,敲了一段文字之后,感覺像是抄書,覺得沒有太大意義。時間長了,連筆記都不知道如何寫。先來個簡單點的開始吧。

uC/OS-III 的任務(wù)就緒表和uC/OS-II的就緒表差別很大,簡單的了很多。
拿8位的單片機,64級優(yōu)先級來說。
// 優(yōu)先級表
unsigned char OSPrioTbl[8] = {0};



每個元素的每bit位對應(yīng)的優(yōu)先級如下,優(yōu)先級數(shù)值越低表示優(yōu)先級越高:

OSPrioTbl[0]->bit0~bit7  優(yōu)先級 0 ~ 7
OSPrioTbl[1]->bit0~bit7  優(yōu)先級 8 ~ 15
OSPrioTbl[2]->bit0~bit7  優(yōu)先級 16 ~ 23
OSPrioTbl[3]->bit0~bit7  優(yōu)先級 24 ~ 31  
OSPrioTbl[4]->bit0~bit7  優(yōu)先級 32 ~ 39  
OSPrioTbl[5]->bit0~bit7  優(yōu)先級 40 ~ 47  
OSPrioTbl[6]->bit0~bit7  優(yōu)先級 48 ~ 55     
OSPrioTbl[7]->bit0~bit7  優(yōu)先級 56 ~ 63

當某一優(yōu)先級的任務(wù)就緒,對于的bit位就會被置1,例如 優(yōu)先級 7 的任務(wù)就緒,那么
OSPrioTbl[0] 的第bit7位將會被置位。

查找最高優(yōu)先級的流程如下(原理實際上是計算零的個數(shù),零的個數(shù)恰好對于優(yōu)先級數(shù)):

OS_PRIO OS_PrioGetHighest()
{
        char i = 0,  char prio = 0;


        // 一次性比較 8位 由于空閑任務(wù)的存在,所以這個循環(huán)最終都會跳出而數(shù)組不會越界訪問。
        while( 0 == OSPrioTbl[i++])
        {
                prio += 8;
        }
        // 這個函數(shù)實際上是計算前導(dǎo)零,即計算該元素的前面有幾個零,再與prio相加即得到最高優(yōu)先級。
        prio += CPU_CntLeadZeros(OSPrioTbl[i-1]);
        return (prio);
}       


有些CPU支持CLZ指令(即計算簽導(dǎo)零數(shù)量),那么就可以充分利用該指令加速計算過程。

實際上,如果是在32位的單片機下需要64級優(yōu)先級,那么這個算法還可以再優(yōu)化成如下:

unsigned  int  OSPrioTbl[2] = {0};

OS_PRIO OS_PrioGetHighest()
{
        char i = 0,  char prio = 0;


        if( 0 != OSPrioTbl[0])
                prio = CPU_CntLeadZeros(OSPrioTbl[0]);
        else
                prio = CPU_CntLeadZeros(OSPrioTbl[1]) + 32;
       
        return (prio);
}       

上述代碼并不是 uC/OS-III  的實現(xiàn),只是我自己為了方便理解寫了具備相同功能的代碼。
書籍上所述代碼如下:

OS_PRIO OS_PrioGetHighest()
{
        CPU_DATA         *p_tbl;
        OS_PRIO                prio;
        prio = (OS_PRIO)0;
        p_tbl = &OSPrioTbl[0];
        while(*p_tbl == (CPU_DATA)0)
        {
                 prio += DEF_INT_CPU_NBR_BITS;  //  DEF_INT_CPU_NBR_BITS : CPU 位寬
                p_tbl ++;
         }
        prio += (OS_PRIO)CPU_CntLeadZeros(*p_tbl);
        return(prio);
}       



太晚------- 明晚繼續(xù)----------







分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂1 踩

相關(guān)帖子

回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復(fù) 返回頂部 返回列表