STM32的Systick系統(tǒng)滴答定時器
Systick :系統(tǒng)心跳定時器,提供系統(tǒng)節(jié)拍
裸機(jī)程序中可作為獨立的延時定時器
用途:
1.產(chǎn)生操作系統(tǒng)的時鐘節(jié)拍
2.便于不同處理器之間程序移植
SysTick定時器被捆綁在NVIC中,異常號15
3.作為一個鬧鈴測量時間用于測量時間,
但當(dāng)處理器在調(diào)試期間被喊停(halt)時,則SysTick定時器亦將暫停運(yùn)作
它有四個寄存器
STK_CSR, 0xE000E010 -- 控制寄存器
STK_LOAD, 0xE000E014 -- 重載寄存器
STK_VAL, 0xE000E018 -- 當(dāng)前值寄存器
STK_CALRB, 0xE000E01C -- 校準(zhǔn)值寄存器
stm32的時鐘源
選擇外部時鐘源時,則Systick時鐘為HCLK /8
選擇內(nèi)核時鐘源時,則Systick時鐘為HCLK
延時編程原理
systick定時器是24位的遞減計數(shù)器,設(shè)定初值并使能它后,它會每個系統(tǒng)時鐘周期計數(shù)器減1,
計數(shù)到0 時,將從RELOAD 寄存器中自動重裝載定時初值。只要不把它在SysTick控制及狀態(tài)寄存器中的使能位清除,就永不停息.
延時編程步驟
1.計算出產(chǎn)生1us 需要多少個時鐘周期 fac_us;
2.計算出RELOAD寄存器的值
也就是產(chǎn)生相應(yīng)延時所需要的時鐘周期數(shù)
RELOAD=fac_us * nus
3.開啟計數(shù)
4.循環(huán)檢測計數(shù)到0的標(biāo)志位;
5.清空計數(shù)器,關(guān)閉定時器
=======================================
SysTick異常配置步驟
1對CTRL//LOAD/VAL三個寄存器進(jìn)行了配置,
2初始化SysTick使用的時鐘,
3清除系統(tǒng)當(dāng)前值,裝入重裝值,
4使能SysTick,使SysTick能響應(yīng)中斷
=======================
當(dāng)SysTick定時器計到0時,將把COUNTFLAG位置位;而下述方法可以對其清零:
1.讀取SysTick 控制及狀態(tài)寄存器(STCSR)
2.往SysTick 當(dāng)前值寄存器(STCVR)中寫任何數(shù)據(jù)
只有當(dāng)VAL 值為0 時,計數(shù)器自動重載RELOAD
======================
庫函數(shù)
使用ST的函數(shù)庫使用systick的方法,嚴(yán)格按照以下順序:
1、調(diào)用SysTick_CounterCmd() -- 失能SysTick計數(shù)器
2、調(diào)用SysTick_ITConfig () -- 失能SysTick中斷
3、調(diào)用SysTick_CLKSourceConfig() -- 設(shè)置SysTick時鐘源。
4、調(diào)用SysTick_SetReload() -- 設(shè)置SysTick重裝載值。
5、調(diào)用SysTick_ITConfig () -- 使能SysTick中斷
6、調(diào)用SysTick_CounterCmd() -- 開啟SysTick計數(shù)器
Systick中斷服務(wù)函數(shù)
void SysTick_Handler(void);
==========================
寄存器版代碼注解
使用外部8M時鐘,鎖相環(huán)里出來的頻率是72M,AHB預(yù)分頻后是72M,
systick固定HCLK時鐘的1/8,即9M,那么延時1us是9個時鐘
C代碼
void delay_init(u8 SYSCLK) //系統(tǒng)時鐘是72MHz,SYSCLK=72
{
SysTick->CTRL &= 0xfffffffb ; //bit2清0,也就是配置選擇外部時鐘
fac_us=SYSCLK/8; //硬件8分頻,fac_us得出的值是要給下面的時鐘函數(shù)用的
fac_ms =(u16)fac_us*1000;
}
void delay_us(u32 nus) //nus假如為10us
{
u32 temp;
SysTick->LOAD = nus*fac_us; //延時10us的話就是 10*9=90,裝到load寄存器中
SysTick->VAL=0x00;//計數(shù)器清0,因為currrent字段被手動清零時,load將自動重裝到VAL中
SysTick->CTRL = 0x01;//配置使異常生效,也就是計數(shù)器倒數(shù)到0時將發(fā)出異常通知
do
{
temp = SysTick->CTRL; //時間到了之后,該位將被硬件置1,但被查詢后自動清0
}
while(temp & 0x01 && !(tmep &(1<<16))); //查詢
SysTick->CTRL = 0x00; //關(guān)閉計數(shù)器
SysTick->VAL = 0x00; //清空val
}
//這個while循環(huán),判斷如果Systick還在Enable的狀態(tài),并且計數(shù)器還沒數(shù)到0,
就不停的循環(huán)把當(dāng)前的SysTick->CTRL寄存器值寫入變量temp,繼續(xù)下一次判斷。
當(dāng)Systick被Disable或者計數(shù)器數(shù)到0了,就停止循環(huán)
還有一個注意點:
LOAD寄存器是24位的 最大值0xffffff
那么延時最大值計算公式為
nms<=0xffffff*8*1000/SYSCLK (SYSCLK單位Hz)
則nms的最大值為1864.135ms ,即1864毫秒
? 首先我們要明白什么是SysTick定時器?
Sys 系統(tǒng) ,tick 滴答聲 ,系統(tǒng)滴答滴答很形象地表示了它是一個系統(tǒng)節(jié)拍器。SysTick 是一個24 位的倒計數(shù)定時器,當(dāng)計到0 時,將從RELOAD 寄存器中自動重裝載定時初值。只要不把它在SysTick 控制及狀態(tài)寄存器中的使能位清除,就永不停息。
? 為什么要設(shè)置SysTick定時器?
(1)產(chǎn)生操作系統(tǒng)的時鐘節(jié)拍
SysTick定時器被捆綁在NVIC中,用于產(chǎn)生SYSTICK異常(異常號:15)。在以前,大多操作系統(tǒng)需要一個硬件定時器來產(chǎn)生操作系統(tǒng)需要的滴答中斷,作為整個系統(tǒng)的時基。因此,需要一個定時器來產(chǎn)生周期性的中斷,而且最好還讓用戶程序不能隨意訪問它的寄存器,以維持操作系統(tǒng)“心跳”的節(jié)律。SysTick的最大使命,就是定期地產(chǎn)生異常請求,作為系統(tǒng)的時基。OS都需要這種“滴答”來推動任務(wù)和時間的管理。
(2)便于不同處理器之間程序移植。
Cortex‐M3處理器內(nèi)部包含了一個簡單的定時器。因為所有的CM3芯片都帶有這個定時器,軟件在不同 CM3器件間的移植工作得以化簡。該定時器的時鐘源可以是內(nèi)部時鐘(FCLK,CM3上的自由運(yùn)行時鐘),或者是外部時鐘( CM3處理器上的STCLK信號)。不過,STCLK的具體來源則由芯片設(shè)計者決定,因此不同產(chǎn)品之間的時鐘頻率可能會大不相同,你需要檢視芯片的器件手冊來決定選擇什么作為時鐘源。SysTick定時器能產(chǎn)生中斷,CM3為它專門開出一個異常類型,并且在向量表中有它的一席之地。它使操作系統(tǒng)和其它系統(tǒng)軟件在CM3器件間的移植變得簡單多了,因為在所有CM3產(chǎn)品間對其處理都是相同的。
(3)作為一個鬧鈴測量時間。
SysTick定時器還可以用作鬧鐘,作為啟動一個特定任務(wù)的時間依據(jù)。它作為一個鬧鈴,用于測量時間。要注意的是,當(dāng)處理器在調(diào)試期間被喊停(halt)時,則SysTick定時器亦將暫停運(yùn)作。
? 再來看看SysTick的用法
(1)我們對一個系統(tǒng)編程,老說編程編程什么的,到底我們在編什么程?當(dāng)然這個問題要探討起來可能有點遠(yuǎn)了。我來說說對SysTick的編程,對單片機(jī)的編程不過就是對單片機(jī)里面的寄存器進(jìn)行控制,使整個軟硬件系統(tǒng)處于一種在你的掌控之下的狀態(tài)。這就是了嘛,現(xiàn)在我是頭,我對我的手下下達(dá)一些指令,讓它們?nèi)プ鲆恍┦虑?。所以我們想搞清楚怎樣控制SysTick我們還得看我們能對它的哪些部分可以控制。那些部分就是寄存器。
SysTick有4個寄存器 :
寄存器 描述
CTRL SysTick 控制和狀態(tài)寄存器
LOAD SysTick 重裝載值寄存器
VAL SysTick 當(dāng)前值寄存器
CALIB SysTick 校準(zhǔn)值寄存器
對應(yīng)地在固件函數(shù)庫中定義了這個東西
typedef struct
{
vu32 CTRL;
vu32 LOAD;
vu32 VAL;
vuc32 CALIB;
} SysTick_TypeDef;
在這背后,已經(jīng)對定義的寄存器進(jìn)行了一個地址映射。當(dāng)我們操控我們定義的寄存器時實際上已是通過那種映射關(guān)系操控了芯片內(nèi)部的值。其實在STM32中對寄存器的操作都是通過這種方式進(jìn)行的。
具體的映射過程如下,我們可以看一下:
#define SCS_BASE ((u32)0xE000E000)
#define SysTick_BASE (SCS_BASE + 0x0010)
#ifndef DEBUG
...
#ifdef _SysTick
#define SysTick ((SysTick_TypeDef *) SysTick_BASE)
#endif
...
#else
...
#ifdef _SysTick
EXT SysTick_TypeDef *SysTick;
#endif
...
#endif
#ifdef _SysTick
SysTick = (SysTick_TypeDef *) SysTick_BASE;
#endif
為了訪問SysTick寄存器,, _SysTick必須在文件“stm32f10x_conf.h”中定義如下:
#define _SysTick
映射過程就不作討論了??傔@這樣映射的結(jié)果是我們能直接使用SysTick。那就來看一下有關(guān)寄存器的設(shè)置。
(2)SysTick里的寄存器我也簡單地把它理解為是一個32位數(shù)。
這里有一張圖:
在最新的STM32固件庫中的core_cm3.c中提供了這樣一個函數(shù)來供我們配置SysTick,當(dāng)我們須要用到SysTick時調(diào)用它就可以了:
static __INLINE uint32_t SysTick_Config(uint32_t ticks)//ticks 是要重裝載的值
{
if (ticks > SysTick_LOAD_RELOAD_Msk) return (1);
SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;
NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);
SysTick->VAL = 0;
SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk |
SysTick_CTRL_TICKINT_Msk |
SysTick_CTRL_ENABLE_Msk;
return (0);
}
總結(jié):在配置過程對CTRL//LOAD/VAL三個寄存器進(jìn)行了配置,初始化了SysTick使用的時鐘,清除系統(tǒng)當(dāng)前值,裝入重裝值,使能SysTick,使SysTick能響應(yīng)中斷,說了半天其實就這一句話。在主程序中調(diào)用SysTick_Configuration( uint32_t ticks ),輸入重裝值就配置完成了。
(3)SysTick 的中斷處理函數(shù)在stm32f10x_it.c
函數(shù)原型為void SysTick_Handler(void)
{
// user code
}
用戶只要把須要處理的程序填入這里就完成啦。
? 例子:
正如上面敘述,SysTick的使用為:
(1)配置SysTick
(2)寫中斷函數(shù)
我們產(chǎn)生1ms的廷時:
在我們自己編寫的main.c中有:
//前面的省略 ……
Volatile unsigned int TimingDelay ; //定義一個全局變量,用于計數(shù)計時值
//中間部分省略……
void Delay_Ms( uint32_t nTime