作者 | strongerHuang
微信公眾號 | 嵌入式專欄有一些初學者總覺得通信協(xié)議是一個很復雜的知識,把它想的很高深,導致不知道該怎么學。
同時,偶爾有讀者問關于串口自定義通信協(xié)議相關的問題,今天就來寫寫串口通信協(xié)議,并不是你想想中的那么難?
1什么通信協(xié)議?
通信協(xié)議不難理解,就是兩個(或多個)設備之間進行通信,必須要遵循的一種協(xié)議。百度百科的解釋:
通信協(xié)議是指雙方實體完成通信或服務所必須遵循的規(guī)則和約定。通過通信信道和設備互連起來的多個不同地理位置的數(shù)據(jù)通信系統(tǒng),要使其能協(xié)同工作實現(xiàn)信息交換和資源共享,它們之間必須具有共同的語言。交流什么、怎樣交流及何時交流,都必須遵循某種互相都能接受的規(guī)則。這個規(guī)則就是通信協(xié)議。相應該有很多讀者都買過一些基于串口通信的模塊,市面上很多基于串口通信的模塊都是自定義通信協(xié)議,有的比較簡單,有的相對復雜一點。
舉一個很簡單的串口通信協(xié)議的例子:比如只傳輸一個溫度值,只有三個字節(jié)的通信協(xié)議:
幀頭 | 溫度值 | 幀尾 |
---|---|---|
5A | 一字節(jié)數(shù)值 | 3B |
只是說這種通信協(xié)議應用的場合相對比較簡單(一對一兩個設備之間),同時,它存在很多弊端。
2過于簡單的通信協(xié)議引發(fā)的問題
上面那種只有三個字節(jié)的通信協(xié)議,相信大家都看明白了。雖然它也能通信,也能傳輸數(shù)據(jù),但它存在一系列的問題。比如:多個設備連接在一條總線(比如485)上,怎么判斷傳輸給誰?(沒有設備信息)
還比如:處于一個干擾環(huán)境,你能保障傳輸數(shù)據(jù)正確嗎?(沒有校驗信息)
再比如:我想傳輸多個不確定長度的數(shù)據(jù),該怎么辦?(沒有長度信息)。
上面這一系列問題,相信做過自定義通信的朋友都了解。
所以,在通信協(xié)議里面要約定更多的“協(xié)議信息”,這樣才能保證通信的完整。
3通信協(xié)議常見內(nèi)容
基于串口的通信協(xié)議通常不能太復雜,因為串口通信速率、抗干擾能力以及其他各方面原因,相對于TCP/IP這種通信協(xié)議,是一種很輕量級的通信協(xié)議。所以,基于串口的通信,除了一些通用的通信協(xié)議(比如:Modubs、MAVLink)之外,很多時候,工程師都會根據(jù)自己項目情況,自定義通信協(xié)議。
下面簡單描述下常見自定義通信協(xié)議的一些要點內(nèi)容。
(這是一些常見的協(xié)議內(nèi)容,可能不同情況,其協(xié)議內(nèi)容不同)
1.幀頭幀頭,就是一幀通信數(shù)據(jù)的開頭。有的通信協(xié)議幀頭只有一個,有的有兩個,比如:5A、A5作為幀頭。
這種情況,需要在協(xié)議或者附錄中要描述各種設備類型信息,方便開發(fā)者編碼查詢。
當然,有些固定的兩種設備之間通信,可能沒有這個選項。
3.命令/指令命令/指令比較常見,一般是不同的操作,用不同的命令來區(qū)分。
舉例:溫度:0x01;濕度:0x02;
4.命令類型/功能碼這個選項對命令進一步補充。比如:讀、寫操作。
舉例:讀Flash:0x01; 寫Flash:0x02;
5.數(shù)據(jù)長度數(shù)據(jù)長度這個選項,可能有的協(xié)議會把該選項提到前面設備地址位置,把命令這些信息算在“長度”里面。
這個主要是方便協(xié)議(接收)解析的時候,統(tǒng)計接收數(shù)據(jù)長度。
比如:有時候傳輸一個有效數(shù)據(jù),有時候要傳輸多個有效數(shù)據(jù),甚至傳輸一個數(shù)組的數(shù)據(jù)。這個時候,傳輸?shù)囊粠瑪?shù)據(jù)就是不定長數(shù)據(jù),就必須要有【數(shù)據(jù)長度】來約束。
有的長度是一個字節(jié),其范圍:0x01 ~ 0xFF,有的可能要求一次性傳輸更多,就用兩個字節(jié)表示,其范圍0x0001 ~ 0xFFFFF。
當然,有的通信長度是固定的長度(比如固定只傳輸、溫度、濕度這兩個數(shù)據(jù)),其協(xié)議可能沒有這個選項。
6.數(shù)據(jù)數(shù)據(jù)就不用描述了,就是你傳輸?shù)膶崒嵲谠诘臄?shù)據(jù),比如溫度:25℃。
7.幀尾有些協(xié)議可能沒有幀尾,這個應該是可有可無的一個選項。
8.校驗碼校驗碼是一個比較重要的內(nèi)容,一般正規(guī)一點的通信協(xié)議都有這個選項,原因很簡單,通信很容易受到干擾,或者其他原因,導致傳輸數(shù)據(jù)出錯。
如果有校驗碼,就能比較有效避免數(shù)據(jù)傳輸出錯的的情況。
校驗碼的方式有很多,校驗和、CRC校驗算是比較常見的,用于自定義協(xié)議中的校驗方式。
還有一點,有的協(xié)議可能把校驗碼放在倒數(shù)第二,幀尾放在最后位置。