μC/OS-II的任務(wù)之間的通訊與同步
制塊的.OSEventGrp域為非0值時,說明該消息隊列的等待任務(wù)列表中有任務(wù)。這時,調(diào)用
OSEventTaskRdy()函數(shù)[見6.02節(jié),使一個任務(wù)進入就緒狀態(tài),OSEventTaskRdy()]從列表中取出最高優(yōu)先級的任務(wù)[L6.23(3)], 并將它置于就緒狀態(tài)。 然后調(diào)用函數(shù)OSSched()[L6.23(4)]進行任務(wù)的調(diào)度。如果上面取出的任務(wù)的優(yōu)先級在整個系統(tǒng)就緒的任務(wù)里也是最高的,而且OSQPost()函數(shù)不是中斷服務(wù)子程序調(diào)用的,就執(zhí)行任務(wù)切換,該最高優(yōu)先級任務(wù)被執(zhí)行。否則的話,OSSched()函數(shù)直接返回,調(diào)用 OSQPost()函數(shù)的任務(wù)繼續(xù)執(zhí)行。
程序清單L6.23向消息隊列發(fā)送一條消息
INT8UOSQPost(OS_EVENT*pevent,void*msg)
{
OS_Q*pq;
OS_ENTER_CRITICAL();
if(pevent->OSEventType!=OS_EVENT_TYPE_Q){(1)
OS_EXIT_CRITICAL();
return(OS_ERR_EVENT_TYPE);
}
if(pevent->OSEventGrp){(2)
OSEventTaskRdy(pevent,msg,OS_STAT_Q);(3)
OS_EXIT_CRITICAL();
OSSched();
(4)
return(OS_NO_ERR);
}else{
pq=pevent->OSEventPtr;
if(pq->OSQEntries>=pq->OSQSize){(5)
OS_EXIT_CRITICAL();
return(OS_Q_FULL);
}else{
*pq->OSQIn++=msg;(6)
pq->OSQEntries++;
if(pq->OSQIn==pq->OSQEnd){
pq->OSQIn=pq->OSQStart;
}
OS_EXIT_CRITICAL();
}
return(OS_NO_ERR);
}
}
如果沒有任務(wù)等待該消息隊列中的消息,而且此時消息隊列未滿[L6.23(5)],指向該消息的指針被插入到消息隊列中[L6.23(6)]。這樣,下一個調(diào)用OSQPend()函數(shù)的任務(wù)就可以馬上得到該消息。注意,如果此時消息隊列已滿,那么該消息將由于不能插入到消息隊列中而丟失。
此外,如果OSQPost()函數(shù)是由中斷服務(wù)子程序調(diào)用的,那么即使產(chǎn)生了更高優(yōu)先級的任務(wù),也不會在調(diào)用OSSched()函數(shù)時發(fā)生任務(wù)切換。這個動作一直要等到中斷嵌套的最外層中斷服務(wù)子程序調(diào)用OSIntExit()函數(shù)時才能進行(見3.09節(jié),μC/OS-II中的中斷)。
6.8.4 向消息隊列發(fā)送一個消息(后進先出LIFO),OSQPostFront()
OSQPostFront()函數(shù)和OSQPost()基本上是一樣的, 只是在插入新的消息到消息隊列中時,使用.OSQOut作為指向下一個插入消息的單元的指針,而不是.OSQIn。程序清單L6.24是它的源代碼。值得注意的是,.OSQOut指針指向的是已經(jīng)插入了消息指針的單元,所以再插入新的消
息指針前,必須先將.OSQOut指針在消息隊列中前移一個單元。如果.OSQOut指針指向的當(dāng)前單
元是隊列中的第一個單元[L6.24(1)],這時再前移就會發(fā)生越界,需要特別地將該指針指向隊
列的末尾[L6.24(2)]。由于.OSQEnd指向的是消息隊列中最后一個單元的下一個單元,因此.OSQOut必須被調(diào)整到指向隊列的有效范圍內(nèi)[L6.24(3)]。因為QSQPend()函數(shù)取出的消息是
由OSQPend()函數(shù)剛剛插入的,因此OSQPostFront()函數(shù)實現(xiàn)了一個LIFO隊列。
程序清單L6.24向消息隊列發(fā)送一條消息(LIFO)
INT8UOSQPostFront(OS_EVENT*pevent,void*msg)
{
OS_Q*pq;
OS_ENTER_CRITICAL();
if(pevent->OSEventType!=OS_EVENT_TYPE_Q){
OS_EXIT_CRITICAL();
return(OS_ERR_EVENT_TYPE);
}
if(pevent->OSEventGrp){
OSEventTaskRdy(pevent,msg,OS_STAT_Q);
OS_EXIT_CRITICAL();
OSSched();
return(OS_NO_ERR);
}else{
pq=pevent->OSEventPtr;
if(pq->OSQEntries>=pq->OSQSize){
OS_EXIT_CRITICAL();
return(OS_Q_FULL);
}else{
if(pq->OSQOut==pq->OSQStart){(1)
pq->OSQOut=pq->OSQEnd;(2)
}
pq->OSQOut--;(3)
*pq->OSQOut=msg;
pq->OSQEntries++;
OS_EXIT_CRITICAL();
}
return(OS_NO_ERR);
}
}
6.8.5 無等待地從一個消息隊列中取得消息,OSQAccept()
如果試圖從消息隊列中取出一條消息,而此時消息隊列又為空時,也可以不讓調(diào)用任務(wù)等待而直接返回調(diào)用函數(shù)。這個操作可以調(diào)用OSQAccept()函數(shù)來完成。程序清單L6.25是該函數(shù)的源代碼。 OSQAccept()函數(shù)首先查看pevent指向的事件控制塊是否是由OSQCreate()函數(shù)建立的[L6.25(1)],然后它檢查當(dāng)前消息隊列中是否有消息[L6.25(2)]。如果消息隊列中有至少一條消息,那么就從.OSQOut指向的單元中取出消息[L6.25(3)]。OSQAccept()函數(shù)的調(diào)用函數(shù)需要對OSQAccept()返回的指針進行檢查。如果該指針是NULL值,說明消息隊列是空的,其中沒有消息可以[L6.25(4)]。否則的話,說明已經(jīng)從消息隊列中成功地取得了一條消息。當(dāng)中斷服務(wù)子程序要從消息隊列中取消息時, 必須使用OSQAccept()函數(shù), 而不能使用OSQPend()函數(shù)。
程序清單L6.25無等待地從消息隊列中取一條消息
void*OSQAccept(OS_EVENT*pevent)
{
void*msg;
OS_Q*pq;
OS_ENTER_CRITICAL();
if(pevent->OSEventType!=OS_EVENT_TYPE_Q){(1)
OS_EXIT_CRITICAL();
return((void*)0);
}
pq=pevent->OSEventPtr;
if(pq->OSQEntries!=0){(2)
msg=*pq->OSQOut++;(3)
pq->OSQEntries--;
if(pq->OSQOut==pq->OSQEnd){
pq->OSQOut=pq->OSQStart;
}
}else{
msg=(void*)0;(4)
}
OS_EXIT_CRITICAL();
return(msg);
}
6.8.6 清空一個消息隊列,OSQFlush()
OSQFlush()函數(shù)允許用戶刪除一個消息隊列中的所有消息,重新開始使用。程序清單L6.26是該函數(shù)的源代碼。和前面的其它函數(shù)一樣,該函數(shù)首先檢查pevent指針是否是執(zhí)行一個消息隊列[L6.26(1)],然后將隊列的插入指針和取出指針復(fù)位,使它們都指向隊列起始單元,同時,將隊列中的消息數(shù)設(shè)為0[L6.26(2)]。這里,沒有檢查該消息隊列的等待任務(wù)列表是否為空,因為只要該等待任務(wù)列表不空,.OSQEntries就一定是0。唯一不同的是,指針.OSQIn和.OSQOut此時可以指向消息隊列中的任何單元,不一定是起始單元。
評論