2. 等待一個(gè)郵箱中的消息,OSMboxPend();
3. 發(fā)送一個(gè)消息到郵箱中,OSMboxPost();
4. 無(wú)等待地從郵箱中得到一個(gè)消息, OSMboxAccept();
5. 查詢(xún)一個(gè)郵箱的狀態(tài), OSMboxQuery();
使用郵箱之前,必須先建立該郵箱。該操作可以通過(guò)調(diào)用OSMboxCreate()函數(shù)來(lái)完成,并且要指定指針的初始值。一般情況下,這個(gè)初始值是NULL,但也可以初始化一個(gè)郵箱,使其在最開(kāi)始就包含一條消息。如果使用郵箱的目的是用來(lái)通知任務(wù)某一個(gè)事件已經(jīng)發(fā)生(發(fā)送一條消息),那么就要初始化該郵箱為NULL。如果用戶(hù)用郵箱來(lái)共享某些資源,那么就要初始化該郵箱為一個(gè)非NULL的指針。在這種情況下,郵箱被當(dāng)成一個(gè)二值信號(hào)量使用。
下面來(lái)看看創(chuàng)建一個(gè)郵箱函數(shù)的實(shí)現(xiàn)代碼:
OS_EVENT *OSMboxCreate (void *msg) { OS_EVENT *pevent; OS_ENTER_CRITICAL(); pevent = OSEventFreeList; if (OSEventFreeList != (OS_EVENT *)0) { OSEventFreeList = (OS_EVENT *)OSEventFreeList->OSEventPtr; } OS_EXIT_CRITICAL(); if (pevent != (OS_EVENT *)0) { pevent->OSEventType = OS_EVENT_TYPE_MBOX; (1) pevent->OSEventPtr = msg; (2) OSEventWaitListInit(pevent); } return (pevent); (3) }
仔細(xì)看看,其實(shí)和創(chuàng)建一個(gè)信號(hào)量的過(guò)程幾乎是一樣的,先申請(qǐng)一個(gè)空事件控制塊,接著初始化這個(gè)事件控制塊。最后返回一個(gè)指向這個(gè)事件控制塊的指針。不同之處在于事件控制塊的類(lèi)型被設(shè)置成OS_EVENT_TYPE_MBOX[ (1)],以及使用.OSEventPtr域來(lái)容納消息指針。
接著來(lái)看看等待郵箱函數(shù)實(shí)現(xiàn)代碼:
void *OSMboxPend (OS_EVENT *pevent, INT16U timeout, INT8U *err) { void *msg; OS_ENTER_CRITICAL(); if (pevent->OSEventType != OS_EVENT_TYPE_MBOX) { (1) OS_EXIT_CRITICAL(); *err = OS_ERR_EVENT_TYPE; return ((void *)0); } msg = pevent->OSEventPtr; if (msg != (void *)0) { (2) pevent->OSEventPtr = (void *)0; (3) OS_EXIT_CRITICAL(); *err = OS_NO_ERR; } else if (OSIntNesting > 0) { (4) OS_EXIT_CRITICAL(); *err = OS_ERR_PEND_ISR; } else { OSTCBCur->OSTCBStat |= OS_STAT_MBOX; (5) OSTCBCur->OSTCBDly = timeout; OSEventTaskWait(pevent); OS_EXIT_CRITICAL(); OSSched(); OS_ENTER_CRITICAL(); if ((msg = OSTCBCur->OSTCBMsg) != (void *)0) { (6) OSTCBCur->OSTCBMsg = (void *)0; OSTCBCur->OSTCBStat = OS_STAT_RDY; OSTCBCur->OSTCBEventPtr = (OS_EVENT *)0; OS_EXIT_CRITICAL(); *err = OS_NO_ERR; } else if (OSTCBCur->OSTCBStat & OS_STAT_MBOX) { (7) OSEventTO(pevent); (8) OS_EXIT_CRITICAL(); msg = (void *)0; (9) *err = OS_TIMEOUT; } else { msg = pevent->OSEventPtr; (10) pevent->OSEventPtr = (void *)0; (11) OSTCBCur->OSTCBEventPtr = (OS_EVENT *)0; (12) OS_EXIT_CRITICAL(); *err = OS_NO_ERR; } } return (msg); }
同樣,它和OSSemPend()也很相似,說(shuō)白了就是先看有沒(méi)有有用的消息,要是沒(méi)有,就把該任務(wù)掛起來(lái)。
OSMboxPend()首先檢查該事件控制塊是由OSMboxCreate()函數(shù)建立的[ (1)]。當(dāng).OSEventPtr域是一個(gè)非NULL的指針時(shí),說(shuō)明該郵箱中有可用的消息[ (2)]。這種情況下,OSMboxPend()函數(shù)將該域的值復(fù)制到局部變量msg中,然后將.OSEventPtr置為NULL[ (3)]。這正是我們所期望的,也是執(zhí)行OSMboxPend()函數(shù)最快的路徑。
如果此時(shí)郵箱中沒(méi)有消息是可用的(OSEventPtr域是NULL指針),OSMboxPend()函數(shù)檢查它的調(diào)用者是否是中斷服務(wù)子程序[ (4)]。象OSSemPend()函數(shù)一樣,不能在中斷服務(wù)子程序中調(diào)用OSMboxPend(),因?yàn)橹袛喾?wù)子程序是不能等待的。這里的代碼同樣是為了以防萬(wàn)一。但是,如果郵箱中有可用的消息,即使從中斷服務(wù)子程序中調(diào)用OSMboxPend()函數(shù),也一樣是成功的。
如果郵箱中沒(méi)有可用的消息,OSMboxPend()的調(diào)用任務(wù)就被掛起,直到郵箱中有了消息或者等待超時(shí)[ (5)]。當(dāng)有其它的任務(wù)向該郵箱發(fā)送了消息后(或者等待時(shí)間超時(shí)),這時(shí),該任務(wù)再一次成為最高優(yōu)先級(jí)任務(wù),OSSched()返回。這時(shí),OSMboxPend()函數(shù)要檢查是否有消息被放到該任務(wù)的任務(wù)控制塊中[ (6)]。如果有,那么該次函數(shù)調(diào)用成功,對(duì)應(yīng)的消息被返回到調(diào)用函數(shù)。
發(fā)送一個(gè)消息到郵箱中OSMboxPost()的代碼如下:
INT8U OSMboxPost (OS_EVENT *pevent, void *msg) { OS_ENTER_CRITICAL(); if (pevent->OSEventType != OS_EVENT_TYPE_MBOX) { (1) OS_EXIT_CRITICAL(); return (OS_ERR_EVENT_TYPE); } if (pevent->OSEventGrp) { (2) OSEventTaskRdy(pevent, msg, OS_STAT_MBOX); (3) OS_EXIT_CRITICAL(); OSSched(); (4) return (OS_NO_ERR); } else { if (pevent->OSEventPtr != (void *)0) { (5) OS_EXIT_CRITICAL(); return (OS_MBOX_FULL); } else { pevent->OSEventPtr = msg; (6) OS_EXIT_CRITICAL(); return (OS_NO_ERR); } } }
發(fā)送一個(gè)消息到郵箱和發(fā)送一個(gè)信號(hào)量也很相似,就是查看有沒(méi)有任務(wù)在等待這個(gè)消息,如果有就把那個(gè)任務(wù)從睡眠態(tài)拉回就緒態(tài)。
代碼的詳細(xì)解釋如下:
檢查了事件控制塊是否是一個(gè)郵箱后[ (1)],OSMboxPost()函數(shù)還要檢查是否有任務(wù)在等待該郵箱中的消息[ (2)]。如果事件控制塊中的OSEventGrp域包含非零值,就暗示著有任務(wù)在等待該消息。這時(shí),調(diào)用OSEventTaskRdy()將其中的最高優(yōu)先級(jí)任務(wù)從等待列表中刪除[ (3)],加入系統(tǒng)的就緒任務(wù)列表中,準(zhǔn)備運(yùn)行。然后,調(diào)用OSSched()函數(shù)[ (4)],檢查該任務(wù)是否是系統(tǒng)中最高優(yōu)先級(jí)的就緒任務(wù)。如果是,執(zhí)行任務(wù)切換[僅當(dāng)OSMboxPost()函數(shù)是由任務(wù)調(diào)用時(shí)],該任務(wù)得以執(zhí)行。如果該任務(wù)不是最高優(yōu)先級(jí)的任務(wù),OSSched()返回,OSMboxPost()的調(diào)用函數(shù)繼續(xù)執(zhí)行。如果沒(méi)有任何任務(wù)等待該消息,指向消息的指針就被保存到郵箱中[ (6)](假設(shè)此時(shí)郵箱中的指針不是非NULL的[ (5)])。這樣,下一個(gè)調(diào)用OSMboxPend()函數(shù)的任務(wù)就可以立刻得到該消息了。
如果,你對(duì)上面的內(nèi)容還有疑問(wèn),推薦選擇西部數(shù)碼企業(yè)云郵箱!有專(zhuān)人協(xié)助您解答郵箱疑問(wèn)。
西部數(shù)碼是專(zhuān)業(yè)企業(yè)郵箱的官方正規(guī)提供商,21年行業(yè)經(jīng)驗(yàn),提供安全穩(wěn)定,簡(jiǎn)單易用,高性?xún)r(jià)比的企業(yè)郵箱,按需自由定制,不限空間,極速收發(fā),能夠滿(mǎn)足用戶(hù)對(duì)企業(yè)郵箱的不同需求。可以通過(guò)以下幾種方式注冊(cè)、申請(qǐng)、購(gòu)買(mǎi)、試用、開(kāi)通企業(yè)郵箱:
1、登錄http://www.ps-sw.cn/services/mail/在線(xiàn)咨詢(xún)申請(qǐng)?jiān)囉没蛸?gòu)買(mǎi);
2、直接致電028-62778877申請(qǐng)?jiān)囉没蛘劫?gòu)買(mǎi)開(kāi)通;