<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > Small RTOS51中的一個(gè)典型問(wèn)題及其解決方法

Small RTOS51中的一個(gè)典型問(wèn)題及其解決方法

——
作者: 時(shí)間:2007-07-18 來(lái)源:電子設計信息網(wǎng) 收藏

  Small RTOS5l是一款專(zhuān)門(mén)為80C5l系列單片機設計的實(shí)時(shí)操作系統(實(shí)際上應該稱(chēng)其為實(shí)時(shí)內核),大部分代碼用C語(yǔ)言編寫(xiě),易于移植,十分適合于資源緊張的8位機。同時(shí),它也是學(xué)習嵌入式操作系統原理極好的入門(mén)材料。本人就是在學(xué)習完SmallRTOS5l的基礎上進(jìn)一步學(xué)習了著(zhù)名的uC/0S-II,受益頗多。 

1 問(wèn)題描述

  在將Smau 應用于實(shí)驗室某項目時(shí),發(fā)現了一個(gè)奇怪的問(wèn)題。簡(jiǎn)單說(shuō)來(lái),就是一個(gè)以無(wú)條件方式申請消息的任務(wù)竟然在沒(méi)有取到消息的情況下,以指示“等待超時(shí)”的代碼返回了。

  在這里,首先解釋一下任務(wù)申請消息的兩種方式:無(wú)條件方式和超時(shí)方式。所謂五條件方式是指任務(wù)申請消息時(shí),如果暫時(shí)沒(méi)有消息可取,則任務(wù)將一直等待消息,直至取到為止;而超時(shí)方式是指任務(wù)等待消息是有時(shí)間限制的,超過(guò)所設定的最大時(shí)間,即便沒(méi)有取到消息,函數也可以正常返回,只是返回值不是消息,而是“超時(shí)代碼”(此方式可以防止任務(wù)因取不到消息而被永久性?huà)炱???梢?jiàn),如果任務(wù)以無(wú)條件方式申請消息,那么函數若能夠返回,則說(shuō)明任務(wù)一定是取到消息了,而返回值又怎么可能是“等待超時(shí)”呢?經(jīng)過(guò)仔細分析SmallRTS5l的源代碼,找到了問(wèn)題產(chǎn)生的根源。

  假定有任務(wù)IDX以超時(shí)方式調用OSQPend()函數申請消息。OSQPend()函數首先會(huì )把IDX放到此消息隊列的等待任務(wù)表中,然后再去判斷隊列中是否有消息。最佳情況是隊列中確實(shí)有消息,則OSQPend()再把IDX從此消息隊列的等待任務(wù)表中刪除,接著(zhù)OSQPend()返回,任務(wù)取到消息。

  此刻,假定消息隊列中設有消息。那么,OSQPend()就會(huì )調用OSClearSigna1(OSRunningTaskID())和OS-Sched()這兩個(gè)系統函數,迫使IDX進(jìn)入休眠態(tài),同時(shí)調度器調度下一個(gè)最高優(yōu)先級的就緒任務(wù)來(lái)運行。假定任務(wù)IDY被選中,且IDY在運行中通過(guò)調用OSQIntPost()函數向此消息隊列發(fā)送了一則消息。則OSIntPost()將把所有等待這個(gè)消息隊列的任務(wù)中優(yōu)先級最高的那個(gè)任務(wù)喚醒,并且把它從該消息隊列的等待任務(wù)表中刪除,假定它就是IDX。

  當任務(wù)IDY進(jìn)入休眠態(tài)后,操作系統才會(huì )調度IDX來(lái)運行。于是IDX從上次被強迫休眠的地方開(kāi)始運行,即從OSQPend()函數中緊接著(zhù)OSSched()的那條指令開(kāi)始執行。具體來(lái)說(shuō),OSQPend()將首先查看IDX是否滿(mǎn)足超時(shí)條件(用來(lái)判斷任務(wù)是因為等待超時(shí)被喚醒的還是因為確實(shí)取到消息而被喚醒的),若超時(shí)時(shí)限尚未到達,OSQPend()再接著(zhù)檢查消息隊列中是否已經(jīng)有了消息。根據上面的假定,可以知道任務(wù)IDX確實(shí)是因為取到消息而被喚醒的。于是,OSQpend()把IDX從此消息隊列的等待任務(wù)表中刪除,OSQPend()正常返回。這樣,任務(wù)IDX取到消息,接著(zhù)運行。

  以上都沒(méi)有什么問(wèn)題,但是,有一種情況被忽略了,而正是這種情況的出現導致了任務(wù)IDX被長(cháng)時(shí)間掛起,就算隊列中有消息存在,IDX也無(wú)法被喚醒,只能等到其超時(shí)為止。

  為討論方便,不妨仍按上述假定情況來(lái)分析。當任務(wù)IDX被喚醒且IDY進(jìn)入休眠狀態(tài)后,系統必將調度下一個(gè)優(yōu)先級最高的就緒任務(wù)來(lái)運行。在前面,認為這個(gè)任務(wù)就是IDX,然而此時(shí),假定它是另一個(gè)比IDX優(yōu)先級更高的任務(wù)IDZ(因為有可能是中斷把IDZ喚醒的,所以中斷退出時(shí),操作系統強制IDY進(jìn)入休眠態(tài),轉而調度IDZ運行)。非常巧合的是,IDZ在運行的過(guò)程中向同一個(gè)消息隊列也申請了消息。由于之前IDY已經(jīng)向消息隊列發(fā)送過(guò)一條消息,則IDZ將正常取到此條消息。于是,消息隊列中的消息數減為O(Buf[0]==0)。在任務(wù)IDZ進(jìn)入休 眠后,任務(wù)IDX被操作系統調入CPU運行。同樣,函數OSQPend()首先查看IDX是否等待超時(shí)。如果沒(méi)有超時(shí)再檢查消息隊列中是否存在消息。注意到先前已經(jīng)假定消息被任務(wù)IDZ給取走了,所以檢查的結果當然是隊列中不存在消息。IDX就只好再次進(jìn)入休眠,函數OSSched()調度別的任務(wù)運行。

  于是問(wèn)題出現了。IDX是因為暫時(shí)取不到消息而被掛起的,但此時(shí)這個(gè)消息隊列的等待任務(wù)表中已經(jīng)投有IDX的蹤影了,它之前就已被那個(gè)發(fā)送消息的IDY在OSQIntPost()函數中給刪除了。

  結果,即使后面有任務(wù)再次向隊列中發(fā)送消息,IDX也取不到了,因為消息發(fā)送函數OSQIntPost()已經(jīng)無(wú)法從消息隊列的等待任務(wù)表中找到IDX了,它將被長(cháng)時(shí)間掛起,直至超時(shí)。也就是說(shuō),任務(wù)IDX明明可以取到消息的,卻取不到,最后只能以指示其等待超時(shí)的代碼返回。

  這還是一種相對來(lái)說(shuō)不太嚴重的錯誤,無(wú)非就是任務(wù)沒(méi)取到消息,以超時(shí)返回而已.如果任務(wù)IDX以無(wú)條件方式申請消息,而又恰恰發(fā)生了上面的情況,會(huì )有什么樣的后果呢?由于OSQPend()函數自身的特性,所謂五條件等待就是把超時(shí)時(shí)間設為0。結果任務(wù)IDX被喚醒后,OSQPend()必然會(huì )檢測到其已超時(shí),然后又會(huì )檢測到隊列中沒(méi)有消息,所以就必然以“超時(shí)代碼”返回。結果就發(fā)生了文章開(kāi)頭所說(shuō)的一幕;一個(gè)必須在取到消息后才能返回的任務(wù),居然在沒(méi)有取到消息的情況下以指示其等待超時(shí)的代碼返回了。

