作為 ADEOS 的開發(fā)者(或是其他操作系統(tǒng)的開發(fā)者),你需要知道如何創(chuàng)建和使用任務。就像別的抽象數(shù)據(jù)結(jié)構(gòu),Task 類有自己的成員函數(shù)。ADEOS的的任務接口比別的大多數(shù)操作系統(tǒng)要簡單一些,因為它只是創(chuàng)建一個新的Task 對象。一旦創(chuàng)建,ADEOS 任務繼續(xù)在系統(tǒng)中存在,直到相關(guān)的函數(shù)返回。當然,這也許永遠不會發(fā)生(意即ADEOS 任務也許永遠不會結(jié)束),但是,如果一旦發(fā)生了,那么該任務就會被操作系統(tǒng)刪除掉。Task 的構(gòu)造函數(shù)如下所示。調(diào)用者通過構(gòu)造函數(shù)的參數(shù)分配一個函數(shù),一個權(quán)限值,和一個可選擇的新任務的堆棧大小。第一個參數(shù),fUnCtion,是一個指向C/C++語言或匯編語言的函數(shù)指針,該函數(shù)是要在新任務的上下文環(huán)境中運行的。該函數(shù)不需要任何輸人參數(shù),也不返回任何結(jié)果。第二個參數(shù)P,是一個單字節(jié)的整數(shù)(從1 到255),代表了任務的權(quán)限級別,這個權(quán)限級別是與別的任務相對而言的,在
tb任務調(diào)度器選擇新的任務運行的時候會用到(p 的值越大,表示權(quán)限越高)。
TaskId Task::nextId = 0
/**************************************************************
*
* Method : Task()
*
* Description : Create a new task and initialize its state.
*
* Notes :
*
* Returns :
*
**************************************************************/
Task:Task(void (*function)(), Priority p, int stackSize)
{
stackSize /= sizeof(int); //Convert bytes to words.
enterCS(); //Critical Section Begin
//
// Initialize the task-specific data.
//
if = Task::nextId++;
state = Ready;
priority = p;
entryPoint = function;
pStack = new int[stackSize];
pNext = NULL;
//
// Initialize the processor context.
//
contextInit(&context, run, this, pStack + stackSize);
//
// Insert the task into the ready list.
//
os.readyList.insert(this);
os.schedule(); // Scheduling Point
exitCS(); // Critical Section End
} /* Task() */
注意這個例程的功能塊被兩個函數(shù) enterCS()和exitCS()的調(diào)用包圍。在這些調(diào)用之間的代碼塊叫作
tb臨界區(qū)(critical section)。臨界區(qū)是一個程序必須完整執(zhí)行的一部分。也就是說,組成這一個部分的指令必須沒有中斷地按照順序執(zhí)行。因為中斷可能隨時發(fā)生,保證不受到中斷的唯一辦法就是在執(zhí)行關(guān)鍵區(qū)期間禁止中斷。因此在關(guān)鍵區(qū)的開始調(diào)用enterCS 以保存中斷的允許狀態(tài)以及禁止進一步的中斷。在關(guān)鍵區(qū)尾部調(diào)用exitCS 以恢復前面保存的中斷調(diào)用。我們會看到在下面每一個例程中都應用了同樣的技巧。
在前面代碼中,有幾個在構(gòu)造函數(shù)里調(diào)用的其他例程,但是在這里我沒有空間列出。它們是contextInit()和os.readyList.insert()例程。例程contextInit()為任務建立了初始的設(shè)備場景。這個例程必定是處理器專用的,因此是用匯編語言寫的。
contextInit()有四個參數(shù)。第一個是一個指向待初始比的設(shè)備場景數(shù)據(jù)結(jié)構(gòu)指針。第二個是一個指向啟動函數(shù)的指針。這是一個特殊的ADEOS 函數(shù),叫作run(),它被用來啟動一個任務,并且如果以后相關(guān)的函數(shù)退出了,它被用來做其后的清理工作。第三個參數(shù)是一個指向新任務對象的指針。這個參數(shù)被傳遞給run(),因此相關(guān)的任務就能夠被啟動。第四個和最后一個參數(shù)是指向新任務棧的指針。
另一個函數(shù)調(diào)用是 os.readyList.insert()。這個函數(shù)把新任務加入到操作系統(tǒng)內(nèi)部的就緒任務列表中。readyList 是一個TaskList 類型的對象。這個類是那些具有insert()和remove()兩個方法的任務(按照優(yōu)先級排序)的鏈表。感興趣的讀者如果想知道這些函數(shù)是如何實現(xiàn)的就應該下載和研究其ADEOS 的源代碼。你將在下面的討論中了解到更多有關(guān)就緒列表的問題。