新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > uC/OS-II內核超時等待機制的分析

        uC/OS-II內核超時等待機制的分析

        作者: 時間:2006-12-15 來源:網絡 收藏

        1引言

         uC/OS-II是著名的源碼公開的實時內核[1],是專為嵌入式應用設計的,可用于各類8位16位和32位單片機或DSP?,F在有很多使用者正在或已經將其移植到各種類型的芯片。因為源碼公開,uC/OS-II也經常被作為嵌入式實時內核的教材,為專業人員提供了學習實時內核的難得機會。在實際使用中不管基于何種操作系統平臺,應用程序經常會等待一些系統資源,如信號量,事件標志,消息等。等待類型共有三種:(1)如果不能馬上獲取,懸掛等待;(2)不管是否能獲取資源,馬上返回,不會等待;(3) 如果不能馬上獲取資源,將進行有限時間的等待,即超時等待。

        2的基本原理

         應用程序通過操作系統提供的系統調用接口獲取資源時,在系統調用的入口參數里可以指定超時等待的最大時間,通常以毫秒為單位,內核會將其轉化為系統的時鐘滴嗒數(tick)。一般內核都會執行以下流程:

        (1)如果資源能馬上獲取,系統調用將成功返回。

        (2)如果資源不能馬上獲取,內核將設置一定時器進行計時,把當前任務懸掛在該資源的等待隊列上,該任務從就緒表中刪除,并進行調度,讓出CPU的使用權。
         
         (3)如果在指定的時間內資源變得可以獲取了,定時器應馬上停止計時,該任務從等待隊列里摘下并且重新回到就緒表中等候調度。

        (4)如果定時器到時,任務應該從等待隊列里摘下并且重新回到就緒表中,系統調用返回超時信息。

         內核在每一個tick都會做一系列的工作,包括任務的延遲以及超時等待資源的定時器等相關的檢查操作。一般來講,在指定的時間間隔以外到達的資源和信號被認為是無效的,這也是指定超時時間間隔的原意所在,有些對時間要求苛刻的場合就有這種需求,內核必須處理好這方面的問題。

        3uC/OS-II內核的分析

         假設某任務T超時等待信號量資源R,先來分析時鐘節拍函數的源代碼。

        void OSTimeTick(void)

        {

        OS_TCB *ptcb;

        OSTimeTickHook();

        ptcb=OSTCBList;

        while(ptcb->OSTCBPrio!=OS_IDLE_PRIO){

        OS_ENTER_CRITICAL();

        if(ptcb->OSTCBDly!=0){

        if(--ptcb->OSTCBDly==0){

        if(!(ptcb->OSTCBStatOS_STAT_SUSPEND)){//(1)

        OSRdyGrp|=ptcb->OSTCBBity; //(2)

        OSRdyTbl[ptcb->OSTCBY]|=ptcb->OSTCBBitX;//(3)

        }else {

        ptcb->OSTCBDly=1;

        }

        }

        }

        ptcb=ptcb->OSTCBNext;

        OS_EXIT_CRITICAL();

        }

        OS_ENTER_CRITICAL();

        OSTime++;

        OS_EXIT_CRITICAL();

        }

          語句(1),(2),(3)表明:時鐘中斷服務程序在每一個時鐘中斷在需要的情況下對任務的延遲項進行減1操作,如果任務T的定時時間間隔到期(延遲項被減為0),并且任務T沒有附加的掛起操作,任務T就會進入就緒表,然而該函數卻沒有進一步將任務T移出資源R的等待隊列,也就是說此時任務T跨了兩個狀態,這兩個狀態從本質上講是矛盾的。雖然任務T此時處于就緒狀態,但未必馬上就能獲得執行權,這取決于任務T的優先級。在任務T沒有被調度執行之前的這段時間內,假設資源R到達了,比如一個中斷服務程序調用了OSSemPost函數,會是什么情況呢?我們再來分析OSSemPost函數。

        void OSSemPost(OS_EVENT *pevent)

        {

        OS_ENTER_CRITICAL();

        if(pevent->OSEventGrp!=0x00){

        OS_EventTaskRdy(pevent,(void*)0,OS_STAT_SEM);//(4)

        OS_EXIT_CRITICAL();

        OS_Sched();

        return(OS_NO_ERR);

        }

        if(pevent->OSEventCnt65535){

        pevent->OSEventCnt++;

        OS_EXIT_CRITICAL();

        return(OS_NO_ERR);

        }

        OS_EXIT_CRITICAL();

        return(OS_SEM_OVF);

        }

        }

         從語句(4)可以看出,在資源R的等待列表中有等待任務的情況下,等待表中最高優先級的任務將從等待列表中刪除,并且進入就緒表。如果等待表中的最高優先級任務就是前面講的等待超時的任務T,這相當于任務T又一次進入就緒表,不過只有一次從等待表中刪除。任務T獲取到了資源,只不過是在超時時間以外獲取到的。任務T獲得執行權以后從調度程序返回將運行函數OSSemPend()語句(6)處的條件代碼,此時語句(5)處的條件不成立,任務按獲取到資源對待。

        void OSSemPend(OS_EVENT *pevent,INT16U timeout,INT8U *err)

        {

        OS_ENTER_CRITICAL();

        if(pevent->OSEventType!=OS_EVENT_TYPE_SEM){

        OS_EXIT_CRITICAL();

        *err=OS_ERR_EVENT_TYPE;

        }

        if(pevent->OSEventCnt>0){

        pevent->OSEventCnt--;

        OS_EXIT_CRITICAL();

        *err=OS_NO_ERR;

        }else if(OSIntNesting>0){

        OS_EXIT_CRITICAL();

        *err=OS_ERR_PEND_ISR;

        }else{

        OSTCBCur->OSTCBStat|=OS_STAT_SEM;

        OSTCBCur->OSTCBDly=timeout;

        OSEventTaskWait(pevent);

        OS_EXIT_CRITICAL();

        OSSched();

        OS_ENTER_CRITICAL();

        if(OSTCBCur->OSTCBStatOS_STAT_SEM){ //(5)

        OSEventTo(pevent);

        OS_EXIT_CRITICAL();

        *err=OS_TIMEOUT;

        }else{ //(6)

        OSTCBCur->OSTCBEventPtr=(OS_EVENT*0);

        OS_EXIT_CRITICAL();

        *err=OS_NO_ERR;

        }

        }

        }

        void OSEventTo(OS_EVENT *pevent)

        {

        if((pevent->OSEventTbl[OSTCBCur->OSTCBY]=~OSTCBCur->OSTCBBitX)==0)

        {

        pevent->OSEventGrp=~OSTCBBitY;

        }

        OSTCBCur->OSTCBStat=OS_STAT_RDY;

        vOSTCBCur->OSTCBEventPtr=(OS_EVENT*0);}

         如果任務T由于超時進入就緒態,到T獲得執行權之前,仍沒有獲取到資源R,將運行語句(5)處的條件代碼,由函數OSEventTo()可以看出,此時任務T才被從等待表中刪除,最后返回超時狀態。
         通過分析開放源碼的nucleus內核,發現nucleus在超時到期時執行定時器的一個回調函數,此回調函數馬上將等待任務從等待鏈表中刪除,將返回狀態定性為超時。這樣在任務獲得執行權前,即使資源到達,該任務也不會得到。這樣一來,uC/OS-II內核只要在時鐘節拍函數里增加代碼將延時期滿的任務從相應的資源等待列表中刪除即可。這一工作很容易實現,內核任務控制塊有指向所等待的信號量,消息等事件控制塊的指針,事件控制塊里有相應的等待表。對于uC/OS-II新引進的事件標志組[2],任務控制塊有指向相應的等待節點的指針,等待節點有指向相應的事件標志組控制塊的指針,刪除一個等待節點也能實現。

        4結論

         uC/OS-II其它資源的等待機制,比如消息以及包括2.5.2版引入的事件標志組的實現都存在上述的超時時間不嚴格的問題,這是由中斷節拍函數OSTimeTick()決定的,該函數只負責將任務移入就緒表,而不處理相應的等待表。

        本文引用地址:http://www.104case.com/article/258246.htm
        參考文獻:


        [1]Labrosse Jean J.uc/OS-II-源碼公開的實時嵌入式操作系統[M].北京:中國電力出版社,2001.

        [2]Labrosse Jean J. 嵌入式實時操作系統uc/OS-II[M].北京:北京航空航天大學出版社,2003.


        關鍵詞: 超時等待機制

        評論


        技術專區

        關閉
        主站蜘蛛池模板: 邯郸县| 侯马市| 佛冈县| 德安县| 金寨县| 赣州市| 察雅县| 福州市| 云南省| 乌鲁木齐市| 乳源| 峡江县| 丰顺县| 大方县| 唐海县| 开鲁县| 明星| 辉南县| 齐齐哈尔市| 贵南县| 扎兰屯市| 盘山县| 萝北县| 阜平县| 青铜峡市| 朝阳区| 綦江县| 南丰县| 黄大仙区| 尼木县| 西峡县| 乐都县| 长垣县| 咸阳市| 东海县| 潍坊市| 达孜县| 平定县| 桓台县| 盐源县| 翼城县|