AVR單片機教程——數字IO寄存器


前兩篇教程中我們學習了LED、按鍵、開關的基本原理,數字輸入輸出的使用以及兩者之間的關系。我們用到了 pin_mode 、 pin_read 和 pin_write 這三個函數,實際上它們離最底層(至少是單片機制造商允許我們接觸到的最底層)就只有一步之遙了。而學單片機要是不了解一點底層,那跟Arduino玩家還有什么區別?(為防止有忠實的Arduino粉絲罵我,我得承認還是有一小部分Arduino玩家是知道本篇教程所介紹內容的。)根本不好意思說自己學過單片機好吧。這所謂的最底層,就是數字IO寄存器了。

在開始之前,你需要下載兩份文檔:

單片機的數據手冊。官網鏈接極慢,我在國內平台上傳了一份,在本篇教程寫成之時是最新的。

開發板的原理圖。本應在教程之初就放出來,但事實證明沒有原理圖也不影響使用。現在是肯定需要的。

 

等等,你可能還不知道寄存器是什么。那我們就從寄存器開始吧。

寄存器是一類CPU內部的存儲器,分為通用寄存器與特殊功能寄存器(8086對特殊功能寄存器還有細分)。通用寄存器,顧名思義是通用的,可以存儲操作數、運算結果、內存地址等數據,在使用C語言編程時,一般不直接接觸通用寄存器,而由編譯器負責安排其使用。特殊功能寄存器有特定的功能,一些作用於CPU內部,如PC存放下一條指令的地址,SP記錄內存中棧頂的位置(現在無需了解這些);另一些與IO模組相連接,單片機程序通過這類寄存器來控制各種外設,我們今天要學習的數字信號IO寄存器就屬於這一類,並且應該是其中最簡單的了。

我們使用的單片機的型號是ATmega324PA,它有多種封裝,引腳(pin)數不盡相同,但都有32個通用輸入輸出(GPIO)引腳。由於AVR架構是8位字長的,CPU一次處理1位數據和8位數據所需的時間是一樣的,這32個引腳被組織為4個端口(port),分別是PA、PB、PC和PD。

在AVR架構tiny與mega系列的單片機中,每個端口都有3個寄存器控制數字信號IO,分別是PORTx、DDRx和PINx。這里的x是A、B、C或D,由於這4個端口在數字IO方面完全相同,就把它們合並起來講。相應地,對於每個引腳Pxn,有PORTxn、DDxn(沒有R)和PINxn三個bit控制其數字IO。

DDxn控制引腳方向:當DDxn為1時,Pxn為輸出;當DDxn為0時,Pxn為輸入。

當Pxn為輸入時,如果PORTxn為1,則該引腳通過一個上拉電阻連接到VCC;否則引腳懸空。

當Pxn為輸出時,如果PORTxn為1,引腳輸出高電平;否則輸出低電平。

PINxn的值為Pxn引腳的電平。如果給PINxn寫入1,PORTxn的值會翻轉。

還有很多細節問題,如MCUCR寄存器中PUD位的功能、復位后寄存器的值、輸入輸出切換的方法、讀取引腳電平的延遲、未連接引腳的處理方法等,留作今天的作業,閱讀數據手冊I/O-Ports一章中除Alternate Port Functions一節以外的內容(一共8頁不到,不多吧),找出這些問題的答案,並以此為基礎回答上一篇教程最后的問題。

 

講了這么多,相信你也沒記住多少,而且你也不知道去哪里用這些寄存器。

要使用寄存器,你需要在C語言程序中寫 #include <avr/io.h> (在創建項目時自動生成的代碼中就有),然后就可以使用 PORTA 、 DDRB 、 PINC 等寄存器了。它們是宏定義,你不必去探究它們展開后是怎樣的,只需知道這些宏可以讀取,可以賦值,可以位操作,就像 uint8_t 類型變量一樣。

