嵌入式系統(tǒng)上的異步串口通信的實(shí)現(xiàn)
在嵌入式系統(tǒng)中,異步串口(UART)使用非常頻繁,可以用于與各種外部系統(tǒng)(幀括PC)之間的通信。在硬件上UART通過(guò)在每個(gè)字節(jié)的傳輸中插入開(kāi)始位和停止位,保證接收端可以正確地找到字節(jié)的開(kāi)始和結(jié)束,同時(shí)也可以通過(guò)插入奇偶校驗(yàn)位,讓接收端檢驗(yàn)收到的字節(jié)是否正確。而且,由于有開(kāi)始位和停止位的存在,使得字節(jié)之間可以插入任意的空閑位(與停止位同為高電平),而不影響下一個(gè)字節(jié)的正常傳輸。因此,UART硬件保證了每個(gè)字節(jié)的正確傳輸,并可以有效檢出字節(jié)傳輸?shù)腻e(cuò)誤。但并不保證一串字節(jié)的正確傳輸,這需要軟件來(lái)完成。
本文引用地址:http://www.butianyuan.cn/article/201711/370982.htm從軟件的角度來(lái)看,所有的通信都是一串字節(jié)(叫做數(shù)據(jù)幀)的連續(xù)傳輸。軟件需要采用適當(dāng)?shù)臋C(jī)制來(lái)保證接收端能夠正確識(shí)別出一個(gè)完整的數(shù)據(jù)幀、能夠檢查接收到的數(shù)據(jù)幀是正確的、在傳輸發(fā)生錯(cuò)誤時(shí)有合適的恢復(fù)機(jī)制。為此就需要定義一個(gè)合適的數(shù)據(jù)幀格式。
數(shù)據(jù)幀的提取
為了識(shí)別出一個(gè)完整的數(shù)據(jù)幀,基本上有兩個(gè)機(jī)制:一是在軟件上規(guī)定字節(jié)之間的間隔最大值,一旦兩個(gè)字節(jié)間的間隔超過(guò)某個(gè)閾值,就認(rèn)為一個(gè)數(shù)據(jù)幀結(jié)束;另一種機(jī)制不對(duì)字節(jié)間的間隔作規(guī)定,而是用特殊的字節(jié)來(lái)定義數(shù)據(jù)報(bào)的開(kāi)始和結(jié)束,當(dāng)收到該特殊字節(jié)時(shí),就認(rèn)為一個(gè)數(shù)據(jù)幀的傳輸已完成。
采用第一種機(jī)制的,比如Modbus-RTU。就是規(guī)定了同一個(gè)數(shù)據(jù)幀的字節(jié)間隔不能大于1.5個(gè)字節(jié)的傳輸時(shí)間,一旦大于該間隔,則認(rèn)為前一個(gè)幀的傳輸已經(jīng)結(jié)束,或者出錯(cuò)。同時(shí)為了保證不同數(shù)據(jù)幀之間有足夠的間隔,還規(guī)定了兩個(gè)數(shù)據(jù)幀之間最少插入3.5個(gè)字節(jié)的空閑位。下圖摘自Modbus協(xié)議,表示了一個(gè)正確的數(shù)據(jù)幀、錯(cuò)誤的數(shù)據(jù)幀以及數(shù)據(jù)幀之間的間隔。同時(shí)在數(shù)據(jù)報(bào)中引入了CRC,用于檢出數(shù)據(jù)幀層面的傳輸錯(cuò)誤。
第二種機(jī)制需要選用一個(gè)特殊字節(jié)作為幀頭幀尾(也可以給幀頭幀尾選用不同的字節(jié)),比如說(shuō),選用0xFF作為幀頭幀尾字節(jié),也即發(fā)送端在每個(gè)數(shù)據(jù)幀的頭尾都插入至少一個(gè)0xFF,接收端收到該字節(jié)就結(jié)束上一個(gè)幀的接收,同時(shí)從下一個(gè)非0xFF字節(jié)開(kāi)始一個(gè)新的數(shù)據(jù)幀的接收。但該機(jī)制有一個(gè)問(wèn)題,那就是所選的特殊字節(jié)(此處是0xFF)也可能是數(shù)據(jù)幀中的一個(gè)有效字節(jié),如果不作特殊處理,接收端就可能把這個(gè)數(shù)據(jù)幀中的字節(jié)誤認(rèn)為是數(shù)據(jù)幀的結(jié)束標(biāo)識(shí),從而導(dǎo)致接收錯(cuò)誤。因此,此處采取字節(jié)替換的特殊處理,以消除在數(shù)據(jù)幀的中間出現(xiàn)的特殊字節(jié),以下是一個(gè)例子。
在此處,選0xFE作為字節(jié)替換的標(biāo)識(shí)字節(jié),將0xFF替換成0xFE 0x0F。由于0xFE是字節(jié)替換的標(biāo)志,也成了一個(gè)特殊字符,同樣需要進(jìn)行替換。在上圖中,將0xFE替換成0xFE 0x0E。
采用該機(jī)制時(shí),發(fā)送端對(duì)數(shù)據(jù)幀中的所有0xFF,0xFE進(jìn)行替換。在接收端則進(jìn)行反向替換。如果在接收端的0xFE后面的字節(jié)收到非0x0F,0x0E的字符,說(shuō)明發(fā)生了傳輸錯(cuò)誤,丟棄已經(jīng)接收的內(nèi)容,重新搜索幀頭字符開(kāi)始一個(gè)新數(shù)據(jù)幀的接收。
數(shù)據(jù)幀的校驗(yàn)
考慮到串口通信有一定的誤碼率,無(wú)論是采用那種傳輸方式,必須有錯(cuò)誤幀檢出機(jī)制。通常選用的是CRC。一般用2個(gè)字節(jié)或4個(gè)字節(jié)。
串口驅(qū)動(dòng)的實(shí)現(xiàn)
在嵌入式系統(tǒng)中,串口驅(qū)動(dòng)的實(shí)現(xiàn)一般有兩個(gè)途徑:一個(gè)是通過(guò)串口的收發(fā)中斷與串口控制器的Buffer直接交換數(shù)據(jù);另一種是通過(guò)DMA,利用DMA與串口Buffer交互自動(dòng)收發(fā)數(shù)據(jù)。
與串口控制器直接交互的方式,實(shí)現(xiàn)簡(jiǎn)單,既可以實(shí)時(shí)進(jìn)行字節(jié)替換以節(jié)省內(nèi)存,也可以方便設(shè)置字節(jié)間的定時(shí)器。但頻繁打斷CPU,對(duì)有一定BUFFER大小的串口,還可以利用BUFFER進(jìn)行性能優(yōu)化,而對(duì)一些只有一個(gè)字節(jié)或兩個(gè)字節(jié)BUFFER的串口,就需要一個(gè)字節(jié)起一次中斷,如果CPU負(fù)荷較重,很容易由于中斷處理不及時(shí)而導(dǎo)致字節(jié)丟失,特別是在高波特率的情況。
利用DMA與串口控制器交互的方式,由于不能實(shí)時(shí)進(jìn)行字節(jié)替換,需要在放入DMA之前完成字節(jié)替換,導(dǎo)致占用較大內(nèi)存空間。另外對(duì)于字節(jié)間間隔敏感的傳輸策略,需要用另外的機(jī)制,比如PRS,去操作定時(shí)器。用DMA的另一個(gè)問(wèn)題是在接收時(shí)無(wú)法實(shí)時(shí)確定幀尾(采用幀頭幀尾模式時(shí)),一般只能采用定時(shí)器與DMA配合,用字節(jié)流的方式處理接收。
評(píng)論