典型范例:uCOSii在Coldfire MCF52235上的移植
引言
C/ OS 是一種多任務實時操作系統(tǒng)。內核源代碼公開、短小精干、可裁剪、執(zhí)行時間可確定, 可移植性較強, 非常適用于一些中小型嵌入式系統(tǒng)開發(fā)。uC/OS 可以移植到8~ 64 位的不同類型、不同規(guī)模的嵌入式系統(tǒng), 并能在大部分的8 位、16 位、32 位, 甚至64 位的微處理器和DSP上運行[ 1] 。
MCF52235 是飛思卡爾公司Co ldf ire 系列32 位單片機解決方案的嵌入式微控制器, 采用的是V2 版本的
RISC 內核。MCF52235 內部有32 KB SRAM 和256 KB FLASH, 并且集成了標準的Coldfire 外圍設備, 包括三個適合中長距離通信的SCI, 一個I2 C 和一個用于系統(tǒng)內部和外圍設備通信的Q SPI。在60Hz的核心頻率下, MCF52235 的處理能力為56 MIPS, 具備較高的性能價格比[ 24] 。MCF52235 對于移植C/ OS 來說有足夠的
RAM 和FLASH, 且有較快的處理速度和較低的成本,所以對于嵌入式應用系統(tǒng)的開發(fā)來說, 嵌入C/ OS
到MCF52235 微控制器是一個不錯的選擇。uC/ OS 的體系結構要實現(xiàn)C/ OS 向MCF52235 的移植, 需要做兩方面的工作: 一是重新定義內核的大小和功能; 二是為內核編寫與硬件相關的代碼。C/ OS 的文件結構如圖1 所示。可以看到, C/ OS 與CPU 類型無關的C 代碼文件COS . C 包括很多文件, 它們是C/ OS 的內核和很多功能函數(shù), 其中前三個文件是實時內核、任務管理和時鐘節(jié)拍, 這三個文件是一定要用的。后面6 個功能函數(shù)用于任務間的通信, 應用程序中可能只用到其中
的幾個, 不用的可以不包含進去, 以免編譯時生成沒用的代碼。這部分代碼與CPU 類型無關, 在移植時, 這些文件不要改動。配置文件OS_CFG. H 需要根據應用要求來進行,主要作用是確定C/ OS 提供的系統(tǒng)功能函數(shù), 應用
程序用哪些和不用哪些, 這個文件移植時需要修改。與CPU 類型有關的代碼文件主要有三個: OS _CPU. H, OS_CPU_A. ASM 和OS_CPU_C. C。文件定義用于特定CPU 的數(shù)據類型來定義相關的宏。OS _CPU_A . ASM 是用匯編語言寫的與硬件有關的代碼,OS_CPU_C. C 是用C 語言寫的與硬件有關的代碼。由于移植使用C 交叉編譯工具, 在C 代碼中可以插入匯編語句, 在移植中可將這兩個文件合并成一個文件[ 5] 。
產生時鐘節(jié)拍的定時中斷來自微控制器內部, 但并非來自V2 內核內部, 可以用實時時鐘產生定時中斷,
也可以用片內的外設模塊定時器單元來產生定時中斷,這部分代碼顯然與硬件相關, 移植時要自己寫[ 6] 。
2 移植過程
所謂移植, 就是使一個實時內核能在某個微處理器或微控制器上運行。為了方便移植, 大部分的C/ OS代碼是用C 語言寫的, 但仍需要用C 和匯編語言寫一些與處理器相關的代碼, 這是因為C/ OSII 在讀寫處理器寄存器時只能通過匯編語言來實現(xiàn) 。移植過程主要包括移植前的準備、BSP ( 板級支持包) 的編寫和與處理器相關代碼的修改和編寫。C/OS 核心代碼、與CPU 相關的接口程序、BSP 和用戶應用程序之間的關系如圖2 所示。
2. 1 移植前的準備
進入C/ OS 官方網站下載C/ OS 源代碼。打開Codew arrior 6. 4 建立MCF52235 的工程文件, 然后把C/ OS 的源代碼文件加入到工程里面[ 8] 。其中有幾個地方需要改動:
( 1) 下載的源代碼中os_cfg _r. h 改為o s_cfg. h;os_dbg_r. c改為os_dbg. c。
( 2) 由于會引起重復定義錯誤, 需要把源代碼中重復包含的文件注釋掉。
( 3) 需要在INT ERNAL_FLASH 模式下編譯, 而不能在RAM 模式下, 否則會產生溢出錯誤。
2. 2 編寫B(tài)SP
板級支持包( BSP) 是介于底層硬件和操作系統(tǒng)之間的軟件層次, 負責進行系統(tǒng)啟動后最初的硬件和軟件
初始化, 并對底層硬件進行封裝, 使得操作系統(tǒng)不再面對具體的硬件[ 9] 。在此建立兩個BSP 文件: BSP. ASM 和BSP. C。其中, BSP. ASM 中包含了匯編語言寫的中斷接口程序。BSP. C 中包含了硬件和軟件的初始化程序和產生時鐘節(jié)拍的中斷服務程序。
2. 3 與處理器相關代碼的修改和編寫
有三個與處理器相關的文件, 即OS_CPU . H, OS_CPU _ A. ASM 和OS _ CPU _ C. C 需要修改。由于MCF52235 有eMAC 模塊, 所以還需要編寫OS_CPU _I. ASM 文件, 用來在任務切換和中斷時以及中斷返回
時保存和恢復相關寄存器。
2. 3. 1 OS_CPU. H 的移植
OS_CPU. H 包含了一些與處理器和編譯器相關的宏定義和數(shù)據類型定義。由于使用Codew arrior 編譯
器, shor t 類型是16 位的, int 類型是32 位的。MCF52235 的堆棧是32 位寬的, 因此OS_STK 定義為
32 位, 所有任務的堆棧必須聲明使用OS_ST K 這種數(shù)據類型。數(shù)據類型定義如下:
ty pedef unsigned char BOOLEAN;
ty pedef unsigned char INT 8U;
ty pedef signed char INT8S;
ty pedef unsigned sho rt INT16U;
ty pedef signed shor t INT16S;
ty pedef unsigned int INT32U;
ty pedef signed int INT32S;
typedef floatFP32;
typedef double FP64;
typedef unsigned int OS_STK;
typedef unsigned shor t OS_CPU_SR;
( 1) 臨界區(qū)域處理。像所有的實時性內核一樣, 在進入代碼臨界區(qū)時要關中斷, 完成時要開中斷。C/
OS 定義了兩個宏來關閉和使能中斷: OS_ENT ER_CRITICAL( ) 和OS_EXIT _CRIT ICAL( ) 。C/ OS定義了三種方法來關閉和使能中斷, 大多數(shù)情況下選擇第三種方法。
# define OS_CRITICAL_METH OD # 3
# define OS_ENTER_CRITICAL( ) { cpu_sr = OS _CPU_
SR_Save( ) ; } / / 關中斷
# def ineOS _ EXIT _ CRITICAL( ) { OS _ CPU _ SR_ Resto re
( cpu_sr) ; } / / 開中斷
( 2) 任務層上下文切換。當C/ OS 調用OS _TASK_SW( ) 時發(fā)生任務層的上下文切換。因為上下
文切換是根據處理器的不同而不同的, 所以需要執(zhí)行一個匯編的函數(shù)。在這種情況下, 用TRA P 指令來產
生一個異常, 用T RAP 指令的優(yōu)點是能使它像發(fā)生了一次中斷一樣。這里用# 14 T RAP, 因為大多數(shù)情況
下, # 15 TRAP 被調試和監(jiān)控程序保留了。# 14TRAP 定位于VBR+ 0x00B8, 然后跳轉到相應的地址。
在這個向量處放置OSCtx Sw( ) 的地址。這個函數(shù)聲明在OS_CPU_A. ASM 里。VBR 代表向量基址寄存器,
包含異常向量表的基址, 程序開始時被初始化為0x00000000, 但是在運行時可以改變。
# define OS_TASK_SW( ) asm( T RAP # 14; )
( 3) 堆棧的增長方向。MCF52235 的堆棧增長方向是從高地址向低地址, 因此OS _ST K_GROWTH 置
為1。
# define OS_STK_GROWTH 1
2. 3. 2 OS_CPU _C. C 的移植
OS_CPU_C. C 里面包含10 個比較簡單的C 語言函數(shù), 一般來說C/ OS 只需要OST askStkInit ( ) 。其他函數(shù)是用來讓用戶在自己的程序里擴展操作系統(tǒng)功能的。如果需要使用這些函數(shù), 需要在OS_CFG. H 里設置OS_CPU _HOOKS_EN 為1。堆棧的初始化: OSTaskStkInit ( ) 雖然是用C 語言編
寫的, 但它是一個與CPU 硬件相關的函數(shù)。這個函數(shù)功能是初始化任務的堆棧, 由建立任務函數(shù)OSTask
Create( ) 或擴展地建立任務函數(shù)OSTaskCreateExit ( ) 調用。任務堆棧初始化的實質就是模擬一次中斷, 使堆??雌饋砭拖駝偘l(fā)生過中斷一樣。任務堆棧中保存了任務代碼的起始地址和一些CPU 寄存器的值, 一旦條件滿足, 就可以執(zhí)行該任務。初始化后的任務堆棧結構如圖3所示。
2. 3. 3 OS_CPU_A. ASM 的移植
這個文件包含5 個相當簡單的匯編函數(shù), 因為一般不能用C 語言來保存和恢復寄存器。
( 1) OS_CPU_SR_Save( )
這個函數(shù)是通過保存中斷屏蔽寄存器, 然后關閉中斷來實現(xiàn)OS_CRITICAL_MET HOD # 3 的。當函數(shù)返回時, D0 包含了狀態(tài)寄存器的內容, 里面包含當前的中斷關閉狀態(tài)。這個返回值被調用函數(shù)保存到變量
cpu_sr 中。
( 2) OS_CPU_SR_Restore( )
這個函數(shù)用來實現(xiàn)恢復中斷屏蔽到調用OS _ENTER_CRITICAL( ) 之前的狀態(tài)。也就是說調用OS_
ENTER_CRITICAL( ) 之前中斷是關閉的, 那么在OS_EXIT_CRITICAL( ) 之后, 中斷是關閉的。
( 3) OSStartHighRdy( )
這個函數(shù)被OSStar t ( ) 調用來運行優(yōu)先級最高的任務。OSStar t ( ) 設置OSTCBHighRdy 指向優(yōu)先級最高任務的OS _T CB。一旦從OSTaskSwHoo k( ) 返回,就把OSRunning 設為OS_T RU E, 它表明現(xiàn)在RT OS
將要運行。從最高優(yōu)先級任務的OS_T CB 中恢復堆棧指針, 然后從任務堆棧里取出CPU 寄存器。最后執(zhí)行
一個RET 指令, 這個指令可以從堆棧中彈出SR 和PC,現(xiàn)在的任務代碼就開始執(zhí)行。
( 4) OSCtx Sw( )
當一個任務不再運行時就會發(fā)生一個任務級的任務切換, 比如任務調用一個延遲10 個時鐘節(jié)拍的函數(shù)。
這時, C/ OS 需要找出下一個最重要的任務準備去運行。OSCtx Sw ( ) 的功能是保存需