但是諸如 PORTA0 和 DDB7 等宏定義卻不代表寄存器上的那一位,它們實際上就是字面值常量,如 PORTAx 的意義是寄存器 PORTA 的第x位(第0位為最低位,第7位為最高位),它的值就是x。因此,直接對這些宏復制是不正確的(不僅意義不正確,編譯也不會通過)。

在開發板的庫函數中的 <ee1/bit.h> 提供了包含幾個用於位操作的宏函數。我們先按照手冊來用,稍后來看它們是如何實現的。

 

我們先返璞歸真一下,回到最初的例子,點亮一個LED,不過這次我們不再使用 <ee1/led.h> 提供的函數,而是直接操作寄存器。

先點亮紅色LED吧。在原理圖的第2頁左上角,紅色LED通過一個電阻連接到網絡LED0,而在第1頁中LED0連接的是單片機PC4引腳,因此我們需要讓PC4引腳輸出高電平。回到上面看一下三個寄存器的功能,輸出高電平需要DDxn和PORTxn同時為1。這里把x和n分別用C和4帶入,即我們要讓DDC4和PORTC4為1。

將一個寄存器的一位置為1可以由 set_bit 實現。它需要兩個參數,要操作的整型變量與表示第幾位的整數。把DDC4置為1應該寫 set_bit(DDRC, 4); ,4 可以用 DDC4 替換,這個定義就是這么用的。類似地也可以將PORTC4置為1。點亮紅色LED的整個程序如下:

1 #include <avr/io.h>
2 #include <ee1/bit.h>
3 
4 int main(void)
5 {
6     set_bit(DDRC , 4);
7     set_bit(PORTC, 4);
8 }

相信聰明的你已經知道閃爍和流水燈怎么寫了。翻轉輸出電平可以使用 flip_bit(PORTC, 4); ,也可以使用 set_bit(PINC, 4); 。

 

下面來看數字輸入。還是用第一個與按鍵相關的例子,讓LED狀態與按鍵保持一致,即按下亮起。

讀取一個寄存器中的一位可以使用 read_bit。如果引腳上電平為高,read_bit 的運算結果非0(但不一定是布爾值1)。如果你沒有忘記的話,按鍵按下時引腳電平為低,因此對讀取引腳電平的結果取非才是按鍵是否按下。

在原理圖中,按鍵一端連接在BTN0網絡上,進而連接到單片機的PA4引腳。因此按鍵是否按下應該寫為:!read_bit(PINA, 4) 。

在讀取之前應該先把引腳配置為輸入。盡管復位后默認為輸入,在這個例子中沒有必要向DDA4寫0,但明確寫出來可以讓看這段代碼的人(可能別人也可能是你自己)明白PA4是作輸入的,這樣做是一種良好的習慣。至於PORTA4,由於這一引腳在外部有連接上拉電阻,就沒有必要啟用內部上拉電阻了。

 1 #include <avr/io.h>
 2 #include <ee1/bit.h>
 3 
 4 int main(void)
 5 {
 6     reset_bit(DDRA, 4);
 7     set_bit(DDRC, 4);
 8     while (1)
 9     {
10         cond_bit(!read_bit(PINA, 4), PORTC, 4);
11     }
12 }

再結合按鍵動作的知識,你應該知道怎樣直接通過寄存器操作來判斷按鍵動作了吧。

順便說一句,以上兩個程序都不必在項目屬性中給linker加上libee1庫。雖然代碼中使用了 <ee1/bit.h> ,但其中都是宏定義,與linker無關。

 

之前留了一個問題,就是位操作是如何實現的。以下為 <ee1/bit.h> 中部分代碼:

1 #define set_bit  (r, b) ((r) |=  (1u << (b)))
2 #define reset_bit(r ,b) ((r) &= ~(1u << (b)))
3 #define read_bit (r, b) ((r) &   (1u << (b)))
4 #define flip_bit (r, b) ((r) ^=  (1u << (b)))

