新聞中心

uCOS-II在ARM上的移植

作者:清華大學(xué)智能技術(shù)與系統(tǒng)國(guó)家重點(diǎn)實(shí)驗(yàn)室,李明 時(shí)間:2004-06-14 來(lái)源:電子設(shè)計(jì)應(yīng)用 收藏
摘要:本文介紹了_C/OS-II概況和體系結(jié)構(gòu)中與工作相關(guān)的一些概念,并在此基礎(chǔ)上分析了基于工作。
關(guān)鍵詞:;

引言
隨著技術(shù)的進(jìn)步,嵌入式系統(tǒng)設(shè)計(jì)及其應(yīng)用在近年中,對(duì)人類生活產(chǎn)生了巨大影響,并將繼續(xù)改變?nèi)藗兾磥?lái)的生活方式。研究嵌入式系統(tǒng),一個(gè)必不可少的基礎(chǔ)工作就是實(shí)現(xiàn)嵌入式操作系統(tǒng)在相關(guān)處理器平臺(tái)上的移植。本文基于目前應(yīng)用非常廣泛的ARM處理器體系結(jié)構(gòu),對(duì)嵌入式實(shí)時(shí)操作系統(tǒng)內(nèi)核的移植工作做了分析和介紹,并給出了在國(guó)內(nèi)一個(gè)開(kāi)源項(xiàng)目 SkyEye 仿真器上的移植實(shí)例。



表2 CSPR的模式位


表3 ARM寄存器的命名和含義

概述
uC/OS-II 是一個(gè)簡(jiǎn)單、高效的嵌入式實(shí)時(shí)操作系統(tǒng)內(nèi)核,被應(yīng)用到各種嵌入式系統(tǒng)中。目前,它支持 x86、ARM、PowerPC、MIPS 等眾多體系結(jié)構(gòu),并有上百個(gè)商業(yè)應(yīng)用實(shí)例,其穩(wěn)定性和可用性是經(jīng)過(guò)實(shí)踐驗(yàn)證的。同時(shí),它的源代碼公開(kāi),可以從www.ucos-ii.com網(wǎng)站上獲得全部源碼以及其在各種體系結(jié)構(gòu)平臺(tái)上的移植范例。
最新的 uC/OS-II 2.0 版以上的內(nèi)核都具有可搶占的實(shí)時(shí)多任務(wù)調(diào)度功能,另外它還提供了許多系統(tǒng)服務(wù),例如信號(hào)量、消息隊(duì)列、郵箱、內(nèi)存管理、時(shí)間函數(shù)等,這些功能可以根據(jù)不同的需求進(jìn)行裁減。可以說(shuō),uC/OS-II是一個(gè)具備現(xiàn)代操作系統(tǒng)特點(diǎn)的RTOS,同時(shí)它結(jié)構(gòu)清晰、注解詳盡,具有良好的可擴(kuò)展性和可移植性,被廣泛地應(yīng)用于各種架構(gòu)的微處理器上。

ARM 的體系結(jié)構(gòu)
ARM是目前嵌入式領(lǐng)域由應(yīng)用最廣泛的RISC微處理器結(jié)構(gòu),以其低成本、低功耗、高性能等優(yōu)點(diǎn)占據(jù)了嵌入式系統(tǒng)應(yīng)用領(lǐng)域的領(lǐng)先地位。ARM系列的處理器當(dāng)前有ARM7、ARM9、ARM9E、ARM10等多個(gè)產(chǎn)品,此外,作為ARM公司合作伙伴的Intel,也提供基于XScale微體系結(jié)構(gòu)的相關(guān)處理器產(chǎn)品。所有的ARM處理器都共享ARM通用的基礎(chǔ)體系結(jié)構(gòu),開(kāi)發(fā)者在不同的ARM處理器上做操作系統(tǒng)移植時(shí),將大大節(jié)省工作量,降低軟件開(kāi)發(fā)成本。
任何操作系統(tǒng)的移植都有相當(dāng)一部分工作和所用處理器的體系結(jié)構(gòu)密切相關(guān),因此在做具體的移植工作之前,需要了解該處理器的體系結(jié)構(gòu)和相應(yīng)的匯編語(yǔ)言。下面將簡(jiǎn)要介紹ARM的體系結(jié)構(gòu),主要側(cè)重于與移植相關(guān)的一些概念。
處理器模式(CPU Mode):
ARM處理器可以工作在7種模式下,如表1所示。

