uC/OS應(yīng)用的簡(jiǎn)單剖析
一.uC/OS-II的移植
本文引用地址:http://cafeforensic.com/article/149125.htm移植uC/OS對(duì)目標(biāo)處理器有一定要求,這個(gè)可以參照
整個(gè)嵌入式系統(tǒng)分為兩大層:硬件層和軟件層.這里主要研究軟件層的架構(gòu).
軟件層主要分為四個(gè)部分:實(shí)時(shí)操作系統(tǒng)內(nèi)核,與處理器相關(guān)部分,與應(yīng)用相關(guān)部分,用戶(hù)的應(yīng)用系統(tǒng).
l實(shí)時(shí)操作系統(tǒng)內(nèi)核
實(shí)時(shí)操作系統(tǒng)對(duì)系統(tǒng)資源進(jìn)行管理。主要包括任務(wù)分配和調(diào)度、系統(tǒng)時(shí)鐘服務(wù)、內(nèi)存管理、消息機(jī)制、異常處理等等。uC/OS所有系統(tǒng)服務(wù)均由內(nèi)核提供。內(nèi)核將應(yīng)用系統(tǒng)和底層硬件結(jié)合成一個(gè)完整的實(shí)時(shí)系統(tǒng)。
移植的時(shí)候內(nèi)核是不變的,開(kāi)發(fā)者根據(jù)自己應(yīng)用系統(tǒng)的需要來(lái)選擇實(shí)時(shí)操作系統(tǒng)內(nèi)核,開(kāi)發(fā)者不能對(duì)內(nèi)核隨意訪(fǎng)問(wèn),只能使用內(nèi)核提供的功能服務(wù)來(lái)開(kāi)發(fā)自己的應(yīng)用系統(tǒng)。內(nèi)核確定,那么所提供的系統(tǒng)管理能力,系統(tǒng)服務(wù)也就得到了限定。開(kāi)發(fā)者只能在規(guī)定的范圍內(nèi)對(duì)系統(tǒng)作些改動(dòng).
2與處理器相關(guān)的代碼
這是移植中最關(guān)鍵的部分.內(nèi)核將應(yīng)用系統(tǒng)和底層硬件有機(jī)的結(jié)合成一個(gè)實(shí)時(shí)系統(tǒng),要使同一個(gè)內(nèi)核能適用于不同的硬件體系,就需要在內(nèi)核和硬件之間有一個(gè)中間層,這就是與處理器相關(guān)的代碼.處理器不同,這部分代碼也不同.
我們?cè)谝浦矔r(shí)需要自己處理這部分代碼,可以自己編寫(xiě),也可以直接使用已經(jīng)成功移植的代碼.
在uC/OS中這一部分代碼分成三個(gè)文件:OS_CPU.H,OS_CPU_A.ASM,OS_CPU_C.C
1)OS_CPU.H
包括了用#define定義的與處理器相關(guān)的常量,宏和類(lèi)型定義.
具體來(lái)講有系統(tǒng)數(shù)據(jù)類(lèi)型定義,棧增長(zhǎng)方向定義,關(guān)中斷和開(kāi)中斷定義,系統(tǒng)軟中斷的定義等等.
2)OS_CPU_A.ASM
這部分需要對(duì)處理器的寄存器進(jìn)行操作,所以必須用匯編語(yǔ)言來(lái)編寫(xiě).包括四個(gè)子函數(shù):OSStartHighRdy(),OSCtxSw(),OSIntCtxSw(),OSTickISR().
OSStartHighRdy()在多任務(wù)系統(tǒng)啟動(dòng)函數(shù)OSStart()中調(diào)用.完成的功能是:設(shè)置系統(tǒng)運(yùn)行標(biāo)志位OSRunning=
TRUE;將就緒表中最高優(yōu)先級(jí)任務(wù)的棧指針Load到SP中,并強(qiáng)制中斷返回.這樣就緒的最高優(yōu)先級(jí)任務(wù)就如同從中斷里返回到運(yùn)行態(tài)一樣,使得整個(gè)系統(tǒng)得以運(yùn)轉(zhuǎn).
OSCtxSw()在任務(wù)級(jí)任務(wù)切換函數(shù)中調(diào)用的.任務(wù)級(jí)切換是通過(guò)SWI或者TRAP人為制造的中斷來(lái)實(shí)現(xiàn)的.ISR的向量地址必須指向OSCtxSw().這一中斷完成的功能:保存任務(wù)的環(huán)境變量(主要是寄存器的值,通過(guò)入棧來(lái)實(shí)現(xiàn)),將當(dāng)前SP存入任務(wù)TCB中,載入就緒最高優(yōu)先級(jí)任務(wù)的SP,恢復(fù)就緒最高優(yōu)先級(jí)任務(wù)的環(huán)境變量,中斷返回.這樣就完成了任務(wù)級(jí)的切換.
OSIntCtxSw()在退出中斷服務(wù)函數(shù)OSIntExit()中調(diào)用,實(shí)現(xiàn)中斷級(jí)任務(wù)切換.由于是在中斷里調(diào)用,所以處理器的寄存器入棧工作已經(jīng)做完,就不用作這部分工作了.具體完成的任務(wù):調(diào)整棧指針(因?yàn)檎{(diào)用函數(shù)會(huì)使任務(wù)棧結(jié)構(gòu)與系統(tǒng)任務(wù)切換時(shí)堆棧標(biāo)準(zhǔn)結(jié)構(gòu)不一致),保存當(dāng)前任務(wù)SP,載入就緒最高優(yōu)先級(jí)任務(wù)的SP,恢復(fù)就緒最高優(yōu)先級(jí)任務(wù)的環(huán)境變量,中斷返回.這樣就完成了中斷級(jí)任務(wù)切換.
OSTickISR()系統(tǒng)時(shí)鐘節(jié)拍中斷服務(wù)函數(shù),這是一個(gè)周期性中斷,為內(nèi)核提供時(shí)鐘節(jié)拍.頻率越高系統(tǒng)負(fù)荷越重.其周期的大小決定了內(nèi)核所能給應(yīng)用系統(tǒng)提供的最小時(shí)間間隔服務(wù).一般只限于ms級(jí)(跟MCU有關(guān)),對(duì)于要求更加苛刻的任務(wù)需要用戶(hù)自己建立中斷來(lái)解決.該函數(shù)具體內(nèi)容:保存寄存器(如果硬件自動(dòng)完成就可以省略),調(diào)用OSIntEnter(),調(diào)用OSTimeTick(),調(diào)用OSIntExit(),恢復(fù)寄存器,中斷返回.
3)OS_CPU_C.C
UC/OS中共定義了6個(gè)函數(shù)在該文件中.但是最重要的是OSTaskStkInit().其他都是對(duì)系統(tǒng)內(nèi)核的擴(kuò)展時(shí)用的.
OSTaskStkInit()是在用戶(hù)建立任務(wù)時(shí)系統(tǒng)內(nèi)部自己調(diào)用的,對(duì)用戶(hù)任務(wù)的堆棧進(jìn)行初始化.使建立好的進(jìn)入就緒態(tài)任務(wù)的堆棧與系統(tǒng)發(fā)生中斷并且將環(huán)境變量保存完畢時(shí)的棧結(jié)構(gòu)一致.這樣就可以用中斷返回指令使就緒的任務(wù)運(yùn)行起來(lái).
具體的入棧方式要根據(jù)不同mcu而定.需要參考用戶(hù)使用的mcu說(shuō)明書(shū).同時(shí)還要考慮mcu的棧生成方式.這需要根據(jù)具體問(wèn)題來(lái)分析,在此不做過(guò)多論述.
3與應(yīng)用相關(guān)的代碼
這一部分是用戶(hù)根據(jù)自己的應(yīng)用系統(tǒng)來(lái)定制合適的內(nèi)核服務(wù)功能.包括兩個(gè)文件:OS_CFG.H,INCLUDES.H.
OS_CFG.H來(lái)配置內(nèi)核,用戶(hù)根據(jù)需要對(duì)內(nèi)核進(jìn)行定制,留下需要的部分,去掉不需要的部分,設(shè)置系統(tǒng)的基本情況.比如系統(tǒng)可提供的最大任務(wù)數(shù)量,是否定制郵箱服務(wù),是否需要系統(tǒng)提供任務(wù)掛起功能,是否提供任務(wù)優(yōu)先級(jí)動(dòng)態(tài)改變功能等等.
INCLUDES.H系統(tǒng)頭文件,整個(gè)實(shí)時(shí)系統(tǒng)程序所需要的文件,包括了內(nèi)核和用戶(hù)的頭文件.
4用戶(hù)應(yīng)用系統(tǒng)
這是整個(gè)實(shí)時(shí)系統(tǒng)的最高層,用戶(hù)通過(guò)利用實(shí)時(shí)操作系統(tǒng)提供的服務(wù)來(lái)開(kāi)發(fā)自己的具體程序.
二.用戶(hù)應(yīng)用系統(tǒng)編寫(xiě)的模式
kernel提供給用戶(hù)一些功能函數(shù),使得用戶(hù)的系統(tǒng)建立更加方便,但是kernel內(nèi)部不會(huì)處理用戶(hù)的工作,對(duì)于整個(gè)系統(tǒng)的具體應(yīng)用工作還得需要用戶(hù)自己去考慮,如何利用好這些功能服務(wù)函數(shù)就成為一個(gè)比較重要的問(wèn)題.
1.main函數(shù)的結(jié)構(gòu)
voidmain(void)
{
初始化系統(tǒng)的硬件;
OSInit();
任務(wù)的建立,消息機(jī)制的建立;
OSStart();
}
這里需要的是在OSStart()執(zhí)行之前不得啟動(dòng)中斷,硬件系統(tǒng)還不能工作.必須先讓軟件系統(tǒng)進(jìn)入工作狀態(tài)后才行.
2.中斷的結(jié)構(gòu)
ISR:
{
保存處理器寄存器的值;
調(diào)用OSIntEnter();
執(zhí)行用戶(hù)的工作;
調(diào)用OSIntExit();
恢復(fù)處理器寄存器的值;
RTI;
}
用戶(hù)的中斷形式和以前一樣,沒(méi)有什么大的變化,僅僅是在原來(lái)用戶(hù)ISR的基礎(chǔ)上在固定的位置加了兩個(gè)函數(shù):OSIntEnter(),
OSIntExit().
3.各個(gè)任務(wù)的結(jié)構(gòu)
voidYourTask(void)
{
for(;;)
{
用戶(hù)代碼
調(diào)用的系統(tǒng)服務(wù)
}
}
在任務(wù)啟動(dòng)函數(shù)執(zhí)行完后,系統(tǒng)會(huì)切換到最高優(yōu)先級(jí)的任務(wù)去執(zhí)行,此時(shí),可以將系統(tǒng)硬件部分的啟動(dòng)放在該任務(wù)的最前邊,僅僅是啟動(dòng)時(shí)執(zhí)行一次,主要是啟動(dòng)系統(tǒng)的節(jié)拍中斷,或者一些必須在多任務(wù)系統(tǒng)調(diào)度后才能初始化的部分,使系統(tǒng)的真正開(kāi)始工作,達(dá)到軟件硬件的基本同步.
VoidHighestPrioTask(void)
{
OSStartHardware();
For(;;)
{
用戶(hù)代碼
調(diào)用的系統(tǒng)服務(wù)
}
}
用戶(hù)可以按照這些格式去編寫(xiě)自己的任務(wù),建立自己的應(yīng)用系統(tǒng).
評(píng)論