寫那么多括號是為了防止出現運算符優先級的問題。假設r就是一個寄存器,比如PORTC,b就是一個數字,比如4,也可以是一個變量,那么 (r) |= (1u << (b)) 就相當於 r = r | 1u << b (后綴u表示無符號數,位操作的運算數一般都是無符號數)。對於二進制表示下的每一位,如果不是第b位,那么位或運算符右邊此位為0,運算結果等於左邊,即r的這些為保持不變;對於第b位,右邊此位為1,無論左邊此位的值是多少,結果一定是1,即這一位被置1;這樣就實現了將一位置為1的功能。

reset_bit 的實現還要繞一個彎。1u << b 是一個第b位為1,其余位為0的數,那么 ~(1u << b) ,即位與賦值號右邊,是一個第b位為0而其余為都是1的數。仿照上面的分析可得,運算結果的第b位一定是0而其余位與r中原來的值相同。類似的分析也可應用於 flip_bit :兩個bit進行異或運算的結果,若相同則為1,不同則為0;當一個運算數是1時,結果就是另一個運算數取反;當一個運算數是0時,結果與另一個運算數相同;因此 flip_bit 就使r的第b為取反而其他為不變。

以上是向寄存器中的位寫入的操作。用於讀取位的 read_bit 的原理也大致相同,用寄存器的值與 1u << b 相與,僅當第b位為1時結果是 1u << b ,這是個非零數,否則結果為0。read_bit 語句可以直接放在 if 語句的條件部分,但如果是根據其結果決定一個變量是否加1,不能直接加上其運算結果,可以轉換成 bool 類型或用 if 語句判斷。

 

這篇教程有點長。好好消化一下,然后把以前寫過的程序用寄存器重新寫一遍,以此鞏固所學的知識。

 

從本教程開始至今,我們先了解了LED燈、按鍵、撥動開關、數字輸入輸出的使用方法,然后學習C語言位操作與數字IO寄存器,終於打通了一條從底層到應用的路。而網絡上很多教程都是反過來講的,即先介紹寄存器,然后直接通過寄存器來驅動LED、檢測按鍵等,甚至有直接寫諸如 DDRB |= 0x0C; 或 if (PINB & 0x40) 這樣的代碼的,初學者怎么看得明白?站在我的角度,我覺得以上都是常識,都不用講,盡管我學習的時候也頗費周折(正是因為那些反過來的教程)。現在我站在初學者的角度,認為本教程的講解順序是更容易理解的。

我學習計算機之前,總對計算機抱有特殊的幻想,覺得它什么都能干,很神奇。現在這些想法都沒有了,尤其是在學習單片機的過程中。學習計算機教會我們分析問題、解決問題,而學習單片機讓我們更好地理解計算機是如何按照我們的想法來解決問題的。這篇教程帶你了解了寄存器,在你學習單片機的全過程中,它都會伴隨着你。寄存器是硬件和軟件之間的一個重要紐帶,計算機的任何功能都離不開寄存器。CPU?有寄存器。總線通信?通過寄存器。內存分頁?需要寄存器。萬物基於寄存器。又有更多像寄存器一樣的紐帶,在電子空穴與豐富多彩的計算機世界之間建立起聯系。它們看起來如此復雜,卻又清晰明了,就算一夜之間所有計算機都突然消失,人類也能從電子管和打孔紙帶開始,一層一層地構建起計算機的世界。而我們了解的只不過是這個巨大體系中的滄海一粟。

初入計算機世界,你想着計算機能干什么,學完計算機我能干什么。而計算機世界是如此高深,在逐漸深入后,你會明白計算機不能干什么,我不能干什么。數碼管、蜂鳴器,它們一直在你的開發板上,你卻不知道如何使用它們;更不用說那些高級到你沒聽說過的東西。學得越多,你會發現雖然原本不會的減少了,但腦海中萌生出的“不切實際”的想法更多——學習的速度永遠趕不上認知的速度。本系列教程不能幫你消除所有“不會”,而是要在帶你一步步消除一些“不會”的過程中讓你學會發現更多“不會”並消除的方法。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM