通信,按照傳統的理解就是信息的傳輸與交換。對于單片機來說,通信則與傳感器、存儲芯片、外圍控制芯片等技術緊密結合,成為整個單片機系統的“神經中樞”。沒有通信,單片機所實現的功能僅僅局限于單片機本身,就無法通過其它設備獲得有用信息,也無法將自己產生的信息告訴其它設備。如果單片機通信沒處理好的話,它和外圍器件的合作程度就受到限制,最終整個系統也無法完成強大的功能,由此可見單片機通信技術的重要性。UART(Universal Asynchronous Receiver/Transmitter,即通用異步收發器)串行通信是單片機最常用的一種通信技術,通常用于單片機和電腦之間以及單片機和單片機之間的通信。
11.1 串行通信的初步認識
通信按照基本類型可以分為并行通信和串行通信。并行通信時數據的各個位同時傳送,可以實現字節為單位通信,但是通信線多占用資源多,成本高。比如我們前邊用到的P0 = 0xFE;一次給P0的8個IO口分別賦值,同時進行信號輸出,類似于有8個車道同時可以過去8輛車一樣,這種形式就是并行的,我們習慣上還稱P0、P1、P2和P3為51單片機的4組并行總線。
而串行通信,就如同一條車道,一次只能一輛車過去,如果一個0xFE這樣一個字節的數據要傳輸過去的話,假如低位在前高位在后的話,那發送方式就是0-1-1-1-1-1-1-1-1,一位一位的發送出去的,要發送8次才能發送完一個字節。
STC89C52有兩個引腳是專門用來做UART串行通信的,一個是P3.0一個是P3.1,它們還分別有另外的名字叫做RXD和TXD,由它們組成的通信接口就叫做串行接口,簡稱串口。用兩個單片機進行UART串口通信,基本的演示圖如圖11-1所示。
圖11-1 單片機之間UART通信示意圖
圖中,GND表示單片機系統電源的參考地,TXD是串行發送引腳,RXD是串行接收引腳。兩個單片機之間要通信,首先電源基準得一樣,所以我們要把兩個單片機的GND相互連接起來,然后單片機1的TXD引腳接到單片機2的RXD引腳上,即此路為單片機1發送而單片機2接收的通道,單片機1的RXD引腳接到單片機2的TXD引腳上,即此路為單片機2發送而單片機1接收的通道。這個示意圖就體現了兩個單片機相互收發信息的過程。
當單片機1想給單片機2發送數據時,比如發送一個0xE4這個數據,用二進制形式表示就是0b11100100,在UART通信過程中,是低位先發,高位后發的原則,那么就讓TXD首先拉低電平,持續一段時間,發送一位0,然后繼續拉低,再持續一段時間,又發送了一位0,然后拉高電平,持續一段時間,發了一位1……一直到把8位二進制數字0b11100100全部發送完畢。這里就涉及到了一個問題,就是持續的這“一段時間”到底是多久?由此便引入了通信中的一個重要概念——波特率,也叫做比特率。
波特率就是發送二進制數據位的速率,習慣上用baud表示,即我們發送一位二進制數據的持續時間=1/baud。在通信之前,單片機1和單片機2首先都要明確的約定好它們之間的通信波特率,必須保持一致,收發雙方才能正常實現通信,這一點大家一定要記清楚。
約定好速度后,我們還要考慮第二個問題,數據什么時候是起始,什么時候是結束呢?不管是提前接收還是延遲接收,數據都會接收錯誤。在UART通信的時候,一個字節是8位,規定當沒有通信信號發生時,通信線路保持高電平,當要發送數據之前,先發一位0表示起始位,然后發送8位數據位,數據位是先低后高的順序,數據位發完后再發一位1表示停止位。這樣本來要發送一個字節的8位數據,而實際上我們一共發送了10位,多出來的兩位其中一位起始位,一位停止位。而接收方呢,原本一直保持的高電平,一旦檢測到了一位低電平,那就知道了要開始準備接收數據了,接收到8位數據位后,然后檢測到停止位,再準備下一個數據的接收。我們圖示看一下,如圖11-2所示。
圖11-2 串口數據發送示意圖
圖11-2串口數據發送示意圖,實際上是一個時域示意圖,就是信號隨著時間變化的對應關系。比如在單片機的發送引腳上,左邊的是先發生的,右邊的是后發生的,數據位的切換時間就是波特率分之一秒,如果能夠理解時域的概念,后邊很多通信的時序圖就很容易理解了。
11.2 RS232通信接口
在我們的臺式電腦上,一般都會有一個9針的串行接口,這個串行接口叫做RS232接口,它和UART通信有關聯,但是由于現在筆記本電腦都不帶這種9針串口了,所以和單片機通信越來越趨向于使用USB虛擬的串口,因此這一節的內容作為了解內容,大家知道有這么回事就行了。
我們先來認識一下這個標準串口,在物理結構上分為9針的和9孔的,習慣上我們也稱之為公頭和母頭,如圖11-3所示。
圖11-3 RS232通信接口
RS232接口一共有9個引腳,分別定義是:1、載波檢測DCD;2、接收數據RXD;3、發送數據TXD;4、數據終端準備好DTR;5、信號地線SG;6、數據準備好DSR;7、請求發送RTS;8、清除發送CTS;9、振鈴提示RI。我們要讓這個串口和我們單片機進行通信,我們只需要關心其中的2腳RXD、3腳TXD和5腳GND即可。
雖然這三個引腳的名字和我們單片機上的串口名字一樣,但是卻不能直接和單片機對連通信,這是為什么呢?隨著我們了解的內容越來越多,我們得慢慢知道,不是所有的電路都是5V代表高電平而0V代表低電平的。對于RS232標準來說,它是個反邏輯,也叫做負邏輯。為何叫負邏輯?它的TXD和RXD的電壓,-3V~-15V電壓代表是1,+3~+15V電壓代表是0。低電平代表的是1,而高電平代表的是0,所以稱之為負邏輯。因此電腦的9針RS232串口是不能和單片機直接連接的,需要用一個電平轉換芯片MAX232來完成,如圖11-4所示。
圖11-4 MAX232轉接圖
這個芯片就可以實現把標準RS232串口電平轉換成我們單片機能夠識別和承受的UART 0V/5V電平。從這里大家似乎慢慢有點明白了,其實RS232串口和UART串口,它們的協議類型是一樣的,只是電平標準不同而已,而MAX232這個芯片起到的就是中間人的作用,它把UART電平轉換成RS232電平,也把RS232電平轉換成UART電平,從而實現標準RS232接口和單片機UART之間的通信連接。
11.3 USB轉串口通信
隨著技術的發展,工業上還有RS232串口通信的大量使用,但是商業技術的應用上,已經慢慢的使用USB轉UART技術取代了RS232串口,絕大多數筆記本電腦已經沒有串口這個東西了,那我們要實現單片機和電腦之間的通信該怎么辦呢?
我們只需要在電路上添加一個USB轉串口芯片,就可以成功實現USB通信協議和標準UART串行通信協議的轉換,在我們的開發板上,我們使用的是CH340T這個芯片,如圖11-5所示。
圖11-5 USB轉串口電路
圖中左下方J1和J2是兩個跳線的組合,大家可以在我們板子左下方的位置找到,我們需要用跳線帽把中間和下邊的針短接在一起。右側的CH340T這個電路很簡單,把電源、晶振接好后,6腳和7腳的DP和DM分別接USB口的2個數據引腳上去,3腳和4腳通過跳線接到了我們單片機的TXD和RXD上去。
CH340T的電路里3腳位置加了個4148的二極管,是一個小技巧。因為STC89C52這個單片機下載程序時需要冷啟動,就是先點下載后上電,上電瞬間單片機會先檢測需要不需要下載程序。雖然單片機的VCC是由開關來控制,但是由于CH340T的3腳是輸出引腳,如果沒有此二極管,開關后級單片機在斷電的情況下,CH340T的3腳和單片機的P3.0(即RXD)引腳連在一起,有電流會通過這個引腳流入后級電路并且給后級的電容充電,造成后級有一定幅度的電壓,這個電壓值雖然只有兩三伏左右,但是可能會影響到正常的冷啟動。加了二極管后,一方面不影響通信,另外一個方面還可以消除這種不良影響。這個地方可以暫時作為了解,大家如果自己做這類電路,可以參考一下。
11.4 IO口模擬UART串口通信
為了讓大家充分理解UART串口通信的原理,我們先把P3.0和P3.1當做IO口來進行模擬實際串口通信的過程,原理搞懂后,我們再使用寄存器配置實現串口通信過程。
對于UART串口波特率,常用的值是300、600、1200、2400、4800、9600、14400、19200、28800、38400、57600、115200等速率。IO口模擬UART串行通信程序是一個簡單的演示程序,我們使用串口調試助手下發一個數據,數據加1后,再自動返回。
串口調試助手,這里我們直接使用STC-ISP軟件自帶的串口調試助手,先把串口調試助手的使用給大家說一下,如圖11-6所示。第一步要選擇串口助手菜單,第二步選擇十六進制顯示,第三步選擇十六進制發送,第四步選擇COM口,這個COM口要和自己電腦設備管理器里的那個COM口一致,波特率按我們程序設定好的選擇,我們程序中讓一個數據位持續時間是1/9600秒,那這個地方選擇波特率就是選9600,校驗位選N,數據位8,停止位1。
圖11-6 串口調試助手示意圖
串口調試助手的實質就是利用電腦上的UART通信接口,發送數據給我們的單片機,也可以把我們的單片機發送的數據接收到這個調試助手界面上。
因為初次接觸通信方面的技術,所以我把后面的IO模擬串口通信程序進行一下解釋,大家可以邊看我的解釋邊看程序,把底層原理先徹底弄懂。
變量定義部分就不用說了,直接看main主函數。首先是對通信的波特率的設定,在這里我們配置的波特率是9600,那么串口調試助手也得是9600。配置波特率的時候,我們用的是定時器T0的模式2。模式2中,不再是TH0代表高8位,TL0代表低8位了,而只有TL0在進行計數,當TL0溢出后,不僅僅會讓TF0變1,而且還會將TH0中的內容重新自動裝到TL0中。這樣有一個好處,就是我們可以把想要的定時器初值提前存在TH0中,當TL0溢出后,TH0自動把初值就重新送入TL0了,全自動的,不需要程序中再給TL0重新賦值了,配置方式很簡單,大家可以自己看下程序并且計算一下初值。
波特率設置好以后,打開中斷,然后等待接收串口調試助手下發的數據。接收數據的時候,首先要進行低電平檢測while (PIN_RXD),若沒有低電平則說明沒有數據,一旦檢測到低電平,就進入啟動接收函數StartRXD()。接收函數最開始啟動半個波特率周期,初學可能這里不是很明白。大家回頭看一下我們的圖11-2里邊的串口數據示意圖,如果在數據位電平變化的時候去讀取,因為時序上的誤差以及信號穩定性的問題很容易讀錯數據,所以我們希望在信號最穩定的時候去讀數據。除了信號變化的那個沿的位置外,其它位置都很穩定,那么我們現在就約定在信號中間位置去讀取電平狀態,這樣能夠保證我們讀的一定是正確的。
一旦讀到了起始信號,我們就把當前狀態設定成接收狀態,并且打開定時器中斷,第一次是半個周期進入中斷后,對起始位進行二次判斷一下,確認一下起始位是低電平,而不是一個干擾信號。以后每經過1/9600秒進入一次中斷,并且把這個引腳的狀態讀到RxdBuf里邊。等待接收完畢之后,我們再把這個RxdBuf加1,再通過TXD引腳發送出去,同樣需要先發一位起始位,然后發8個數據位,再發結束位,發送完畢后,程序運行到while (PIN_RXD),等待第二輪信號接收的開始。
11.5 UART串口通信的基本應用
11.5.1 通信的三種基本類型
常用的通信從傳輸方向上可以分為單工通信、半雙工通信、全雙工通信三類。
單工通信就是指只允許一方向另外一方傳送信息,而另一方不能回傳信息。比如電視遙控器、收音機廣播等,都是單工通信技術。
半雙工通信是指數據可以在雙方之間相互傳播,但是同一時刻只能其中一方發給另外一方,比如我們的對講機就是典型的半雙工。
全雙工通信就發送數據的同時也能夠接收數據,兩者同步進行,就如同我們的電話一樣,我們說話的同時也可以聽到對方的聲音。
11.5.2 UART模塊介紹
IO口模擬串口通信,讓大家了解了串口通信的本質,但是我們的單片機程序卻需要不停的檢測掃描單片機IO口收到的數據,大量占用了單片機的運行時間。這時候就會有聰明人想了,其實我們并不是很關心通信的過程,我們只需要一個通信的結果,最終得到接收到的數據就行了。這樣我們可以在單片機內部做一個硬件模塊,讓它自動接收數據,接收完了,通知我們一下就可以了,我們的51單片機內部就存在這樣一個UART模塊,要正確使用它,當然還得先把對應的特殊功能寄存器配置好。
51單片機的UART串口的結構由串行口控制寄存器SCON、發送和接收電路三部分構成,先來了解一下串口控制寄存器SCON。如表11-1表11-2所示。
表11-1 SCON——串行控制寄存器的位分配(地址0x98、可位尋址)
表11-2 SCON——串行控制寄存器的位描述
前邊學了那么多寄存器的配置,相信SCON這個地方,對于大多數同學來說已經不是難點了,應該能看懂并且可以自己配置了。對于串口的四種模式,模式1是最常用的,就是我們前邊提到的1位起始位,8位數據位和1位停止位。下面我們就詳細介紹模式1的工作細節和使用方法,至于其它3種模式與此也是大同小異,真正遇到需要使用的時候大家再去查閱相關資料就行了。
在我們使用IO口模擬串口通信的時候,串口的波特率是使用定時器T0的中斷體現出來的。在硬件串口模塊中,有一個專門的波特率發生器用來控制發送和接收數據的速度。對于STC89C52單片機來講,這個波特率發生器只能由定時器T1或定時器T2產生,而不能由定時器T0產生,這和我們模擬的通信是完全不同的概念。
如果用定時器2,需要配置額外的寄存器,默認是使用定時器1的,我們本章內容主要就使用定時器T1作為波特率發生器來講解,方式1下的波特率發生器必須使用定時器T1的模式2,也就是自動重裝載模式,定時器的重載值計算公式為:
TH1 = TL1 = 256 - 晶振值/12 /2/16 /波特率
和波特率有關的還有一個寄存器,是一個電源管理寄存器PCON,他的最高位可以把波特率提高一倍,也就是如果寫PCON |= 0x80以后,計算公式就成了:
TH1 = TL1 = 256 - 晶振值/12 /16 /波特率
公式中數字的含義這里解釋一下,256是8位定時器的溢出值,也就是TL1的溢出值,晶振值在我們的開發板上就是11059200,12是說1個機器周期等于12個時鐘周期,值得關注的是這個16,我們來重點說明。在IO口模擬串口通信接收數據的時候,采集的是這一位數據的中間位置,而實際上串口模塊比我們模擬的要復雜和精確一些。他采取的方式是把一位信號采集16次,其中第7、8、9次取出來,這三次中其中兩次如果是高電平,那么就認定這一位數據是1,如果兩次是低電平,那么就認定這一位是0,這樣一旦受到意外干擾讀錯一次數據,也依然可以保證最終數據的正確性。
了解了串口采集模式,在這里要給大家留一個思考題。“晶振值/12/2/16/波特率”這個地方計算的時候,出現不能除盡,或者出現小數怎么辦,允許出現多大的偏差?把這部分理解了,也就理解了我們的晶振為何使用11.0592M了。
串口通信的發送和接收電路在物理上有2個名字相同的SBUF寄存器,它們的地址也都是0x99,但是一個用來做發送緩沖,一個用來做接收緩沖。意思就是說,有2個房間,兩個房間的門牌號是一樣的,其中一個只出人不進人,另外一個只進人不出人,這樣的話,我們就可以實現UART的全雙工通信,相互之間不會產生干擾。但是在邏輯上呢,我們每次只操作SBUF,單片機會自動根據對它執行的是“讀”還是“寫”操作來選擇是接收SBUF還是發送SBUF,后邊通過程序,我們就會徹底了解這個問題。
11.5.3 UART串口程序
一般情況下,我們編寫串口通信程序的基本步驟如下所示:
1、配置串口為模式1。
2、配置定時器T1為模式2,即自動重裝模式。
3、根據波特率計算TH1和TL1的初值,如果有需要可以使用PCON進行波特率加倍。
4、打開定時器控制寄存器TR1,讓定時器跑起來。
這里還要特別注意一下,就是在使用T1做波特率發生器的時候,千萬不要再使能T1的中斷了。
我們先來看一下由IO口模擬串口通信直接改為使用硬件UART模塊時的程序代碼,看看程序是不是簡單了很多,因為大部分的工作硬件模塊都替我們做了。程序功能和IO口模擬的是完全一樣的。
當然了,這個程序還是用在主循環里等待接收中斷標志位和發送中斷標志位的方法來編寫的,而實際工程開發中,當然就不能這么干了,我們也只是為了用直觀的對比來告訴同學們硬件模塊可以大大簡化程序代碼,那么實際使用串口的時候就用到串口中斷了,來看一下用中斷實現的程序。請注意一點,因為接收和發送觸發的是同一個串口中斷,所以在串口中斷函數中就必須先判斷是哪種中斷,然后再作出相應的處理。
大家可以試驗一下,看看是不是和前邊用IO口模擬通信實現的效果一致,而主循環卻完全空出來了,我們就可以隨意添加其它功能代碼進去。
11.6 通信實例與ASCII碼
我們學習串口通信主要是要實現單片機和電腦之間的信息交互,可以用電腦控制單片機的一些信息,可以把單片機的一些信息狀況發給電腦上的軟件。下面我們就做一個簡單的例程,實現單片機串口調試助手發送的數據,在我們開發板上的數碼管上顯示出來。
大家在做這個實驗的時候,有個小問題要注意一下。因為STC89C52下載程序是使用了UART串口下載,下載完程序后,程序運行起來了,可是下載軟件最后還會通過串口發送一些額外的數據,所以程序剛下載進去不是顯示00,而可能是其他數據。大家只要把電源開關關閉,重新打開一次就好了。
細心的同學可能會發現,在串口調試助手發送選項和接收選項處,還有個“字符格式發送”和“字符格式顯示”,這是什么意思呢?
先拋開我們使用的漢字不談,那么我們常用的字符就包含了09的數字、AZ/a~z的字母、還有各種標點符號等。那么在單片機系統里面我們怎么來表示它們呢?ASCII碼(American Standard Code for Information Interchange,即美國信息互換標準代碼)可以完成這個使命:我們知道,在單片機中一個字節的數據可以有0~255共256個值,我們取其中的0~127共128個值賦予了它另外一層涵義,即讓它們分別來代表一個常用字符,其具體的對應關系如表11-3所示。
表11-3 ASCII碼字符表
這樣我們就在常用字符和字節數據之間建立了一一對應的關系,那么現在一個字節就既可以代表一個整數又可以代表一個字符了,但它本質上只是一個字節的數據,而我們賦予了它不同的涵義,什么時候賦予它哪種涵義就看編程者的意圖了。ASCII碼在單片機系統中應用非常廣泛,我們后續的課程也會經常使用到它,下面我們來對它做一個直觀的認識,同學們一定要深刻理解其本質。
對照上述表格,我們就可以實現字符和數字之間的轉換了,比如還是這個程序,我們發送的時候改成字符格式發送,接收還是用十六進制接收,這樣接收和數碼管好做一下對比。
我們用字符格式發送一個小寫的a,返回一個十六進制的0x61,數碼管上顯示的也是61,ASCII碼表里字符a對應十進制是97,等于十六進制的0x61;我們再用字符格式發送一個數字1,返回一個十六進制的0x31,數碼管上顯示的也是31,ASCII表里字符1對應的十進制是49,等于十六進制的0x31。這下大家就該清楚了:所謂的十六進制發送和十六進制接收,都是按字節數據的真實值進行的;而字符格式發送和字符格式接收,是按ASCII碼表中字符形式進行的,但它實際上最終傳輸的還是一個字節數據。這個表格,當然不需要大家去記住,理解它,用的時候過來查就行了。
通信的學習,不像前邊控制部分那么直觀了,通信部分我們的程序只能獲得一個結果,而其過程我們卻無法直接看到,所以慢慢的可能大家就會知道有示波器和邏輯分析儀這類測量儀器。如果學校實驗室或者公司里有示波器或者邏輯分析儀這類儀器,可以拿過來抓一下串口波形,直觀的了解一下。如果暫時還沒有這些儀器,先知道這么回事,有條件再說。因為工具類設備有的比較昂貴,有條件可以盡量使用學校或者公司的。在這里我用一款簡易的邏輯分析儀把串口通信的波形抓出來給大家看一下,大家了解一下即可,如圖11-7所示。
圖11-7 邏輯分析儀串口數據示意圖
分析儀和示波器的作用,就是把通信過程的波形抓出來進行分析。先大概說一下波形的意思。波形左邊是低位,右邊是高位,上邊這個波形是電腦發送給單片機的,下邊這個波形是單片機回發給電腦的。以上邊的波形為例,左邊第一位是起始位0,從低位到高位依次是10001100,順序倒一下,就是數據0x31,也就是ASCII碼表里的‘1’。大家可以注意到分析儀在每個數據位都給標了一個白色的點,表示是數據,起始位和無數據的時候都沒有這個白點。時間標T1和T2的差值在右邊顯示出來是0.102ms,大概是9600分之一,稍微有點偏差,在容許范圍內即可。通過圖11-7,我們可以清晰的了解了串口通信的收發的詳細過程。
那我們這里再來了解一下,如果我們使用串口調試助手,用字符格式直接發送一個“12”,我們在我們的數碼管上應該顯示什么呢?串口調試助手應該返回什么呢?經過試驗發現,我們數碼管顯示的是32,而串口調試助手返回十六進制顯示的是31、32兩個數據,如圖11-8所示。
圖11-8 串口調試助手數據顯示
我們再用邏輯分析儀把這個數據抓出來看一下,如圖11-9所示。
圖11-9 邏輯分析儀抓取數據
對于ASCII碼表來說,數字本身是字符而非數據,所以如果發送“12”的話,實際上是是分別發送了“1”和“2”兩個字符,單片機呢,先收到第一個字符“1”,在數碼管上會顯示出31這個對應數字,但是馬上就又收到了“2”這個字符,數碼管瞬間從31變成了32,而我們視覺上呢,是沒有辦法發現這種快速變化的,所以我們感覺數碼管直接顯示的是32。
評論
查看更多