除usr模式以外的其他模式都叫做特權(quán)模式,除 usr 和 sys 外的其他5種模式叫做異常模式。在usr模式下,對(duì)系統(tǒng)資源的訪問(wèn)是受限制的,也無(wú)法主動(dòng)地改變處理器模式。異常模式通常都和硬件相關(guān),例如中斷或執(zhí)行未定義指令等。與移植相關(guān)的處理器模式有兩種:svc態(tài)和irq態(tài),分別指操作系統(tǒng)的保護(hù)模式和通用中斷處理模式。這兩種模式之間的轉(zhuǎn)換可以通過(guò)硬件方式或軟件方式。uC/OS-II內(nèi)核在執(zhí)行過(guò)程中,大部分時(shí)間工作在svc態(tài),當(dāng)有硬件中斷,例如時(shí)鐘中斷到來(lái)時(shí),CPU硬件自動(dòng)完成從svc態(tài)進(jìn)入irq態(tài),在中斷程序結(jié)束處,則需要通過(guò)編程的方法使得CPU從irq態(tài)恢復(fù)到svc態(tài)。
程序狀態(tài)寄存器( PSR:Program status register )在任意一種處理器模式中,都使用同一個(gè)寄存器來(lái)標(biāo)識(shí)當(dāng)前處理器的工作模式,這個(gè)寄存器叫做 CPSR ( Current Program Status Register ),它的 [0--4] 位用來(lái)表示CPU Mode,表2 為CSPR的模式位每一種處理器異常模式,都有一個(gè)對(duì)應(yīng)的SPSR ( Saved Program Status Register )寄存器,用來(lái)保存進(jìn)入異常模式前的CPSR。SPSR的作用就是當(dāng)CPU從異常模式退出時(shí),通過(guò)一條簡(jiǎn)單的匯編指令就能夠恢復(fù)進(jìn)入異常模式前的CPSR,該值保存在當(dāng)前異常模式的SPSR中。例如:當(dāng)從usr態(tài)進(jìn)入中斷irq 態(tài)時(shí),原先的 CPSR_all 將被保存在當(dāng)前的 SPSR_irq中,類似的異常模式下的 SPSR 還有 SPSR_fiq、SPSR_svc、SPSR_abt、SPSR_und。非異常模式的 usr和sys模式下沒(méi)有 SPSR,只有 CPSR。不能顯式地指定把 CPSR 保存到某個(gè)異常模式下的 SPSR,比如 SPSR_irq,而必須是變更到 irq 態(tài)之后CPU自動(dòng)完成,不能硬性賦值,因?yàn)镾PSR_irq在其他狀態(tài)下不可見(jiàn)。
1、 ARM寄存器:( register )
ARM處理器共有37個(gè)寄存器,其中31個(gè)是通用寄存器,包括一個(gè)程序計(jì)數(shù)器 PC。另外6個(gè)就是上面提到的程序狀態(tài)寄存器。
·通用寄存器:
i. R0-R7:與所有處理器模式無(wú)關(guān)的寄存器,可以用作任何用途。
ii. R8-R14:與處理器模式有關(guān)的寄存器,在不同的模式下,對(duì)應(yīng)到不同的物理寄存器。其中 R13又叫做SP,一般用于堆棧指針。R14又叫做lr,一般用于保存返回地址。這兩個(gè)寄存器在每種異常模式下都對(duì)應(yīng)到不同的物理寄存器上,例如LR_irq、LR_svc、LR_fiq 等。
iii. R15:又叫做程序計(jì)數(shù)器,即PC,所有的模式下都使用同一個(gè)PC。
·狀態(tài)寄存器:
iv. CPSR:當(dāng)前程序狀態(tài)寄存器,所有的模式下都使用同一個(gè) CPSR。
v. SPSR:保存的程序狀態(tài)寄存器,每種異常模式下都有自己的SPSR,一共有5種SPSR,即 SPSR_irq、SPSR_fiq、SPSR_svc、SPSR_abt、SPSR_und。usr和sys 態(tài)下沒(méi)有 SPSR 。
所有的ARM寄存器的命名和含義,可以用表3來(lái)說(shuō)明,其中相同命名的都是同一個(gè)物理寄存器,不同命名的寄存器都對(duì)應(yīng)不同的物理寄存器。
表3 ARM寄存器的命名和含義

移植工作介紹
實(shí)際上uC/OS-II可以簡(jiǎn)單地看作是一個(gè)多任務(wù)調(diào)度器,在這個(gè)任務(wù)調(diào)度器上完善地添加了與多任務(wù)操作系統(tǒng)相關(guān)的一些系統(tǒng)服務(wù),如信號(hào)量、郵箱等。其90%的代碼是用C語(yǔ)言寫(xiě)的,可以直接移植到有C語(yǔ)言編譯器的處理器上。移植工作主要都集中在多任務(wù)切換的實(shí)現(xiàn)上,因?yàn)檫@部分代碼用來(lái)保存和恢復(fù)CPU現(xiàn)場(chǎng)(即寫(xiě)/讀相關(guān)寄存器),不能用C語(yǔ)言,只能使用匯編語(yǔ)言完成。
uC/OS-II的全部源代碼量大約是6000-7000行,共15個(gè)文件。將 uC/OS-II 移植到ARM處理器上,需要修改三個(gè)與ARM體系結(jié)構(gòu)相關(guān)的文件,代碼量大約是500行。以下分別介紹這三個(gè)文件的移植工作。
OS_CPU.H 文件


