一、定時器簡介
1、時鍾來源
2、定時器結構(以基本定時器為例)
二、基本定時器的編程方法
1、基本定時器的寄存器
2、例程
/** * @brief 定時器6的初始化,定時周期0.01s * @param 無 * @retval 無 */ void TIM6_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; /*AHB = 72MHz,RCC_CFGR的PPRE1 = 2,所以APB1 = 36MHz,TIM2CLK = APB1*2 = 72MHz */ RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE); /* 時基初始化 */ TIM_TimeBaseStructure.TIM_Period = 99; //當定時器從0計數到99,即定時周期為100次 TIM_TimeBaseStructure.TIM_Prescaler = 7199; //設置預分頻:10KHz TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure); TIM_ARRPreloadConfig(TIM6, ENABLE); //使能TIM6重載寄存器ARR /* 設置更新請求源只在計數器上溢或下溢時產生中斷 */ TIM_UpdateRequestConfig(TIM6,TIM_UpdateSource_Global); /* 定時器6的上溢或下溢中斷使能 */ TIM_ITConfig(TIM6, TIM_IT_Update, ENABLE); /* 定時器6啟動 */ TIM_Cmd(TIM6, ENABLE); //使能定時器6 TIM_ClearITPendingBit(TIM6,TIM_IT_Update); /* 定時器6的NVIC中斷配置 */ NVIC_TIM6_Configuration(); }
三、疑惑與解答
以下問題皆以基本定時器為例進行闡述
1、何謂更新事件
更新事件就是指這個事件發生后,將會將定時器的寄存器進行更新,以使定時器工作在新的配置下,例如當一個定時周期結束(計數器上溢)或者其他事件。
2、何謂自動重裝載寄存器(auto-reload register)
自動重裝載寄存器決定了定時器的上溢時機,當定時器的計數器中數值達到了自動重裝載寄存器規定的值,計數器就要歸零。也就是說自動重裝載寄存器決定了定時器的周期。假定TIMx_ARR=0x36,而且分頻系數為1,則可以看到下邊的情況。
3、自動重裝載寄存器和預加載寄存器的區別與聯系
當“TIMx_CR1.ARPE = 1”的時候,STM32中有自動重裝載寄存器和預加載寄存器(TIMx_ARR)。
預加載寄存器是自動重裝載寄存器的“影子”,也就是預加載寄存器是自動重裝載寄存器的緩沖器。自動重裝載寄存器的功能在2點已經說明,但是自動重裝載寄存器不是用戶用程序可以直接進行操作的,用戶需要借助於預加載寄存器(緩沖區)才能訪問它。
其目的是為了保證自動重裝載寄存器在合適的時候被修改,不允許其隨便被修改,否則可能導致在過渡的時候發生不期望的結果。
這是什么一個概念呢?
在定時器一個周期結束的時候,產生了一個更新中斷,我們在中斷服務程序中修改預加載寄存器(TIMx_ARR),但是並沒有直接寫入到自動重裝載寄存器。在中斷剛一產生的時候(早於我們的服務程序),原來TIMx_ARR的值被硬件自動裝入自動重裝載寄存器中。所以下一個定時器周期的長度取決於“原來TIMx_ARR的值”,而非我們在中斷服務程序中的修改值。
那么什么時候,我們的修改值才起作用呢?
當下一個定時器周期結束的時候,我們對TIMx_ARR的修改值就被硬件自動寫入到自動重裝載寄存器中,所以我們的修改值在下下個定時器周期才起作用。
而當“TIMx_CR1.ARPE = 0”的時候,STM32中只有自動重裝載寄存器(TIMx_ARR),沒有預加載寄存器。自動重裝載寄存器沒有緩沖區,對TIMx_ARR的修改,也就是直接對自動重裝載寄存器的修改。
這種情形又怎樣看呢?
在定時器一個周期結束的時候,產生了一個更新中斷,我們在中斷服務程序中修改自動重裝載寄存器(TIMx_ARR)。所以下一個定時器周期的定時長度要取決於我們的這個修改值。
總結:
① TIMx_CR1.ARPE = 0,自動重裝載寄存器沒有緩沖區,對TIMx_ARR的修改直接影響下一個周期的定時長度。
② TIMx_CR1.ARPE = 1,自動重裝載寄存器有緩沖區,對TIMx_ARR的修改影響的是下下一個周期的定時長度。
③ TIMx_CR1.ARPE = 1,自動重裝載寄存器有緩沖區預加載寄存器(TIMx_ARR),預加載寄存器更新到自動重裝載寄存器的時機是:當定期器一個定時周期結束產生一個更新事件的時候。
④ TIMx_CR1.ARPE = 1,注意我們在寫程序的時候,給TIMx_ARR賦值,並沒有真正的寫入到自動重裝載寄存器中,而是寫入到了預加載寄存器中。
當我們需要定時器以T1和T2交替工作:
⑤ TIMx_CR1.ARPE = 0,自動重裝載寄存器沒有緩沖區,我們是在T1定時周期已經開始一會兒的時候,才去設定定時周期T1長度;在T2定時周期已經開始一會兒的時候,才去設定定時周期T2長度。因為當T1結束的時候,中斷發生后,我們在中斷程序中設定定時周期為T2。其實,此時定時器周期T2已經開始一段時間了。要知道定時器一個周期結束的時候,硬件自動進入下一個周期的計數,而不受軟件的控制。
⑥ TIMx_CR1.ARPE = 1,自動重裝載寄存器有緩沖區,我們是在T1定時周期一開始,就去設定定時周期T1的長度;在T2定時周期一開始,就去設定定時周期T2的長度。因為當T1結束的時候,更新事件產生(中斷也發生),(我們在上一個定時周期的中斷程序中已經設定定時周期為T2),TIMx_ARR中的T2值被硬件更新進入到自動重裝載寄存器中。
⑦ 當T1、T2兩個周期都很大的時候,需要ticks比較多,兩種方式都不會出現錯誤。
但是當T1、T2兩個周期都很小的時候,需要ticks比較少,對於“TIMx_CR1.ARPE = 0”的情況,就有可能出現問題。因為有可能在T1定時周期已經超過T1時間長度的時候,才去設定定時周期T1;在T2周期已經超過T2時間長度的時候,才去設定定時周期T2。
總結:
在需要不斷切換定時器的周期時,而且周期都比較短,程序員需要通過預加載寄存器配合自動重裝載寄存器,來操作定時器,以保證定時器周期的平穩過渡。
自動重裝載寄存器是預加載的,每次讀寫自動重裝載寄存器時,實際上是通過讀寫預加載寄存器實現。根據TIMx_CR1 寄存器中的自動重裝載預加載使能位(ARPE) ,寫入預加載寄存器的內容能夠立即或在每次更新事件時,傳送到它的影子寄存器。 摘自《STM32中文版芯片手冊》