博客專欄

EEPW首頁 > 博客 > 如何高效的擴展定時/計數(shù)器?

如何高效的擴展定時/計數(shù)器?

發(fā)布人:魚鷹談單片機 時間:2020-12-29 來源:工程師 發(fā)布文章

我們都知道,單片機往往都有定時器這個外設,定時器有時候也會用來作為計數(shù)器使用,在項目中它的的使用非常頻繁,但有時候卻滿足不了項目的需求。

比如 STM32F1 定時器,通過配置,可以讓定時器的時基為 1 ms,即1ms 計數(shù)器增加一次,等達到16位的極限,就會溢出,此時溢出時間 65536 ms = 65.5 s。

這個溢出時間一般能滿足需要,但時間精度卻是 ms 級別的,如何能達到更高的精度又能計時更長時間呢?

STM32 系列有兩種辦法:

1、使用更高級的單片機,比如 STM32F4,它的計數(shù)器是 32 位的,精度為 us 的話,也可以延時 4294967295 us = 71.5 min。但是涉及到成本問題。

2、使用主從方式定時,可以將 16 位計數(shù)器擴展到 32 位,但這將使用兩個定時器,對于定時器緊張的單片機不合適。

還有一種方式是采用軟件的方式,比如在一個定時器 1ms溢出時,使用變量遞增達到更長的延時時間,同理這種方式精度為 1 ms,如果想達到更高的精度,比如us,就必須把計時器的時間換算到us,然后加上變量的值:

5.png

這種方式很容易想到,一般人都會采用這種方式,同時很容易進行擴展,比如將變量從 32 位擴展到 64 位,即使精度為 us,也要很長很長的時間,這段時間,機器早就報廢了。

但是這種方式擴展的計數(shù)器,除非將變量擴展為 64位,否則,總會有溢出風險。

而且上述計算方式也是有問題的,32 位 * 1000,最后計算結(jié)果賦值給32位,這里會出現(xiàn)問題。

我們可以反算 us 精度下,time_ms 在什么值下time_us會出現(xiàn)溢出問題:

4294967295 / 1000 us 等于4,294,967.295 ms,也就是說,time_ms 不能達到這個溢出值,否則計算就會出現(xiàn)問題。

解決這個問題也簡單,就是將計算擴展為 64 位計算:

4.png

如果想延時更長時間,time_ms 使用 64 位,這樣就不必煩惱溢出風險問題,因為在機器有生之年應該是達不到溢出的時候(具體時間可以自己計算一下)。

如果真的需要運行很長的時間,溢出問題還是避免不了。那有什么辦法可以避免溢出風險呢?

事實上,魚鷹接下來介紹的方法,在計時方面唯一的好處就是可以避免溢出風險,但在脈沖計數(shù)方面卻有奇效!

如果只是單純的遞增計數(shù)器,那么也看不出比上面介紹的軟件方式有多好,但是如果你需要計數(shù)的是電機脈沖數(shù)呢,這個電機需要正反轉(zhuǎn)呢?

我們知道,電機有正反轉(zhuǎn),一般使用增量式編碼器來確定電機位置和運行方向:

3.png

比如上面編碼器輸出的脈沖波形,通過計數(shù)和判斷兩個波形的相位差,就可以知道電機處于正轉(zhuǎn)還是反轉(zhuǎn),同時通過計數(shù)器,即可達到精確的位置信息。

有經(jīng)驗的工程師應該知道,一般這種情況下的計數(shù)會采用定時器自帶的編碼器接口功能,使用該功能有以下幾個好處:

1、使用硬件計數(shù)方式,不占用 CPU(軟件方式是使用外部中斷進行計數(shù),需要占用CPU資源)

2、在電機轉(zhuǎn)速快的情況下,也不容易丟失脈沖數(shù)據(jù),更不會占用 CPU。

3、可以消除變向時的脈沖抖動問題

3、由硬件提供方向信息,即使你的電機控制程序未運行(已初始化),也能準確知道電機是否轉(zhuǎn)動和轉(zhuǎn)動方向(當有外部干擾電機運行時,也能準確知道位置和實際運行方向)。

正因為定時器的編碼器功能如此優(yōu)秀,一般在平衡車等需要精確知道電機的速度、方向、位置等信息時都會采用該接口功能。

但是你在網(wǎng)上看到的大部分資料只能獲得一圈的脈沖(位置)數(shù)據(jù),換向換的多了,你就不知道,當前位置是反轉(zhuǎn)或正轉(zhuǎn)的第幾圈的哪個位置了。

