使用ADS移植uC/OS-II實(shí)例分析
目前可以用來(lái)編譯鏈接產(chǎn)生 ARM 處理器執行代碼的開(kāi)發(fā)工具主要有如下幾類(lèi):1. ARM 公司提供的 ARM Developer Suite 集成開(kāi)發(fā)環(huán)境主要工具有 armasm、armcc、armlink、fromelf 等。
2. GNU 組織提供的 tool chain for arm主要工具有 arm-elf-gcc、arm-elf-gdb、arm-elf-objcopy 等3. Microsoft公司提供的 eMbedded Visual Tools主要工具有 clarm、clthumb、c2_arm、link、lib等這里我們選用 ARM 公司提供的 ADS 下的工具集來(lái)編譯我們的程序和鏈接目標代碼并最終生成可執行的二進(jìn)制映像。這里介紹一下主要會(huì )用到的一些工具:armasm.exe : 匯編文件編譯器armcc.exe : C 文件編譯器armlink.exe : 目標文件連接器fromelf.exe : 用于將 axf 或者 elf 格式轉換成其他格式的文件,例如二進(jìn)制映像。
armprof.exe : 對調試過(guò)程中生成的 profiling 記錄文件做分析用的工具軟件啟動(dòng)代碼由于板子的 0x0 地址處是 32M 的Flash ROM,因此在板子加電后,會(huì )從 Flash 中順序執行啟動(dòng)代碼。為了能使得mC/OS-II 運行,啟動(dòng)代碼需要完成如下工作:1. 設置 異常向量表,即在 0x0-0x1c 位置放置7條跳轉指令(其中 0x14 為空)
2.分別實(shí)現每種異常的處理程序,其中包括 Reset_Handler、Undefined_Handler、SWI_Handler、Prefetch_Handler、Abort_Handler、IRQ_Handler、FIQ_Handler. 3. 程序從 Reset_Handler 進(jìn)入后,需要首先進(jìn)行相關(guān)硬件的初始化操作,例如 初始化SDRAM、CPU speed、Interrupt Controller、UART、timer 等。
4. 建立每種異常狀態(tài)下的系統堆棧,為了簡(jiǎn)單起見(jiàn)可以只在 svc 態(tài) 和 irq 態(tài)下建立堆棧:setup_svc_stack,setup_irq_stack. 5. 強制 arm 處理器狀態(tài)轉換為 svc 管理態(tài)。
6. 跳轉到mC/OS-II 代碼的 main 入口,實(shí)際上是編譯鏈接后產(chǎn)生的 __main 入口。
時(shí)鐘與中斷處理時(shí)鐘控制邏輯在圖1中,有4種和系統時(shí)鐘相關(guān)寄存器,它們的含義如下:● OSCR: 一個(gè)自動(dòng)遞增計數的 32 位計數器。
● OSMR3-0: 4 個(gè) 32 位的匹配寄存器,當 OSCR 的值匹配時(shí)產(chǎn)生中斷。
● OSSR: 狀態(tài)寄存器,當 OSCR 和 OSMR 匹配時(shí),會(huì )對 OSSR 做標志。
● OIER: 使能寄存器,表示當匹配發(fā)生時(shí),允許在 OSSR 設置一個(gè)標識位。
OSCR 在自動(dòng)累加的過(guò)程中,與OSMR里面設定的那些匹配寄存器進(jìn)行匹配,發(fā)現有匹配的事件時(shí),就會(huì )對 OSSR 中的相應位置設一個(gè)標志位“1”,表示OSCR與對應的OSMR 發(fā)生了匹配。當然這個(gè)匹配發(fā)生的前提是發(fā)生匹配的那個(gè)OSMR在OIER中的相應位被使能,否則OSMR中的設置將不起作用。
系統時(shí)鐘初始化流程mC/OS-II 中創(chuàng )建的第一個(gè)任務(wù)將負責啟動(dòng)時(shí)鐘節拍,時(shí)鐘的初始化設置流程如下:1) 設置 OSMR0 = x ,表示 初始化 OSMR0,即當計數器為x時(shí)發(fā)生匹配2) 設置 OSSR = 0xf ,表示 清除所有已經(jīng)發(fā)生的匹配,寫(xiě)“1”清除3) 設置 OIER = OIER_EO ,表示 使能 OSMR0 來(lái)產(chǎn)生匹配4) 設置 OSCR = 0 ,表示 初始化計數器的開(kāi)始值 為 0系統時(shí)鐘中斷復位1) 清除 OSSR 中的相應位,即向發(fā)生匹配的OSMR的那個(gè)對應位寫(xiě)“1”
2) 設置 OSCR = 0 ,表示 繼續初始化計數器的值為 0中斷控制器相關(guān)的寄存器● ICPR: 中斷標示寄存器,表示了當前系統正處于激活狀態(tài)的中斷源。
● ICMR: 中斷屏蔽寄存器,用來(lái)屏蔽相應位的中斷。
● ICLR: 中斷級別設置寄存器,設定報告中斷的級別是 IRQ 或者是 FIQ .● ICIP: IRQ 級別的中斷源寄存器,用來(lái)標識 IRQ 中斷發(fā)生的源設備。
● ICFP: FIQ 級別的中斷源寄存器,用來(lái)標識 FIQ 中斷發(fā)生的源設備。
中斷控制器初始化流程1) 設置 ICMR 屏蔽位為不屏蔽時(shí)鐘中斷 OSMR0 (相應位寫(xiě)“1”)
2) 設置 ICLR 為都報告為 IRQ 級別(所有位寫(xiě)“0”)
移植工作總結難點(diǎn)分析移植mC/OS-II 到 StrongARM 的芯片上,基本上和移植到 ARM7 的芯片例如S3C4510,AT91x等工作類(lèi)似,因為所有的ARM處理器都共享arm通用的基礎體系結構,這使得移植工作變得相對簡(jiǎn)單,其中絕大部分工作都集中在 os_cpu_a.S 文件的移植,這個(gè)文件的實(shí)現集中體現了所要移植到處理器的體系結構和mC/OS-II 的移植原理;在這個(gè)文件里,最困難的工作主要是在 OSIntCtxSw 和 OSTickISR 這兩個(gè)函數的實(shí)現上。因為它們的實(shí)現是和移植者的移植思路以及相關(guān)硬件定時(shí)器、中斷寄存器的設置有關(guān)。在實(shí)際的移植工作中,這兩個(gè)地方也是比較容易出錯的地方。
OSIntCtxSw 最重要的作用就是它完成了在中斷ISR中直接進(jìn)行任務(wù)切換,從而提高了實(shí)時(shí)響應的速度。它發(fā)生的時(shí)機是在 ISR 執行到 OSIntExit 時(shí),如果發(fā)現有高優(yōu)先級的任務(wù)因為等待的 time tick 到來(lái)獲得了執行的條件,這樣就可以馬上被調度執行,而不用返回被中斷的那個(gè)任務(wù)之后再進(jìn)行任務(wù)切換,因為那樣的話(huà)就不夠實(shí)時(shí)了。
實(shí)現 OSIntCtxSw 的方法大致也有兩種情況:一種是通過(guò)調整 sp 堆棧指針的方法,根據所用的編譯器對于函數嵌套的處理,通過(guò)精確計算出所需要調整的 sp 位置來(lái)使得進(jìn)入中斷時(shí)所作的保存現場(chǎng)的工作可以被重用。這種方法的好處是直接在函數嵌套內部發(fā)生任務(wù)切換,使得高優(yōu)先級的任務(wù)能夠最快的被調度執行。但是這個(gè)辦法需要和具體的編譯器以及編譯參數的設置相關(guān),需要較多技巧。
另一種是設置需要切換標志位的方法,在 OSIntCtxSw 里面不發(fā)生切換,而是設置一個(gè)需要切換的標志,等函數嵌套從進(jìn)入OSIntExit => OS_ENTER_CRITICAL() => OSIntCtxSw() => OS_EXIT_CRITICAL() => OSIntExit退出后,再根據標志位來(lái)判斷是否需要進(jìn)行中斷級的任務(wù)切換。這種方法的好處是不需要考慮編譯器的因素,也不用做計算,但是從實(shí)時(shí)響應上不是最快,不過(guò)這種方法實(shí)現起來(lái)比較簡(jiǎn)單。
在中斷態(tài)下進(jìn)行任務(wù)切換,需要特別說(shuō)明的一個(gè)問(wèn)題是如何獲得被中斷任務(wù)的 lr_svc .因為進(jìn)入中斷態(tài)后,lr 變成了lr_irq ,原來(lái)任務(wù)的 lr_svc 無(wú)法在中斷態(tài)下獲得,這樣要得到lr_svc,就必須在中斷 ISR 里面進(jìn)行一次cpu mode強制轉換,即對CPSR賦值為0x000000d3,只有返回到svc態(tài)之后才能得到 原來(lái)任務(wù)的lr,這個(gè)對于任務(wù)切換很重要。還有一個(gè)需要留意的問(wèn)題是在強制CPSR變成svc態(tài)之后,SPSR 也會(huì )相應地變成 SPSR_irq,這樣就需要在強制轉變之前保存 SPSR ,也就是被中斷任務(wù)中斷前的 CPSR .移植中使用的編程技巧ADS 編譯器在編譯 C 語(yǔ)言的程序時(shí),如果程序中使用了 main 函數,則編譯器將自動(dòng)添加如下代碼,完成初始化堆棧和C庫等工作,工作流程如下:1> 將執行文件中的 RO 段和 RW 段從 load address 復制到 execution address 2> 初始化 ZI 區域,用 0 來(lái)初始化變量3> 跳轉到 __rt_entry 執行如下 4 個(gè)調用3.1> 調用 __rt_statckheap_init ,建立程序的堆和棧3.2> 調用 __rt_lib_init ,初始化程序用到的 C 庫,并為 main 傳遞參數3.3> 調用 main ,即用戶(hù)程序的入口3.4> 調用 exit因為系統復位后,在啟動(dòng)代碼中已經(jīng)設置了系統堆棧,同時(shí)也不需要使用C庫,因此可以從 __rt_entry 處直接跳轉到mC/OS-II 的代碼中,即直接執行 main 函數,可以用新的 __rt_entry 來(lái)作為鏈接的目標入口。
IMPORT main EXPORT __rt_entry __rt_entry b main這樣在啟動(dòng)代碼的最后,加入一條跳轉語(yǔ)句:bl __main __main 入口是用戶(hù)程序執行的真正入口,我們利用 armCC 編譯 C 里面的 main 入口以求得到 1> 和 2> 的代碼,使得可以支持全局變量。否則的話(huà),必須自己來(lái)實(shí)現全局變量的初始化或者把這些初始化操作放到函數內部來(lái)實(shí)現。
另外一個(gè)非常有用的編程技巧是通過(guò)串口實(shí)現自己的 printf 輸出。 如果使用armCC編譯器的 semihosting 的話(huà),會(huì )把 printf 通過(guò) target 的 swi 0x123456 輸出。如果已經(jīng)實(shí)現的 serial_putchar 之類(lèi)的函數,那么可以用它來(lái)實(shí)現 fputc 接口,也就是低級的輸出函數,這樣就可以使用 printf 來(lái)輸出了,詳細的做法在 ADS 安裝目錄下面的文檔里可以找到,這里就不再贅述
評論