2 解決方法

  問(wèn)題已經(jīng)找到,就有解決的辦法.以《嵌入式實(shí)時(shí)操作系統SmallRT0s5l原理與應用》(陳明計,北京航空航天大學(xué)出版社,2004)中程序清單7.5為例。書(shū)中的代碼如下:
#if OS_MAX_TASKS<9              //把當前任務(wù)加入到此消息隊
                                    //列的等待任務(wù)表中
    Buf[3]=OSMapTbl[OSRunnmgTasklD()]; (5)
#else
    if(OSRunningTasklD()<8){ (6)
    Buf[3]=OSMap Tbl[OSRunningTasklD()]; (7)
    else{
    Buf[4] |= OSMapTbl[OSRunningTasklD( ) &()x07]; (8)
    }
#endif
    while(Buf[O]==0) //消息隊列中暫時(shí)投有消息 (9)
    {
#ifdef_C51_
    SP=SP+sizeof(Buf)。 (10)
    *((uint8 0S_Q_MEM_SEL * data*)(SP+l-slzeof(Buf))=Buf; (11)
#endif{
    OSClearSignal(OSRunningTask()); //當前任務(wù)進(jìn)入休眠
                                     //狀態(tài) (12)
    OSSched(); //調度下一個(gè)最高優(yōu)先級的就緒任務(wù)運行(13)
#ifdef_C51_
    Buf= *((uint8 OS_Q_MEM SEL*dota*)(SP+1-sizeof(Buf)); (14)
    SP=SP-sizeof(Buf); (15)
#endif
    if(OSWaitTick[OSRunningTasklD()]==O)(16)
    {
    break; //任務(wù)再次運行,如果超時(shí)到,退出循環(huán)
    }
    } //while(Buf[0]==O)

  修改的方法是把(5)~(8)放在(9)后面作為while()循環(huán)的第一步,其他不變。即只有在OSQPend()函數檢測到?jīng)]有消息可取的情況下,才把任務(wù)添加到對應于此消息隊列的等待任務(wù)表中。一來(lái),若隊列中已經(jīng)存在消息,這可以加快。SQPend()的執行速度;二來(lái),對于以超時(shí)方式申請消息的任務(wù),不會(huì )發(fā)生如前所述的隊列申明明有消息,任務(wù)卻取不到,只能等待至超時(shí)為止的情況。即使IDZ搶先一步取走消息,只要尚未超時(shí),IDX會(huì )再一次被OSQPend()添加到消息隊列的等待任務(wù)表中。這樣,以后運行的OSQIntPost()函數就能夠知道IDX仍然在等待消息,從而使lDX有機會(huì )獲得消息。

  此法簡(jiǎn)單易行,但對于以無(wú)條件方式申請消息的任務(wù)并不湊效。因為無(wú)條件等待時(shí),任務(wù)被喚醒,其必然滿(mǎn)足超時(shí)條件。所以無(wú)論其能否取到消息,指令都會(huì )跳出while(Buf[0]==0)循環(huán),結果就有可能返回讓人難以理解的超時(shí)代碼。這時(shí),需應用程序通過(guò)額外檢測OSQPend()的返回代碼類(lèi)型來(lái)判斷任務(wù)是否真正取到消息。

  另一種方法是仿照uC/OS—II的思路(較復雜,不推薦)。在發(fā)送消息的OSQInatPost()函數中,如果檢測到有任務(wù)正在等待此消息,則并不把消息數(Buf[0])加l,其他部分不變。這樣,即使IDZ搶在IDX再次運行前申請消息,也會(huì )因為發(fā)現隊列中沒(méi)有消息而被迫進(jìn)入休眠.但對于正在函數OSQPCnd()中等待消息的任務(wù)IDX來(lái)說(shuō),當它再次運行時(shí)也會(huì )發(fā)現消息隊列中的消息數為O。這時(shí),OSQPend()就需要另外檢測IDX是否仍然在等待任務(wù)表中來(lái)判斷任務(wù)到底能否取到消息。若任務(wù)還處在消息隊列的等待任務(wù)表中,則任務(wù)必定是由“超時(shí)”喚醒的,可直接返回超時(shí)代碼;否則,說(shuō)明是有別的任務(wù)通過(guò)發(fā)送消息將其喚醒的,則可以取到消息。這樣,以無(wú)條件方式等待消息的任務(wù)也就不會(huì )返回指示其等待超時(shí)的代碼了。只是此方法也有一個(gè)BUG,那就是如果有任務(wù)搶在IDX取走此消息之前向隊列中又發(fā)送了另一個(gè)消息,則新入隊列的消息存放位置將會(huì )重疊前一條消息(uC/OS—II中不存在此問(wèn)題)。也就是說(shuō),首先進(jìn)入隊列的那條消息被覆蓋掉了,而誰(shuí)也不知道它曾經(jīng)存在過(guò)。

  同樣的問(wèn)題也會(huì )發(fā)生在互斥型信號量的操作中,感興趣的讀者可以參看相關(guān)代碼分析。

結 語(yǔ)

  需說(shuō)明一點(diǎn),以上分析僅僅針對Small (V1.12.1)。在本文發(fā)表前,本人曾與《嵌入式實(shí)時(shí)操作系統Smell RTOS61原理與應用》一書(shū)的作者陳明計先生就此問(wèn)題通過(guò)E_mail進(jìn)行過(guò)討論。他指出:“V1.12.1的版本確實(shí)有此問(wèn)題,但在V1.20的版本中不會(huì )(盡管代碼相似)”。只是筆者在看過(guò)V1.20版本后覺(jué)得,只有當任務(wù)IDX優(yōu)先級較高時(shí),才有可能不發(fā)生此類(lèi)錯誤,否則就很難保證了。



關(guān)鍵詞: RTOS51

評論


相關(guān)推薦

技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>