實(shí)時(shí)多任務(wù)操作系統(RTOS)能有效提高嵌入式平臺的資源利用效率,是嵌入式應用的必然趨勢。本文闡述基于MSP430F149的RTOS——M430/OS。它由匯編寫(xiě)成、短小精干、占用系統資源少、運行穩定可靠,目前已在思達高科配網(wǎng)技術(shù)公司產(chǎn)品上得到應用。 1 在MSP430上使用RTOS的意義 一般的觀(guān)點(diǎn)認為,MSP430上使用RTOS是沒(méi)有意義的這是可以理解的。因為MSP430的硬件資源有限(以MSP430F149為例,只有2KBRAM),任何商業(yè)操作系統都不可能移植到MSP430上。目前在MSP430上得到應用的RTOS,只有μC/OS-II,但使用μC/OS-II必須有昂貴的C編譯器,這嚴重地限制了其在MSP430上的使用。 正是基于以上情況,筆者在應用MSP430過(guò)程中,編寫(xiě)了一個(gè)基于MSP430F149的RTOS,暫定名為M430/OS。它占用RAM量少、代碼短小,稍加改動(dòng)就可適用于大多數其它MSP430單片機。 在MSP430單片機系統上使用M430/OS,對系統有以下意義: 、 實(shí)現軟件設計的模塊化?蓪⒉煌墓δ苣K編制成相應的任務(wù),由操作系統按級別調用,不必為先執行哪個(gè)功能、后執行哪個(gè)功能而費神。 、 能更合理、有效地利用CPU有限的資源。按任務(wù)的重要程度安排任務(wù)的級別,能夠保證最重要的任務(wù)得以最及時(shí)執行。 、 大大降低系統故障率。低優(yōu)先級的任務(wù)發(fā)生阻塞時(shí),高優(yōu)先級任務(wù)的執行不受影響。 2 M430/OS在MSP430F149上的實(shí)現 2.1 M430/OS功能特點(diǎn) M430/OS有以下特點(diǎn): 、 采用占先式內核,即高優(yōu)先級的任務(wù)可以從低優(yōu)先級任務(wù)“搶”回CPU控制權; 、 每個(gè)任務(wù)都單獨開(kāi)辟一個(gè)任務(wù)棧; 、 每個(gè)任務(wù)占十幾到幾百字節的任務(wù)堆棧,任務(wù)棧的大小可以根據任務(wù)中現場(chǎng)數據、局部變量和嵌套調用的情況估算; 、 每個(gè)任務(wù)各分配一個(gè)優(yōu)先級,不支持兩個(gè)任務(wù)有相同的優(yōu)先級; 、 不支持信號量、郵箱功能; 、 任務(wù)狀態(tài)只有三種:運行(executing)、就緒(ready)、掛起(suspended); 、 系統占用RAM量=((任務(wù)個(gè)數+1)×4)+6字節,不包括任務(wù)堆棧; 、 代碼量少,目前版本的代碼共有86行匯編代碼,256字節目標代碼; 、 理論上最多支持126個(gè)任務(wù); 、 任務(wù)鎖定功能:在一段低優(yōu)先級的代碼中,不想讓操作系統把CPU權切換到別的任務(wù),這時(shí)可以把這代碼鎖定,在運行這段代碼時(shí),就不會(huì )引起任務(wù)切換; 任務(wù)喚醒功能:在一個(gè)任務(wù)中產(chǎn)生一個(gè)的事件來(lái)觸發(fā)其它任務(wù)運行(如果被觸發(fā)的任務(wù)優(yōu)先級高的話(huà),就會(huì )馬上運行)。 2.2 系統函數介紹 、 OS_Init:多任務(wù)初始化,進(jìn)行任務(wù)棧(任務(wù)棧的結構見(jiàn)圖1)、任務(wù)延時(shí)計數、任務(wù)狀態(tài)的初始化。初始化完成后,系統直接切換到最高優(yōu)先級的任務(wù),多任務(wù)系統啟動(dòng)。 、 OS_Time_Dly:把當前任務(wù)掛起一段指定時(shí)間讓其它任務(wù)運行。 、 OS_Sched:任務(wù)調度,它先把每個(gè)任務(wù)的延時(shí)數減1,然后再找出最高優(yōu)先級的就緒任務(wù),并切換到這個(gè)就緒任務(wù)。如果無(wú)就緒任務(wù),就切換到空閑任務(wù)。 、 OS_Free_Task:空閑任務(wù),是一個(gè)很重要的系統任務(wù),當所有任務(wù)都掛起時(shí),運行此任務(wù)。它主要是對一個(gè)計數器Free_Count一直進(jìn)行累加,用戶(hù)可以根據這個(gè)計數器計算出CPU的利用率。 、 OS_Task_Lock:鎖定任務(wù)調度,禁止任務(wù)調度。主要用來(lái)鎖定在低優(yōu)先級中的一些可重入的代碼或一些重要代碼。 、 OS_Task_Unlock:解鎖任務(wù)調度,和上面的子程序功能相反。 、 OS_Task_Wakeup:?jiǎn)拘阎付▋?yōu)先級的任務(wù),并產(chǎn)生一次任務(wù)調度,如果被喚醒任務(wù)的優(yōu)先級比當前運行的任務(wù)的優(yōu)先級高,任務(wù)就會(huì )切換到被喚醒的任務(wù)中,否則等待下一個(gè)調度時(shí)機。 2.3 主要功能的實(shí)現 (1) 任務(wù)初始化 系統加電運行后,首先對硬件資源進(jìn)行初始化,接著(zhù)就要對多任務(wù)進(jìn)行初始化了。主要是初始化每個(gè)任務(wù)的任務(wù)棧、每個(gè)任務(wù)的時(shí)鐘滴答數和堆棧指針位置。我們把每個(gè)任務(wù)棧都初始化成圖1形式。 圖1 初始化好的任務(wù)棧結構 任務(wù)棧的初始化如下程序(r11是用來(lái)初始任務(wù)堆棧的一個(gè)指針,r10是一個(gè)循環(huán)計數器): mov.w #(棧底 + 2) , r11 clr.w Task_Tick(r10) ;清0時(shí)鐘滴答數 mov.w #任務(wù)首地址 , 0(r11) ;把任務(wù)地址壓入堆棧 mov.w SR , -2(r11) ;把標志寄存器放入任務(wù)棧 mov.w r11 , Task_SP(r10) sub.w #現場(chǎng)所占的字節數 , Task_SP(r10) ;SP位置放入堆棧 初始化完任務(wù)棧之后,就把堆棧指針指向最高任務(wù)優(yōu)先級任務(wù)棧的任務(wù)首地址處,再執行ret返回。這樣,多任務(wù)就啟動(dòng)開(kāi)了,程序如下: mov.w #09feh , sp ;最高優(yōu)先級的任務(wù)棧任務(wù)首地址位置 ret ;返回到最高優(yōu)先級的任務(wù) 任務(wù)初始化的流程如圖2所示。 圖2 任務(wù)初始化流程 (2) 時(shí)鐘節拍 時(shí)鐘節拍由MSP430F149的TimerA產(chǎn)生。TimerA工作于上升模式,CCR0中是TimerA計數最大值。TimerA初始化代碼如下: bis.w #(TASSEL1+TACLR+MC_1),&TACTL mov.w 2(sp),&CCR0 ;計數最大值,此值決定時(shí)鐘節拍 bis.w #CCIE,&CCTL0 (3) 任務(wù)調度 應用程序調用OS_init進(jìn)行初始化后,直接切換到最高優(yōu)先級的任務(wù)。 每個(gè)任務(wù)在運行一個(gè)循環(huán)后執行OS_ Time_Dly掛起。這是通過(guò)把該任務(wù)的延時(shí)數填到該任務(wù)的Task_ Tick中,然后再執行任務(wù)調度程序實(shí)現。 任務(wù)調度就是在定時(shí)中斷時(shí)對所有任務(wù)的Task_ Tick減1,然后再按優(yōu)先級高低的順序查找Task_Tick減到0的任務(wù),并直接跳到任務(wù)切換程序。 下面是任務(wù)切換程序(r10的內容是就緒任務(wù)的標志,由調度程序找出): pushALL ;把當前任務(wù)現場(chǎng)入棧 mov.b Now_Task,r11 ;當前任務(wù)標志放r11 mov.w sp,Task_SP(r11) ;保存當前任務(wù)堆棧指針 mov.b r10,Now_Task ;就緒任務(wù)標志變?yōu)楫斍叭蝿?wù)標志 mov.w Task_SP(r10),sp ;就緒任務(wù)的任務(wù)棧指針?lè )湃隨P;此時(shí)再進(jìn)行堆棧操作就是對就緒任務(wù)的任務(wù)棧操作了。 popALL ;把就緒任務(wù)的現場(chǎng)出棧 reti ;中斷返回,返回到就緒任務(wù) 任務(wù)調度的調度時(shí)機有兩種:一種是在任務(wù)掛起時(shí),一種是定時(shí)中斷。任務(wù)掛起時(shí)的任務(wù)調度一定會(huì )引起任務(wù)切換,定時(shí)中斷就不一定引起任務(wù)切換了。因為,如果就緒任務(wù)是當前正在運行的任務(wù)時(shí)不會(huì )引起切換。正是如此,任務(wù)調度是RTOS中執行得最頻繁的一個(gè)功能,也是最重要的一個(gè)功能,所以必須盡量縮減其代碼量,盡量用可靠的調度算法來(lái)減少任務(wù)調度所占的時(shí)間。這個(gè)子程序的流程如圖3所示。 圖3 任務(wù)調度流程 (4) 任務(wù)鎖和其它功能的實(shí)現 任務(wù)的加鎖與解鎖,是為了使一些在低優(yōu)先級任務(wù)的不可重入代碼,或對實(shí)時(shí)性要求較高的I/O操作在執行中不產(chǎn)生任務(wù)切換。這項功能是通過(guò)設置一個(gè)標志位實(shí)現的。當調度程序檢查到任務(wù)被鎖定時(shí),就算有就緒任務(wù)也必須等開(kāi)鎖之后才能切換。 如果系統突然產(chǎn)生一個(gè)事件要某個(gè)掛起的任務(wù)來(lái)處理,可以在事件產(chǎn)生的程序中調用任務(wù)喚醒。它的原理是把Task_Tick清0,然后執行一次任務(wù)調度。如果這個(gè)任務(wù)優(yōu)先級較高,就直接切換到這個(gè)任務(wù)里執行。 3 總結 M430/OS已在筆者開(kāi)發(fā)的基于MSP430F149的系統上應用,運行穩定可靠。該操作系統稍加改動(dòng),就可應用于其它MSP430單片機。當然,它的功能還是很有限的,也可能還存在一些尚未暴露的問(wèn)題;但無(wú)論如何,它向我們證明,在MSP430單片機系統中使用RTOS是完全可能的。M430/OS源程序見(jiàn)本刊網(wǎng)站(http://www.dpj.com.cn)。 希望有興趣的同志與我們交流。劉兵:public_rtos@163.com 張慶強:starzqq@163.com |