引言
在嵌入式系統(tǒng)的設(shè)計中,時間特性在很多應(yīng)用中都是一個很重要的參數(shù),很多控制邏輯和協(xié)議的實現(xiàn),需要用到計時、超時和統(tǒng)計功能,時間長短的計時及精度、周期定時和超時的設(shè)定成為某些控制邏輯和協(xié)議的關(guān)鍵條件。筆者在為某車型設(shè)計其車身中央控制器時,輸入條件的判定、輸出邏輯的控制和LIN協(xié)議的實現(xiàn)都包含一些時間特性相關(guān)的功能[1],為此筆者設(shè)計了一個高效的定時器模塊。下面就從設(shè)計原理、結(jié)構(gòu)設(shè)計、定時器管理及應(yīng)用上講述下該定時器模塊的設(shè)計與實現(xiàn)。
設(shè)計原理
在車身控制器(以下以BCM來代稱)的設(shè)計中,其定時功能可以分為兩類,一個是TIMEOUT超時,一個是COUNTER計時,前者超時時間已知,超時則觸發(fā)相關(guān)操作,后者是計時請求,在一種狀態(tài)發(fā)生變化時統(tǒng)計新的狀態(tài)的維持時間。通過對BCM定時功能及特性進(jìn)行分析,發(fā)現(xiàn)其定時應(yīng)用的特點為:
1、定時精度要求不高,但定時數(shù)量比較多;
2、TIMEOUT超時中,有循環(huán)定時,也有單次定時和多次定時。
由于硬件資源有限,不可能為每一個定時應(yīng)用單獨分配一個硬件定時器,所以采用單個硬件定時器模擬多個軟件定時器的方法,來滿足應(yīng)用中的定時需要[2]。首先根據(jù)定時應(yīng)用的特點和分類,設(shè)計軟件定時器的數(shù)據(jù)結(jié)構(gòu),以結(jié)構(gòu)體數(shù)組的形式將這些軟件定時器組織起來,數(shù)組成員便是各個軟件定時器節(jié)點;然后在驅(qū)動程序的設(shè)計中設(shè)計良好的API接口,該API接口足夠清晰,提供統(tǒng)一的調(diào)用接口,可以涵蓋定時器所有功能,這樣便在一個硬件定時器上實現(xiàn)了多個軟件定時器。下面便詳述一下定時器結(jié)構(gòu)和驅(qū)動程序設(shè)計。
定時器結(jié)構(gòu)設(shè)計
由于軟件定時器數(shù)量繁多,根據(jù)定時應(yīng)用的實現(xiàn)方式及特征設(shè)計一個良好的數(shù)據(jù)結(jié)構(gòu)是非常必要的[3]。在其結(jié)構(gòu)體設(shè)計中,其成員變量可以描述所有的“定時特征”并且提供良好的可讀寫接口。其定義如下所示:
typedef struct {
TimerState timer_state;
ulong timeout;
ulong duration;
unsigned cycle:1;
unsigned overow_ag:1;
unsigned cnt_times:8;
TimerId timer_id;
}Timer;
timer_state有STALL和RUNNING兩種取值,表示該定時器是否處于運轉(zhuǎn)狀態(tài);timeout是超時應(yīng)用的超時值,duration表示定時器啟動以來的計時時間長度;cycle表示該定時是周期定時還是單次定時;cnt_times是對多次定時(既非周期也非單次)的統(tǒng)計,當(dāng)定時次數(shù)到達(dá)后,則停止該定時器;overow_ag標(biāo)識是否發(fā)生超時溢出;所有的“軟件定時器節(jié)點”組成Timer數(shù)組,其數(shù)據(jù)成員由timer_id枚舉,其枚舉類型TimerId定義如下所示:
typedef enum{
FEEDWATCHDOG_300MS,
INPUT_DETECT_MS,
LIN_TIMESLOT_MS,
SPEED_PULSE_ACCUMU_200MS,
ALLDR_LOCK_200MS,
ALARM_30S,
IGNITION_OFF_60S,
。..
。..
MAX_TIMER_NUM
}TimerId;
這樣,定義數(shù)組Timer TIMER[MAX_TIMER_NUM],在初始化中,TIMER [i].timer_id=i;之后通過TIMER[timer_id]即可枚舉某軟件定時器節(jié)點。
引言
在嵌入式系統(tǒng)的設(shè)計中,時間特性在很多應(yīng)用中都是一個很重要的參數(shù),很多控制邏輯和協(xié)議的實現(xiàn),需要用到計時、超時和統(tǒng)計功能,時間長短的計時及精度、周期定時和超時的設(shè)定成為某些控制邏輯和協(xié)議的關(guān)鍵條件。筆者在為某車型設(shè)計其車身中央控制器時,輸入條件的判定、輸出邏輯的控制和LIN協(xié)議的實現(xiàn)都包含一些時間特性相關(guān)的功能[1],為此筆者設(shè)計了一個高效的定時器模塊。下面就從設(shè)計原理、結(jié)構(gòu)設(shè)計、定時器管理及應(yīng)用上講述下該定時器模塊的設(shè)計與實現(xiàn)。
設(shè)計原理
在車身控制器(以下以BCM來代稱)的設(shè)計中,其定時功能可以分為兩類,一個是TIMEOUT超時,一個是COUNTER計時,前者超時時間已知,超時則觸發(fā)相關(guān)操作,后者是計時請求,在一種狀態(tài)發(fā)生變化時統(tǒng)計新的狀態(tài)的維持時間。通過對BCM定時功能及特性進(jìn)行分析,發(fā)現(xiàn)其定時應(yīng)用的特點為:
1、定時精度要求不高,但定時數(shù)量比較多;
2、TIMEOUT超時中,有循環(huán)定時,也有單次定時和多次定時。
由于硬件資源有限,不可能為每一個定時應(yīng)用單獨分配一個硬件定時器,所以采用單個硬件定時器模擬多個軟件定時器的方法,來滿足應(yīng)用中的定時需要[2]。首先根據(jù)定時應(yīng)用的特點和分類,設(shè)計軟件定時器的數(shù)據(jù)結(jié)構(gòu),以結(jié)構(gòu)體數(shù)組的形式將這些軟件定時器組織起來,數(shù)組成員便是各個軟件定時器節(jié)點;然后在驅(qū)動程序的設(shè)計中設(shè)計良好的API接口,該API接口足夠清晰,提供統(tǒng)一的調(diào)用接口,可以涵蓋定時器所有功能,這樣便在一個硬件定時器上實現(xiàn)了多個軟件定時器。下面便詳述一下定時器結(jié)構(gòu)和驅(qū)動程序設(shè)計。
定時器結(jié)構(gòu)設(shè)計
由于軟件定時器數(shù)量繁多,根據(jù)定時應(yīng)用的實現(xiàn)方式及特征設(shè)計一個良好的數(shù)據(jù)結(jié)構(gòu)是非常必要的[3]。在其結(jié)構(gòu)體設(shè)計中,其成員變量可以描述所有的“定時特征”并且提供良好的可讀寫接口。其定義如下所示:
typedef struct {
TimerState timer_state;
ulong timeout;
ulong duration;
unsigned cycle:1;
unsigned overow_ag:1;
unsigned cnt_times:8;
TimerId timer_id;
}Timer;
timer_state有STALL和RUNNING兩種取值,表示該定時器是否處于運轉(zhuǎn)狀態(tài);timeout是超時應(yīng)用的超時值,duration表示定時器啟動以來的計時時間長度;cycle表示該定時是周期定時還是單次定時;cnt_times是對多次定時(既非周期也非單次)的統(tǒng)計,當(dāng)定時次數(shù)到達(dá)后,則停止該定時器;overow_ag標(biāo)識是否發(fā)生超時溢出;所有的“軟件定時器節(jié)點”組成Timer數(shù)組,其數(shù)據(jù)成員由timer_id枚舉,其枚舉類型TimerId定義如下所示:
typedef enum{
FEEDWATCHDOG_300MS,
INPUT_DETECT_MS,
LIN_TIMESLOT_MS,
SPEED_PULSE_ACCUMU_200MS,
ALLDR_LOCK_200MS,
ALARM_30S,
IGNITION_OFF_60S,
。..
。..
MAX_TIMER_NUM
}TimerId;
這樣,定義數(shù)組Timer TIMER[MAX_TIMER_NUM],在初始化中,TIMER [i].timer_id=i;之后通過TIMER[timer_id]即可枚舉某軟件定時器節(jié)點。
定時器管理
定時器管理包括啟動、運行、停止、重啟和讀取計時時間等功能[4],相應(yīng)API函數(shù)定義如下:
void TimerStart(TimerId timer_id,ulong timeout,Bool cycle,uchar cnt_times);
void TimerReStart(TimerId timer_id);
void TimerTick(void);
void TimerStop(TimerId timer_id);
void TimerStall(TimerId timer_id);
ulong TimerGet(TimerId timer_id);
啟動定時器函數(shù)如下:
#dene RTIPERIOD 2
void TimerStart(TimerId timer_id,ulong timeout,Bool cycle,uchar cnt_times)
{
TIMER[timer_id].timer_state=RUNNING;
TIMER[timer_id].duration=0;
TIMER[timer_id].timeout=timeout/RTIPERIOD;
TIMER[timer_id].cycle=cycle;
TIMER[timer_id].cnt_times=cnt_times;
TIMER[timer_id].timer_id=timer_id;
}
可見,啟動某個軟件定時器便是設(shè)置由timer_id枚舉的TIMER數(shù)組成員的各個成員變量,下面詳細(xì)介紹軟件定時器的運行。
作為所有軟件定時器的基準(zhǔn)源,硬件定時器設(shè)定為2ms的周期定時,在時鐘中斷服務(wù)程序中全局時鐘嘀嗒Jiffs累加,TimerTicked置1,軟件定時器運行函數(shù)如下:
void TimerTick(void)
{
uint timer_index;
if(0==TimerTicked)
{
return ;
}
for(timer_index=0;timer_index
{
if(RUNNING==TIMER[timer_index].timer_state)
{
TIMER[timer_index].duration++;
if(TIMER[timer_index].duration》=TIMER[timer_index].timeout)
{
TIMER[timer_index].overflow_ag=1;
if(TIMER[timer_index].cycle)
{
TimerReStart(timer_index);
}
else
{
TIMER[timer_index].cnt_times--;
if(0==TIMER[timer_index].cnt_times)
{
TimerStall(timer_index);
}
else
{
TimerReStart(timer_index);
}}}}}
TimerTicked=0;
}
軟件定時器只有在有嘀嗒發(fā)生且自身狀態(tài)為RUNNING的情況下才會運行,其計時時間-duration隨嘀嗒而累加,如果duration達(dá)到超時值,則置overow_ag,然后判斷該定時器是否周期定時器,是則重啟定時-清零duration,否則判斷是否多次定時,定時次數(shù)計數(shù)未滿則重啟定時,計數(shù)滿后停止該定時器。
定時器應(yīng)用
定時器模塊在軟件系統(tǒng)中是一個基本功能單元,它為其他模塊或上層應(yīng)用提供超時、計時服務(wù)[5]。以門鎖的驅(qū)動為例講述下定時器的應(yīng)用。
在門鎖閉鎖時,BCM對門鎖電機的閉鎖驅(qū)動控制需要保持200ms的時間,在邏輯上便是輸出一個200ms的控制脈沖,該部分代碼如下:
。..
DrivePort(ALLDR_LK_OUT,DRIVEON);
TimerStart(ALLDR_LOCK_200MS,200,0,1);
if(1==TM[ALLDR_LOCK_200MS].overow_ag)
{
DrivePort(ALLDR_LK_OUT,DRIVEOFF);
TM[ALLDR_LOCK_200MS].overow_ag=0;
}
。..
可見,在設(shè)計良好的定時器驅(qū)動以后,實現(xiàn)某種定時應(yīng)用非常簡單,其他應(yīng)用示例在此不再贅述。
結(jié)語
本文設(shè)計的定時器模塊,邏輯清晰,使用方便,做為一個穩(wěn)定的底層驅(qū)動,在實際應(yīng)用中得到了很好的應(yīng)用。