新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 單片機(jī)設(shè)計(jì):軟件UART的設(shè)計(jì)思想

單片機(jī)設(shè)計(jì):軟件UART的設(shè)計(jì)思想

作者: 時(shí)間:2017-09-18 來源:網(wǎng)絡(luò) 收藏

  目前擴(kuò)展串口的方法主要有以下方法, ①、采用串口擴(kuò)展芯片實(shí)現(xiàn),如ST16C550、ST16C554、SP2538、MAX3110等,雖然成本較高, 但系統(tǒng)的可靠性得到了保證,適用于數(shù)據(jù)量較大、串口需求較多的系統(tǒng);②、采用分時(shí)切換的方法將一個(gè)串口擴(kuò)展與多個(gè)串口設(shè)備通信,分時(shí)復(fù)用的方法成本低, 但只適用于數(shù)據(jù)量不大的場合, 并且只能由這個(gè)主動(dòng)和多個(gè)設(shè)備通信,實(shí)時(shí)性差;③、用軟件模擬的方法擴(kuò)展串口,其優(yōu)勢也是成本低、實(shí)時(shí)性好, 但要占用一些CPU時(shí)間。

本文引用地址:http://butianyuan.cn/article/201709/364418.htm

  一般的軟件模擬擴(kuò)展串口方法,使用1個(gè)I/O端口、1個(gè)INT外部中斷和定時(shí)器,該方法擴(kuò)展的串口有2個(gè)缺點(diǎn),①、由于使用了INT外部中斷,故只能使用2個(gè)INT外部中斷擴(kuò)展2個(gè)串口。②、文中的發(fā)送和接收數(shù)據(jù)的效率比較低,占用了CPU的大量時(shí)間,不能與其他任務(wù)同時(shí)進(jìn)行,所以使用范圍有限。

  本文提出的模擬串口方法,僅使用2個(gè)普通I/O和1個(gè)定時(shí)器,由于不需要INT的限制,可以擴(kuò)展出多個(gè)串口,且?guī)IFO的功能,該方法擴(kuò)展模擬串口的收發(fā)數(shù)據(jù)在中斷服務(wù)中完成,所以非常效率高,一般的都支持定時(shí)器中斷,所以所以該方法在大多數(shù)上都可以應(yīng)用。

  對于低速度的單片機(jī)(如89S51)可以擴(kuò)展出低速串口(9600、4800等),對于高速單片機(jī)(如AVR、PIC、C8051、STC12)可以擴(kuò)展高速串口(如19200、28800、38400、57600等)。目前單片機(jī)的處理速度越來越高,而價(jià)格越來越便宜,本文使用的STC12C1052芯片就具有高速度和低價(jià)格,價(jià)格僅為每片人民幣3.8元。電子產(chǎn)品的開發(fā)設(shè)計(jì)時(shí),要求在保證性能的情況下降低硬件成本,軟件模擬擴(kuò)展串口提供了一種降低成本的好方法。

  1、串口通訊原理

  在串口的異步通信中,數(shù)據(jù)以字節(jié)為單位的字節(jié)幀進(jìn)行傳送,發(fā)送端和接收端必須按照相同的字節(jié)幀格式和波特率進(jìn)行通信,其中字節(jié)幀格式規(guī)定了起始位、數(shù)據(jù)位、寄偶效驗(yàn)位、停止位。起始位是字節(jié)幀的開始,使數(shù)據(jù)線處于邏輯0狀態(tài),用于向接收端表明開始發(fā)送數(shù)據(jù)幀,起到使發(fā)送和接收設(shè)備實(shí)現(xiàn)同步。停止位是字節(jié)幀的終止,使數(shù)據(jù)線處于邏輯1狀態(tài),用于向接收端表明數(shù)據(jù)幀發(fā)送完畢。波特率采用標(biāo)準(zhǔn)速度,如4800、9600、19200、28800、38400、57600等。

  2、軟件的設(shè)計(jì)思想

  在本設(shè)計(jì)對硬件要求方面,僅僅占用單片機(jī)的任意2個(gè)I/O端口和1個(gè)定時(shí)器,利用定時(shí)器的定時(shí)中斷功能實(shí)現(xiàn)精確的波特率定時(shí),發(fā)送和接收都在定時(shí)中斷的控制之下進(jìn)行。

  數(shù)據(jù)發(fā)送的思想是,當(dāng)啟動(dòng)字節(jié)發(fā)送時(shí),通過TxD先發(fā)起始位,然后發(fā)數(shù)據(jù)位和奇偶數(shù)效驗(yàn)位,最后再發(fā)停止位,發(fā)送過程由發(fā)送狀態(tài)機(jī)控制,每次中斷只發(fā)送1個(gè)位,經(jīng)過若干個(gè)定時(shí)中斷完成1個(gè)字節(jié)幀的發(fā)送。

  數(shù)據(jù)接收的思想是,當(dāng)不在字節(jié)幀接收過程時(shí),每次定時(shí)中斷以3倍的波特率監(jiān)視RxD的狀態(tài),當(dāng)其連續(xù)3次采樣電平依次為1、0、0時(shí),就認(rèn)為檢測到了起始位,則開始啟動(dòng)一次字節(jié)幀接收,字節(jié)幀接收過程由接收狀態(tài)機(jī)控制,每次中斷只接收1個(gè)位,經(jīng)過若干個(gè)定時(shí)中斷完成1個(gè)字節(jié)幀的接收。

  為了提高串口的性能,在發(fā)送和接收上都實(shí)現(xiàn)了FIFO功能,提高通信的實(shí)時(shí)性。FIFO的長度可以進(jìn)行自由定義,適應(yīng)用戶的不同需要。

  波特率的計(jì)算按照計(jì)算公式進(jìn)行,在設(shè)置最高波特率時(shí)一定要考慮模擬串口程序代碼的執(zhí)行時(shí)間,該定時(shí)時(shí)間必須大于模擬串口的程序的規(guī)定時(shí)間。單片機(jī)的執(zhí)行速度越快,則可以實(shí)現(xiàn)更高的串口通訊速度。

  3、軟件設(shè)計(jì)的實(shí)現(xiàn)

  本程序在宏晶科技(深圳)生產(chǎn)的STC12C1052高速單片機(jī)上進(jìn)行運(yùn)行測試,STC12C1052單片機(jī)是單時(shí)鐘/機(jī)器周期的MCS51內(nèi)核單片機(jī),與89C2051引腳完全兼容,其工作頻率達(dá)35MHz,相當(dāng)與420MHz的89C2051單片機(jī),每片人民幣3.8元。由于該單片機(jī)的高速度,使得軟件擴(kuò)展串口的方法,更方便實(shí)現(xiàn)高速的串口。

  本擴(kuò)展串口的設(shè)計(jì)中,STC12C1052使用的晶振頻率為22.1184Mhz,以波特率的3倍計(jì)算定時(shí)時(shí)間,在接收過程中以此定時(shí)進(jìn)行接收起始位的采樣,在發(fā)送和接收過程中再3分頻得到標(biāo)準(zhǔn)波特率定時(shí),進(jìn)行數(shù)據(jù)發(fā)送與接收。

  3.1、數(shù)據(jù)定義

  定義模擬串口程序所必須的一些資源,如I/O引腳、波特率、數(shù)據(jù)緩沖區(qū)等。

  #define Fosc 22118400 //晶振頻率

  #define Baud 38400 //波特率

  #define BaudT (Fosc/Baud/3/12)

  #define BufLong 16 //FIFO長度

  sbit RxD1=P1^7; //模擬接收RxD

  sbit TxD1=P1^6; //模擬發(fā)送TxD

  bit Brxd1,Srxd1;//RxD檢測電平

  BYTE Rbuf1[BufLong];//FIFO接收區(qū)

  BYTE Rptr1,Rnum1;

  BYTE Tbuf1[BufLong];//FIFO發(fā)送區(qū)

  BYTE Tptr1,Tnum1;

  BYTE TimCnt1A,TimCnt1B;

  BYTE Mtbuf1,Mrbuf1,TxdCnt1,RxdCnt1;

  3.2、數(shù)據(jù)接收子程序

  數(shù)據(jù)接收過程中,依次存儲(chǔ)RxD的邏輯位形成字節(jié)數(shù)據(jù),當(dāng)數(shù)據(jù)接收完畢且停止位為1時(shí),表示接收到了有效數(shù)據(jù),就將結(jié)果存儲(chǔ)到接收FIFO隊(duì)列中去。

  void Recv()

  {

  if(RxdCnt1>0) //存數(shù)據(jù)位8個(gè)

  {

  Mrbuf1>>=1;

  if(RxD1==1) Mrbuf1=Mrbuf1|0x80;

  }

  RxdCnt1--;

  if(RxdCnt1==0&& RxD1==1) //數(shù)據(jù)接收完畢

  {

  Rbuf1[Rptr1]=Mrbuf1; //存儲(chǔ)到FIFO隊(duì)列

  if(++Rptr1>BufLong-1) Rptr1=0;

  if(++Rnum1>BufLong) Rnum1=BufLong;

  }

  }

  3.3、數(shù)據(jù)發(fā)送子程序

  該程序過程中,當(dāng)數(shù)據(jù)發(fā)送狀態(tài)結(jié)束時(shí),檢測發(fā)送FIFO隊(duì)列是否為空,若非空則取出發(fā)送數(shù)據(jù),然后啟動(dòng)發(fā)送狀態(tài);當(dāng)處于發(fā)送狀態(tài)時(shí),則按照狀態(tài)機(jī)的狀態(tài)進(jìn)行起始位、數(shù)據(jù)位和停止位的發(fā)送。

  void Send()

  {

  if(TxdCnt1!=0) //字節(jié)發(fā)送狀態(tài)機(jī)

  {

  if(TxdCnt1==11) TxD1=0;//發(fā)起始位0

  else if(TxdCnt1>2) //發(fā)數(shù)據(jù)位

  { Mtbuf1>>=1; TxD1=CY;}

  else TxD1=1; //發(fā)終止位1

  TxdCnt1--;

  }

  else if(Tnum1>0) //檢測FIFO隊(duì)列

  {

  Tnum1--;

  Mtbuf1=Tbuf1[Tptr1]; //讀取FIFO數(shù)據(jù)

  if(++Tptr1>=BufLong) Tptr1=0;

  TxdCnt1=11; //啟動(dòng)發(fā)送狀態(tài)機(jī)

  }

  }

  3.4、中斷程序

  中斷定時(shí)時(shí)間為波特率定時(shí)的1/3,即以3倍的波特率對RxD進(jìn)行采樣,實(shí)現(xiàn)起始位的判別,當(dāng)起始位到達(dá)時(shí)啟動(dòng)接收過程狀態(tài)機(jī)。將該定時(shí)進(jìn)行3分頻再調(diào)用數(shù)據(jù)的發(fā)送和接收過程,進(jìn)行準(zhǔn)確波特率下的串口通信。

  void Uart() interrupt 1 using 1

  {

  if(RxdCnt1==0 ) //接收起始識(shí)別

  {

  if(RxD1==0 && Brxd1==0 && Srxd1==1) { RxdCnt1=8; TimCnt1B=0;}

  }

  Srxd1=Brxd1; Brxd1=RxD1;

  if(++TimCnt1B>=3 && RxdCnt1!=0) { TimCnt1B=0; Recv();}//數(shù)據(jù)接收

  if(++TimCnt1A>=3) { TimCnt1A=0; Send();} //數(shù)據(jù)發(fā)送

  }

  3.5、串口初始化

  打開定時(shí)器的中斷,將定時(shí)器的設(shè)置為自裝載模式,依照波特率設(shè)置定時(shí)中斷的定時(shí)間隔,啟動(dòng)定時(shí)器,并進(jìn)行各變量的初始化。

  void IniUart()

  {

  IE="0x82"; TMOD="0x22";

  TH0=-BaudT; TL0=-BaudT; TR0=1;

  Rptr1=0;Rnum1=0;Tptr1=0;Tnum1=0;

  }

  4、結(jié)束語

  本文提出的模擬串口設(shè)計(jì)方法,其獨(dú)特之處在于:僅僅使用任意2個(gè)普通I/O引腳和1個(gè)定時(shí)中斷實(shí)現(xiàn)了全雙工串口,對硬件的占用較少,具有多可串口擴(kuò)展能力;在串口接收的起始位判別時(shí)采用了連續(xù)3次采樣的判別方法,該方法實(shí)現(xiàn)簡單、準(zhǔn)確率高;用定時(shí)中斷實(shí)現(xiàn)了串口數(shù)據(jù)的發(fā)送和接收,并實(shí)現(xiàn)了FIFO隊(duì)列,使串口發(fā)送和接收工作效率高。



關(guān)鍵詞: 單片機(jī) UART

評論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