圖1 ARM體系結(jié)構(gòu)的寄存器位置


圖2 任務(wù)堆棧初始化程序
·數(shù)據(jù)類型定義
數(shù)據(jù)類型的修改與所用的編譯器相關(guān),不同的編譯器使用不同的字節(jié)長(zhǎng)度表示同一數(shù)據(jù)類型,比如int,同樣在x86平臺(tái)上,GNU的gcc編譯為4 bytes,而MS VC++則編譯為2 bytes。本文使用GNU 的 arm-elf-gcc,相關(guān)的數(shù)據(jù)類型定義如下:
·堆棧單位
在任務(wù)切換時(shí),CPU現(xiàn)場(chǎng)的寄存器將保存在當(dāng)前運(yùn)行任務(wù)的堆棧中,所以O(shè)S_STK 數(shù)據(jù)類型應(yīng)該與CPU的寄存器長(zhǎng)度一致。
typedef unsigned int os STK;
·堆棧增長(zhǎng)方向
堆棧由高地址向低地址增長(zhǎng),也與編譯器有關(guān),在函數(shù)調(diào)用時(shí),入口參數(shù)和返回地址一般保存在當(dāng)前任務(wù)的堆棧中,編譯器的編譯選項(xiàng)和由此生成的堆棧指令就會(huì)決定堆棧的增長(zhǎng)方向。
#define OS_STK_GROWTH
·宏定義
包括開(kāi)關(guān)中斷的宏定義,以及進(jìn)行任務(wù)切換的宏定義。
#define OS_ENTER_CRITICAL() ARMDisable Int()
#define OS_EXIT_CRITICAL() ARMEnable Int()
#define OS_TASK_SW() OSCtxSw()

