為了讓大家充分了解 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),等待第二輪信号接收的開始。
#include
sbit PIN_RXD = P3^0; //接收引腳定義
sbit PIN_TXD = P3^1; //發送引腳定義
bit RxdOrTxd = 0; //訓示目前狀态為接收還是發送
bit RxdEnd = 0; //接收結束标志
bit TxdEnd = 0; //發送結束标志
unsigned char RxdBuf = 0; //接收緩沖器
unsigned char TxdBuf = 0; //發送緩沖器
void ConfigUART(unsigned int baud);
void StartTXD(unsigned char dat);
void StartRXD();
void main(){
EA = 1; //開總中斷
ConfigUART(9600);
while (1){ //配置波特率為 9600
while (PIN_RXD); //等待接收引腳出現低電平,即起始位
StartRXD(); //啟動接收
while (!RxdEnd); //等待接收完成
StartTXD(RxdBuf+1); //接收到的資料+1 後,發送回去
while (!TxdEnd); //等待發送完成
}
}
void ConfigUART(unsigned int baud){
TMOD &= 0xF0; //清零 T0 的控制位
TMOD |= 0x02; //配置 T0 為模式 2
TH0 = 256 - (11059200/12)/baud; //計算 T0 重載值
}
void StartRXD(){
TL0 = 256 - ((256-TH0)>>1); //接收啟動時的 T0 定時為半個波特率周期
ET0 = 1; //使能 T0 中斷
TR0 = 1; //啟動 T0
RxdEnd = 0; //清零接收結束标志
RxdOrTxd = 0; //設定目前狀态為接收
}
void StartTXD(unsigned char dat){
TxdBuf = dat; //待發送資料儲存到發送緩沖器
TL0 = TH0; //T0 計數初值為重載值
ET0 = 1; //使能 T0 中斷
TR0 = 1; //啟動 T0
PIN_TXD = 0; //發送起始位
TxdEnd = 0; //清零發送結束标志
RxdOrTxd = 1; //設定目前狀态為發送
}
void InterruptTimer0() interrupt 1{
static unsigned char cnt = 0; //位接收或發送計數
if (RxdOrTxd){ //串行發送處理
cnt++;
if (cnt <= 8){ //低位在先依次發送 8bit 資料位
PIN_TXD = TxdBuf & 0x01;
TxdBuf >>= 1;
}else if (cnt == 9){ //發送停止位
PIN_TXD = 1;
}else{ //發送結束
cnt = 0; //複位 bit 計數器
TR0 = 0; //關閉 T0
TxdEnd = 1; //置發送結束标志
}
}else{ //串行接收處理
if (cnt == 0){ //處理起始位
if (!PIN_RXD){ //起始位為 0 時,清零接收緩沖器,準備接收資料位
RxdBuf = 0;
cnt++;
}
}else{ //起始位不為 0 時,中止接收
TR0 = 0; //關閉 T0
}else if (cnt <= 8){ //處理 8 位資料位
RxdBuf >>= 1; //低位在先,是以将之前接收的位向右移
//接收腳為 1 時,緩沖器最高位置 1,
//而為 0 時不處理即仍保持移位後的 0
if (PIN_RXD){
RxdBuf |= 0x80;
}
cnt++;
}else{ //停止位處理
cnt = 0; //複位 bit 計數器
TR0 = 0; //關閉 T0
if (PIN_RXD){ //停止位為 1 時,方能認為資料有效
RxdEnd = 1; //置接收結束标志
}
}
}
}
本文轉自:C語言中文網
http://c.biancheng.net/cpp/html/1921.html