比如一個電機,正轉(zhuǎn)1.5圈、反轉(zhuǎn)2.4圈,再正轉(zhuǎn)3.2 圈……反反復復情況下,你知道它離原點的總運行距離嗎?

在配置好定時器的情況下,使用該該代碼即可得到準確運行位置(CNT值根據(jù)電機轉(zhuǎn)動方向遞增或遞減):

static int16_t last_cnt; // 上一次的脈沖。

static int32_t plus_cnt; // 相對開始位置的脈沖數(shù), 

int16_t temp,temp2; // 保持和 CNT 的位寬一致

temp = TIM2->CNT;  // ARR 設置為最大值 0xFFFF 即可

temp2 = temp - last_cnt;        // 必須分步

plus_cnt += temp2;  // 計算相對脈沖數(shù)  錯誤計算 plus_cnt += (temp - last_cnt);

last_cnt = temp;                // 保存上一次的值

限于篇幅,只說結(jié)論,關于原因,以后有時間再介紹,感興趣可以關注魚鷹。

先說這段代碼要獲得的效果,想象時間可以倒流,即下面的時針可以正向轉(zhuǎn)動,也可反向轉(zhuǎn)動,即可以在 12~6~12之間任意方向轉(zhuǎn)動,并且轉(zhuǎn)動沒有任何規(guī)律。

2.jpg

有一天你想知道,當前時間相比第一次觀察是倒流了還是流逝了多少時間?你是否有辦法準確得到這個時間呢?

1.png

如果僅從時針的位置,我們只能知道半天時間里的哪個時間(12 小時的某個時間點),而且還不知道到底在這半天是屬于倒流還是流逝!

但是通過上面的代碼,如果我們知道每一次時間流動時的方向,我們就可以準確知道這個時間是屬于第幾天的哪個時間點!

比如 plus 的值為 -25,我們就知道,時間倒流了 25 小時,根據(jù)這個時間,換算天數(shù)也就簡單了,倒流了一天又一小時。

現(xiàn)在繼續(xù)說說上面代碼注意點:

1、CNT 溢出值必須是位寬的最大值,即如果是 16 位計數(shù)器,最大值 0xFFFF,如果是 32 位,則是 0xFFFFFFFF。

2、記錄上一次的計數(shù)值 last_cnt 和 plus_cnt 必須是全局(或靜態(tài))變量。

3、因為有方向,所以聲明必須為有符號類型,這樣可以根據(jù)符號確定最終的方向。

4、必須分步計算,至于原因,簡單來說,就是只進行 16 位計算,得到的結(jié)果也只能是 16 位。

5、每次計算時,必須在上一次 CNT 值到它的一半之間內(nèi)計算一次,否則計算將出錯。

比如本次計算時,CNT = 123,下一次必須在它大概變成 123 + 32768 = 32891 或者 123 – 32768 = -32645 之前計算一次,否則最終得到的值將是錯誤的。

這樣的條件還是比較容易達到的,我們只要大概得到它最快的變化規(guī)律,就可以設置定時器讓它定時累積一次。

6、如果你只是單純的擴展定時器,因為定時器只會在一個方向計數(shù),假如是遞增,那么代碼如下:

static uint16_t last_cnt; // 上一次的脈沖。

static uint32_t plus_cnt; // 相對開始位置的脈沖數(shù), 

uint16_t temp,temp2; // 保持和 CNT 的位寬一致

temp = TIM2->CNT;  // ARR 設置為最大值 0xFFFF 即可

temp2 = temp - last_cnt;        // 必須分步

plus_cnt += temp2;  // 計算相對脈沖數(shù)  錯誤計算 plus_cnt += (temp - last_cnt);

last_cnt = temp;                // 保存上一次的值

只要改變變量類型即可。但是也要注意在它溢出前必須計算一次,否則就可能計算出錯,而且溢出值必須是最大值,而不能隨意更改。

以上結(jié)論可能對你而言比較難理解,但是當你有這種類似的需求,回過頭來再看這些,你會發(fā)現(xiàn)其中的巧妙。而當你真正理解了《延時功能進化論》是如何避免溢出風險的,相信有了魚鷹的提醒,理解它們也不是難事。

*博客內(nèi)容為網(wǎng)友個人發(fā)布,僅代表博主個人觀點,如有侵權請聯(lián)系工作人員刪除。

DIY機械鍵盤相關社區(qū):機械鍵盤DIY




關鍵詞:

相關推薦

技術專區(qū)

關閉