OS_CPU_C.C 文件
·任務(wù)堆棧初始化
在此討論任務(wù)初始化時(shí)的堆棧設(shè)計(jì),也就是在堆棧增長(zhǎng)方向上如何定義每個(gè)需要保存的寄存器位置。在ARM體系結(jié)構(gòu)下,任務(wù)堆??臻g由高至低依次將保存著pc、lr、r12、r11、r10、... r1、r0、CPSR、SPSR,如圖1所示。
圖1 ARM體系結(jié)構(gòu)的寄存器位置
有兩點(diǎn)需要說(shuō)明:一是,當(dāng)前任務(wù)堆棧初始化完成后,OSTaskStkInit 返回新的堆棧指針STK,在 OSTaskCreate()執(zhí)行時(shí),將會(huì)調(diào)用 OSTaskStkInit 的初始化過(guò)程,然后通過(guò)OSTCBInit()函數(shù)調(diào)用,將返回的SP指針保存到該任務(wù)的TCB塊中。二是,初始狀態(tài)的堆棧是模擬了一次中斷后的堆棧結(jié)構(gòu),因?yàn)槿蝿?wù)創(chuàng)建后并不是直接就獲得執(zhí)行,而是通過(guò)OSSched()函數(shù)進(jìn)行調(diào)度分配,滿足執(zhí)行條件后才能獲得執(zhí)行。為了使這個(gè)調(diào)度簡(jiǎn)單一致,就預(yù)先將該任務(wù)的PC指針和返回地址LR都指向函數(shù)入口,以便被調(diào)度時(shí)從堆棧中恢復(fù)剛開(kāi)始運(yùn)行時(shí)的CPU現(xiàn)場(chǎng)。
·系統(tǒng)鉤子函數(shù)
在該文件中需要實(shí)現(xiàn)幾個(gè)操作系統(tǒng)規(guī)定的hook函數(shù),如下:
OSSTaskCreateHook( )
OSTaskDelHook( )
OSTaskSwHook( )
OSTaskStatHook( )
OSTimeTickHook( )
若無(wú)特殊需求,只需簡(jiǎn)單地將它們都實(shí)現(xiàn)為空函數(shù)即可。
OS_CPU_A.S 文件
·OSStartHighRdy()
此函數(shù)是在OSStart()多任務(wù)啟動(dòng)后,負(fù)責(zé)從最高優(yōu)先級(jí)任務(wù)的TCB控制塊中獲得該任務(wù)的堆棧指針SP,通過(guò)SP依次將CPU現(xiàn)場(chǎng)恢復(fù),這時(shí)系統(tǒng)就將控制權(quán)交給用戶創(chuàng)建的該任務(wù)進(jìn)程,直到該任務(wù)被阻塞或者被其他更高優(yōu)先級(jí)的任務(wù)搶占CPU。該函數(shù)僅在多任務(wù)啟動(dòng)時(shí)被執(zhí)行一次,即執(zhí)行最高優(yōu)先級(jí)任務(wù),之后多任務(wù)的調(diào)度和切換由以下函數(shù)實(shí)現(xiàn)。
·OSCtxSw()
任務(wù)級(jí)的上下文切換,當(dāng)任務(wù)因?yàn)楸蛔枞鲃?dòng)請(qǐng)求CPU調(diào)度時(shí)被執(zhí)行,由于此時(shí)的任務(wù)切換在非異常模式下進(jìn)行,因此區(qū)別于中斷級(jí)別的任務(wù)切換。它的工作是先將當(dāng)前任務(wù)的CPU現(xiàn)場(chǎng)保存到該任務(wù)堆棧中,然后獲得最高優(yōu)先級(jí)任務(wù)的堆棧指針,從該堆棧中恢復(fù)此任務(wù)的CPU現(xiàn)場(chǎng),使之繼續(xù)執(zhí)行。這樣就完成了一次任務(wù)切換。
·OSIntCtxSw()
中斷級(jí)的任務(wù)切換,在時(shí)鐘中斷ISR(中斷服務(wù)例程)中發(fā)現(xiàn)有高優(yōu)先級(jí)任務(wù)等待的時(shí)鐘信號(hào)到來(lái),則在中斷退出后并不返回被中斷任務(wù),而是直接調(diào)度就緒的高優(yōu)先級(jí)任務(wù)執(zhí)行,從而能夠盡快地讓高優(yōu)先級(jí)的任務(wù)得到響應(yīng),保證系統(tǒng)的實(shí)時(shí)性能。其原理基本上與任務(wù)級(jí)的切換相同,但是由于進(jìn)入中斷時(shí)已經(jīng)保存了被中斷任務(wù)的CPU現(xiàn)場(chǎng),因此不用再進(jìn)行類似的操作,只需對(duì)堆棧指針做相應(yīng)調(diào)整。
·OSTickISR()
時(shí)鐘中斷處理函數(shù),其主要任務(wù)是負(fù)責(zé)處理時(shí)鐘中斷,調(diào)用系統(tǒng)實(shí)現(xiàn)的OSTimeTick函數(shù),如果有等待時(shí)鐘信號(hào)的高優(yōu)先級(jí)任務(wù),則需要在中斷級(jí)別上調(diào)度其執(zhí)行。其他相關(guān)的兩個(gè)函數(shù)是OSIntEnter()和OSIntExit(),都需要在ISR中執(zhí)行。
·ARMEnableInt()& ARMDisableInt()
分別是退出臨界區(qū)和進(jìn)入臨界區(qū)的宏指令實(shí)現(xiàn)。主要用于在進(jìn)入臨界區(qū)之前關(guān)閉中斷,在退出臨界區(qū)的時(shí)候恢復(fù)原來(lái)的中斷狀態(tài)。它的實(shí)現(xiàn)比較簡(jiǎn)單,可以直接開(kāi)關(guān)中斷來(lái)實(shí)現(xiàn),也可以通過(guò)保存關(guān)閉/恢復(fù)中斷屏蔽位來(lái)實(shí)現(xiàn)。
全部移植代碼在SkyEye仿真器上調(diào)試通過(guò),在SkyEye的主頁(yè)上可以下載獲得(http://hpclab.cs.tsinghua.edu.cn/~skyeye/)。

結(jié)語(yǔ)
uC/OS-II作為一個(gè)優(yōu)秀的實(shí)時(shí)操作系統(tǒng)已經(jīng)被移植到各種體系結(jié)構(gòu)的微處理器上,而ARM體系結(jié)構(gòu)在嵌入式領(lǐng)域也獲得了廣泛的應(yīng)用和支持。將uC/OS-II 移植到ARM平臺(tái)上,能夠使我們更深入地了解實(shí)時(shí)操作系統(tǒng)的構(gòu)造,加快在ARM平臺(tái)上的應(yīng)用和開(kāi)發(fā),并為更高層次上的擴(kuò)展和改進(jìn)打下基礎(chǔ)。

參考文獻(xiàn)
1 ARM Architecture Reference Manual. http://www.arm.com
2 《ARM 嵌入式處理器結(jié)構(gòu)與應(yīng)用基礎(chǔ)》.馬忠梅等編著.北京航空航天大學(xué)出版社.2002年出版
3《uC/OS-II -源碼公開(kāi)的實(shí)時(shí)嵌入式操作系統(tǒng)》.Jean J.Labrosse 著.劭貝貝譯.中國(guó)電力出版社.2001年出版
4 uC/OS 網(wǎng)站.http://www.ucos-ii.com



關(guān)鍵詞: ARM uC/OS-II 移植

評(píng)論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