一種嵌入式時(shí)鐘管理器的設計與實(shí)現
目前,在嵌入式產(chǎn)品的研發(fā)中,低檔微處理器軟件多采用裸機開(kāi)發(fā)模式實(shí)現。在這種開(kāi)發(fā)模式中,常有如下需求:
本文引用地址:http://dyxdggzs.com/article/149690.htm(1)在經(jīng)歷特定的時(shí)間段后,執行特定操作;
(2)根據給定周期執行特定操作。
傳統的作法是利用前后臺方式:設定硬件定時(shí)器,使其在后臺以特定周期對各相關(guān)操作的標志變量作計數操作;前臺則不斷對各標志變量巡回查詢(xún),若發(fā)現標志變量達到預定值,則執行特定操作??梢?jiàn),上述需求需直接操作硬件定時(shí)器實(shí)現,其過(guò)程繁瑣,且需要用戶(hù)對相關(guān)硬件有深入了解。因此,本文設計、實(shí)現了一種使用方便的低端系統時(shí)鐘管理器。
本時(shí)鐘管理器適用于可提供至少一個(gè)硬件定時(shí)器的處理器。其為用戶(hù)提供了有益、友好的裁剪途徑,以滿(mǎn)足不同目標系統的實(shí)際需要。通過(guò)裁剪,該時(shí)鐘管理器的目標代碼最小可至100B以下,最大也不超過(guò)1KB。
時(shí)鐘管理器在實(shí)現中,將與硬件密切相關(guān)的部分組成一獨立模塊(文件)。針對不同的目標系統處理器,更換該模塊即可。為使表述不過(guò)抽象,本文以8051系列單片機為目標系統處理器、C51為工具語(yǔ)言闡述該嵌入式時(shí)鐘管理器的設計與實(shí)現。
1 設計
該時(shí)鐘管理器模塊(文件)結構如圖1所示。
(1)cONfigClk.h定義了有關(guān)系統裁剪、配置的可調參數,通過(guò)對configClk.h中相關(guān)宏參數的配置,即可實(shí)現對該時(shí)鐘管理器系統的配置和裁剪。
(2)clk_impl.*功能模塊用來(lái)封裝目標系統的一個(gè)硬件定時(shí)器,以屏蔽不同處理器間的硬件差異,起到HAL(HardwareAbSTractLayer)作用。系統時(shí)鐘在此構建。
(3)clk.*模塊在clk_impl.*提供的HAL基礎上進(jìn)一步封裝,通過(guò)一個(gè)鉤子(Hook)函數,為系統提供時(shí)鐘脈沖,且脈沖寬度可調(配置configClk.h中的相關(guān)宏參即可)。
(4)WdLib.*模塊為用戶(hù)應用提供多個(gè)軟件定時(shí)器。
2 實(shí)現
2.1 硬件定時(shí)器的底層封裝
硬件定時(shí)器底層封裝在圖1所示的clk_impl.*中實(shí)現。其中定義了一個(gè)初始化接口函數和一個(gè)定時(shí)器中斷的ISR(Interrupt Service RouTIne)。令選用的硬件時(shí)鐘為定時(shí)器0(可在configClk.h中配置)。
(1)初始化接口函數void_clkInit(void){ }
用戶(hù)通過(guò)調用該接口函數,可周期性地執行相應的ISR—clkTick_ISR,從而形成邏輯上的系統時(shí)鐘。另外,本接口函數不為用戶(hù)直接訪(fǎng)問(wèn),而在上層模塊clk.*中被調用。
(2)定時(shí)器0的ISR—clkTick_ISR
void clkTick_ISR (void) interrupt 1 using REG_GRP_FOR_
SYS_CLK{ }
其中:REG_GRP_FOR_SYS_CLK為定義于configClk.h中的可調參數,用來(lái)設定本ISR的工作寄存器組。
2.2 時(shí)鐘脈沖的提供
時(shí)鐘脈沖在圖1所示的clk.*中實(shí)現。
本文提供三個(gè)用戶(hù)接口函數和一個(gè)用戶(hù)可修改、但不可調用的鉤子函數(clkTick_ISR_hook僅能在clkTick_ISR中被調用)。其用戶(hù)接口聲明如下:
extern void constructClk(void);
extern void destructClk(void);
extern UINT8 getClkRate(void);
其中:constructClk用以構建系統時(shí)鐘,要使用本文所述的時(shí)鐘管理器,需首先通過(guò)調用_clkInit(定義于clk_impl.*模塊)實(shí)現對本函數的調用;destructClk用以解析業(yè)已構建的系統時(shí)鐘;getClkRate用以獲取系統當前的時(shí)鐘節拍率(即定義于configClk.h中的宏SYS_CLK_RATE的當前值)。
clkTick_ISR_hook由系統聲明,用戶(hù)可修改其定義,其最終僅為系統作周期性調用。用戶(hù)可將自己需進(jìn)行的周期性操作放于其中,后面敘述的軟件定時(shí)器的“守護”例程(wdDaemon)正是置于此處而被周期調用。由于置于其中的操作將在中斷執行,所以這些操作應盡可能簡(jiǎn)短、省時(shí)。
2.3 軟件定時(shí)器的提供
本功能在圖1所示的wdLib.*中實(shí)現。
其為用戶(hù)提供了可快速、便捷地實(shí)現用戶(hù)定時(shí)需求的接口函數和一個(gè)被周期性調用的定時(shí)器守護例程wdDaemon。
extern void constructWDOG(void);//為使用定時(shí)器系統作初始化操作
extern void destructWDOG(void)//置定時(shí)器系統為初始態(tài)
extern WDOG_ID wdCreate(void);//建立一個(gè)定時(shí)器,并返回其ID
extern STATUS wdCancel(WDOG_ID wdId);//終止指定定時(shí)器并復位
extern STATUS wdDelete(WDOG_ID wdId);//刪除指定定時(shí)器
extern STATUS wdStart(WDOG_ID wdId,UINT16 ticks,VOIDFUNCPTR wdr);//啟動(dòng)指定定時(shí)器,它會(huì )在指定時(shí)間后觸發(fā)給定操作
其中:WDOG_ID為定時(shí)器ID類(lèi)型,即UINT8。傳送給wdStart的參數“UINT16 ticks”指明定時(shí)時(shí)間長(cháng)度,單位為系統時(shí)鐘節拍,1節拍=1/SYS_CLK_RATE(s)。因該參數的類(lèi)型定為UINT16,故定時(shí)器的最大定時(shí)長(cháng)度為216×(1/SYS_CLK_RATE),即216/SYS_CLK_RATE(s)。
定時(shí)器的實(shí)現方案有靜態(tài)數組法和delta列表法兩種方法。這兩種方法各有優(yōu)缺點(diǎn):前者邏輯簡(jiǎn)單,ROM用量小,但效率較低(與定時(shí)器數目相關(guān));后者邏輯復雜,ROM用量大,但效率較高(與定時(shí)器數目無(wú)關(guān))。應用中使用哪種方案,可在configClk.h中配置選擇。
2.3.1 靜態(tài)數組法
靜態(tài)數組法的數據結構如下:
struct wdNode {
BOOL flag;//標明本結點(diǎn)是否已被使用
UINT16 ticks;//用以定時(shí)的節拍數
VOIDFUNCPTR rout;//定時(shí)到時(shí)需執行的操作
} data wdList[_MAX_WDOG_NUM_];
其中:_MAX_WDOG_NUM_指出了系統中允許的最大定時(shí)器數,其值決定于應用需求及系統資源量,可在configClk.h中設定。一個(gè)定時(shí)器結點(diǎn)占用5B的RAM空間。具有給定數據結構的靜態(tài)數組是方案實(shí)施的基礎。
另外,該靜態(tài)數組作為軟件定時(shí)器的全局變量而存在,當系統中有多個(gè)定時(shí)器活動(dòng)時(shí),它們都將訪(fǎng)問(wèn)該全局靜態(tài)數組。重要的是:它們的活動(dòng)是異步的,所以,對該靜態(tài)數組(臨界資源)的訪(fǎng)問(wèn)需作臨界保護。對于51系統,應采用開(kāi)關(guān)中斷的方式實(shí)現,且應確保不會(huì )影響關(guān)中斷前的中斷狀態(tài)。
(1)用戶(hù)接口定義
上述用戶(hù)接口皆基于該靜態(tài)數組進(jìn)行,限于篇幅,這里給出關(guān)鍵接口wdStart的定義。
STATUS wdStart(WDOG_ID wdId,UINT16 ticks,
VOIDFUNCPTR wdr) {
if(wdId_MAX_WDOG_NUM_) {
if(wdList[wdId].flag) {//判斷給定定時(shí)器ID有效否
RTX_ENTER_CRITICAL();//進(jìn)入臨界區
wdList[wdId].ticks=ticks;//操作靜態(tài)數組中的特定定時(shí)結點(diǎn)
wdList[wdId].rout=wdr;
RTX_EXIT_CRITICAL();//退出臨界區
return OK;//定時(shí)器啟動(dòng)成功
}
}
return ERROR;//給定定時(shí)器ID無(wú)效
}
調用該接口函數,即可啟動(dòng)已創(chuàng )建(wdCreate)的軟件定時(shí)器。當經(jīng)歷ticks節拍后,給定函數wdr將被執行,以完成用戶(hù)的定時(shí)需求。
(2)定時(shí)器守護例程
定時(shí)器守護例程wdDaemon被置于前述的鉤子函數clkTick_ISR_hook中,以使其周期性執行。由于本例程自身的特點(diǎn),它應作為clkTick_ISR_hook的最后一個(gè)調用函數。本例程是軟件定時(shí)器實(shí)現的核心,而其關(guān)鍵又是對系統棧的調整,為說(shuō)明其實(shí)現流程,給出了如圖2所示的wdDaemon的棧(stack)結構。
由圖2可知:wdDaemon的返回地址沒(méi)有入棧,因其為clkTick_ISR_hook中的最后一個(gè)函數調用,故其返回地址被優(yōu)化掉。wdDaemon將棧頂的8B數據上移2B,然后將定時(shí)器指定函數的地址插入騰出的??臻g(2B)中。如此,該地址將會(huì )被IRET彈入IP中。由于IRET指令的執行而使中斷系統復位以重新響應外部中斷,同時(shí)也使定時(shí)器指定函數在非中斷態(tài)執行,從而不過(guò)分影響系統的響應速度。
2.3.2 delta列表法
delta列表法僅維護有效定時(shí)器的鏈表,且鏈表中的定時(shí)器結點(diǎn)按定時(shí)剩余時(shí)間由小到大排列,使距timeout點(diǎn)最近的定時(shí)器作為鏈表的首結點(diǎn)。鏈表中定時(shí)器結點(diǎn)的順序由其獨特的結點(diǎn)插入算法決定:如有5個(gè)定時(shí)器,其定時(shí)長(cháng)度分別為10、14、21、32和39,當其組成delta列表時(shí),定時(shí)值最小的結點(diǎn)為首結點(diǎn),其定時(shí)存儲值為10,而后依序排列,其定時(shí)存儲值分別為4、7、11、7,即后一個(gè)定時(shí)器的定時(shí)存儲值由自己的實(shí)際定時(shí)值與相鄰的前一個(gè)定時(shí)器的實(shí)際定時(shí)值相減而得??梢?jiàn),除首結點(diǎn)外的所有定時(shí)器的計數操作在其插入delta列表時(shí)就已完成。因而當定時(shí)器守護例程確定timeout的定時(shí)器時(shí),只需對首結點(diǎn)進(jìn)行減1或刪除的操作,而不需遍歷整個(gè)列表,從而使delta列表的操作與定時(shí)器數量無(wú)關(guān)。這使delta列表法在大量定時(shí)器管理中大顯其能。
該法在系統中實(shí)現的數據結構為一靜態(tài)雙向鏈表:
struct wdNode {
BOOL flag;
UINT16 ticks;
VOIDFUNCPTR rout;
UINT8 prior;
UINT8 next;
} idata wdList[_MAX_WDOG_NUM_];
UINT8 headIdx; //索引首結點(diǎn)
有了delta列表法的思路及其實(shí)現的數據結構,在靜態(tài)數組法具體實(shí)現的基礎上,便可得此法的具體實(shí)現。
應用中如果目標系統ROM較小,且系統中啟用的定時(shí)器少,則用靜態(tài)數組法;若目標系統ROM較大,且系統中用到的定時(shí)器較多,則用delta列表法。
3 應用
針對前述的嵌入式系統中的定時(shí)需求,利用定時(shí)器管理系統給出其實(shí)現代碼。
假定“特定操作”為void specFunc(void),“特定時(shí)間段”長(cháng)度為10分鐘。
(1)在經(jīng)歷特定的時(shí)間段后,執行特定操作。
#include ″clk.h″
#include ″wdLib.h″
void main(void ) {
WDOG_ID wdId;
constructClk();constructWDOG();
wdId=wdCreate();
wdStart(wdId,10*ONE_MINUTE,specFunc);
while(1);
}
(2)以給定周期周期性地執行特定操作。
基于前者,只需在void specFunc(void)函數體的最后加入下述代碼即可:
wdStart(wdId,10*ONE_MINUTE,specFunc);
注:該給定周期為10分鐘。
由于本時(shí)鐘管理器只需一個(gè)硬件定時(shí)器的支持,所以其具有廣泛的適用性。使用時(shí),只需進(jìn)行簡(jiǎn)單的配置,即可為裸露的目標系統加以簡(jiǎn)單的軟件抽象層。其友好的用戶(hù)接口有效降低了嵌入式系統的開(kāi)發(fā)難度,提高了目標系統的可靠性。筆者已在實(shí)際項目中多次使用了該時(shí)鐘管理器?;谠摃r(shí)鐘管理器的目標系統運行穩定、可靠,從而充分說(shuō)明該時(shí)鐘管理器設計的實(shí)用性和科學(xué)性。
linux操作系統文章專(zhuān)題:linux操作系統詳解(linux不再難懂)
評論