有一些初學者總覺得通信協(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 |
2過于簡單的通信協(xié)議引發(fā)的問題
上面那種只有三個字節(jié)的通信協(xié)議,相信大家都看明白了。雖然它也能通信,也能傳輸數(shù)據(jù),但它存在一系列的問題。 比如:多個設備連接在一條總線(比如485)上,怎么判斷傳輸給誰?(沒有設備信息) 還比如:處于一個干擾環(huán)境,你能保障傳輸數(shù)據(jù)正確嗎?(沒有校驗信息) 再比如:我想傳輸多個不確定長度的數(shù)據(jù),該怎么辦?(沒有長度信息)。 上面這一系列問題,相信做過自定義通信的朋友都了解。 所以,在通信協(xié)議里面要約定更多的“協(xié)議信息”,這樣才能保證通信的完整。3通信協(xié)議常見內容
基于串口的通信協(xié)議通常不能太復雜,因為串口通信速率、抗干擾能力以及其他各方面原因,相對于TCP/IP這種通信協(xié)議,是一種很輕量級的通信協(xié)議。 所以,基于串口的通信,除了一些通用的通信協(xié)議(比如:Modubs、MAVLink)之外,很多時候,工程師都會根據(jù)自己項目情況,自定義通信協(xié)議。 下面簡單描述下常見自定義通信協(xié)議的一些要點內容。


2.設備地址/類型設備地址或者設備類型,通常是用于多種設備之間,為了方便區(qū)分不同設備。





4通信協(xié)議代碼實現(xiàn)
自定義通信協(xié)議,代碼實現(xiàn)的方式有很多種,怎么說呢,“條條大路通羅馬”你只需要按照你協(xié)議要寫實現(xiàn)代碼就行。 當然,實現(xiàn)的同時,需要考慮你項目實際情況,比如通信數(shù)據(jù)比較多,要用消息隊列(FIFO),還比如,如果協(xié)議復雜,最好封裝結構體等。 下面分享一些以前用到的代碼,可能沒有描述更多細節(jié),但一些思想可以借鑒。 1.消息數(shù)據(jù)發(fā)送a.通過串口直接發(fā)送每一個字節(jié)這種對于新手來說都能理解,這里分享一個之前DGUS串口屏的例子:b.通過消息隊列發(fā)送在上面基礎上,用一個buf裝下消息,然后“打包”到消息隊列,通過消息隊列的方式(FIFO)發(fā)送出去。/* DGUS寄存器地址 *///往DGDS屏指定寄存器寫一字節(jié)數(shù)據(jù)void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data){DGUS_SendByte(DGUS_FRAME_HEAD1);DGUS_SendByte(DGUS_FRAME_HEAD2);DGUS_SendByte(0x04);DGUS_SendByte(DGUS_CMD_W_REG); //指令DGUS_SendByte(RegAddr); //地址DGUS_SendByte((uint8_t)(Data>>8)); //數(shù)據(jù)DGUS_SendByte((uint8_t)(Data&0xFF));}//往DGDS屏指定地址寫一字節(jié)數(shù)據(jù)void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data){DGUS_SendByte(DGUS_FRAME_HEAD1);DGUS_SendByte(DGUS_FRAME_HEAD2);DGUS_SendByte(0x05);DGUS_SendByte(DGUS_CMD_W_DATA); //指令DGUS_SendByte((uint8_t)(DataAddr>>8)); //地址DGUS_SendByte((uint8_t)(DataAddr&0xFF));DGUS_SendByte((uint8_t)(Data>>8)); //數(shù)據(jù)DGUS_SendByte((uint8_t)(Data&0xFF));}
c.用“結構體”代替“數(shù)組SendBuf”方式結構體對數(shù)組更方便引用,也方便管理,所以,結構體方式相比數(shù)組buf更高級,也更實用。(當然,如果成員比較多,如果用臨時變量方式也會導致占用過多堆棧的情況) 比如:static uint8_t sDGUS_SendBuf[DGUS_PACKAGE_LEN];//往DGDS屏指定寄存器寫一字節(jié)數(shù)據(jù)void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data){sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1; //幀頭sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2;sDGUS_SendBuf[2] = 0x06; //長度sDGUS_SendBuf[3] = DGUS_CMD_W_CTRL; //指令sDGUS_SendBuf[4] = RegAddr; //地址sDGUS_SendBuf[5] = (uint8_t)(Data>>8); //數(shù)據(jù)sDGUS_SendBuf[6] = (uint8_t)(Data&0xFF);DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L);sDGUS_SendBuf[7] = sDGUS_CRC_H; //校驗sDGUS_SendBuf[8] = sDGUS_CRC_L;DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3);}//往DGDS屏指定地址寫一字節(jié)數(shù)據(jù)void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data){sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1; //幀頭sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2;sDGUS_SendBuf[2] = 0x07; //長度sDGUS_SendBuf[3] = DGUS_CMD_W_DATA; //指令sDGUS_SendBuf[4] = (uint8_t)(DataAddr>>8); //地址sDGUS_SendBuf[5] = (uint8_t)(DataAddr&0xFF);sDGUS_SendBuf[6] = (uint8_t)(Data>>8); //數(shù)據(jù)sDGUS_SendBuf[7] = (uint8_t)(Data&0xFF);DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L);sDGUS_SendBuf[8] = sDGUS_CRC_H; //校驗sDGUS_SendBuf[9] = sDGUS_CRC_L;DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3);}
d.其他更多串口發(fā)送數(shù)據(jù)的方式有很多,比如用DMA的方式替代消息隊列的方式。 2.消息數(shù)據(jù)接收串口消息接收,通常串口中斷接收的方式居多,當然,也有很少情況用輪詢的方式接收數(shù)據(jù)。 a.常規(guī)中斷接收還是以DGUS串口屏為例,描述一種簡單又常見的中斷接收方式:typedef struct{uint8_t Head1; //幀頭1uint8_t Head2; //幀頭2uint8_t Len; //長度uint8_t Cmd; //命令uint8_t Data[DGUS_DATA_LEN]; //數(shù)據(jù)uint16_t CRC16; //CRC校驗}DGUS_PACKAGE_TypeDef;
void DGUS_ISRHandler(uint8_t Data){static uint8_t sDgus_RxNum = 0; //數(shù)量static uint8_t sDgus_RxBuf[DGUS_PACKAGE_LEN];static portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;sDgus_RxBuf[gDGUS_RxCnt] = Data;gDGUS_RxCnt++;/* 判斷幀頭 */if(sDgus_RxBuf[0] != DGUS_FRAME_HEAD1) //接收到幀頭1{gDGUS_RxCnt = 0;return;}if((2 == gDGUS_RxCnt) && (sDgus_RxBuf[1] != DGUS_FRAME_HEAD2)){gDGUS_RxCnt = 0;return;}/* 確定一幀數(shù)據(jù)長度 */if(gDGUS_RxCnt == 3){sDgus_RxNum = sDgus_RxBuf[2] + 3;}/* 接收完一幀數(shù)據(jù) */if((6 <= gDGUS_RxCnt) && (sDgus_RxNum <= gDGUS_RxCnt)){gDGUS_RxCnt = 0;if(xDGUSRcvQueue != NULL) //解析成功, 加入隊列{xQueueSendFromISR(xDGUSRcvQueue, &sDgus_RxBuf[0], &xHigherPriorityTaskWoken);portEND_SWITCHING_ISR(xHigherPriorityTaskWoken);}}}
b.增加超時檢測
接收數(shù)據(jù)有可能存在接收了一半,中斷因為某種原因中斷了,這時候,超時檢測也很有必要。
比如:用多余的MCU定時器做一個超時計數(shù)的處理,接收到一個數(shù)據(jù),開始計時,超過1ms沒有接收到下一個數(shù)據(jù),就丟掉這一包(前面接收的)數(shù)據(jù)。
static void DGUS_TimingAndUpdate(uint16_t Nms){sDGUSTiming_Nms_Num = Nms;TIM_SetCounter(DGUS_TIM, 0); //設置計數(shù)值為0TIM_Cmd(DGUS_TIM, ENABLE); //啟動定時器}void DGUS_COM_IRQHandler(void){if((DGUS_COM->SR & USART_FLAG_RXNE) == USART_FLAG_RXNE){DGUS_TimingAndUpdate(5); //更新定時(防止超時)DGUS_ISRHandler((uint8_t)USART_ReceiveData(DGUS_COM));}}
c.更多
接收和發(fā)送一樣,實現(xiàn)方法有很多種,比如接收同樣也可以用結構體方式。但有一點,都需要結合你實際需求來編碼。
5最后
以上自定義協(xié)議內容僅供參考,最終用哪些、占用幾個字節(jié)都與你實際需求有關。 基于串口的自定義通信協(xié)議,有千差萬別,比如:MCU處理能力、設備多少、通信內容等都與你自定義協(xié)議有關。 有的可能只需要很簡單的通信協(xié)議就能滿足要求。有的可能需要更復雜的協(xié)議才能滿足。 最后強調兩點:1.以上舉例并不是完整的代碼(有些細節(jié)沒有描述出來),主要是供大家學習這種編程思想,或者實現(xiàn)方式。 2.一份好的通信協(xié)議代碼,必定有一定容錯處理,比如:發(fā)送完成檢測、接收超時檢測、數(shù)據(jù)出錯檢測等等。所以說,以上代碼并不是完整的代碼。原文標題:通信教程 | 自定義串口通信協(xié)議
文章出處:【微信公眾號:strongerHuang】歡迎添加關注!文章轉載請注明出處。
審核編輯:彭菁
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。
舉報投訴
-
通信協(xié)議
+關注
關注
28文章
1092瀏覽量
42158 -
數(shù)據(jù)
+關注
關注
8文章
7335瀏覽量
94767 -
串口通信
+關注
關注
34文章
1662瀏覽量
57971
原文標題:通信教程 | 自定義串口通信協(xié)議
文章出處:【微信號:strongerHuang,微信公眾號:strongerHuang】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
相關推薦
熱點推薦
汽車通信協(xié)議資料總結
1.常見的汽車通信協(xié)議總結。包括can通信, 診斷協(xié)議uds . LIN通信 ,j1939, ISO14229, ISO15675,OBD
發(fā)表于 09-06 20:39
USS通信協(xié)議的基本內容
USS通信技術作為一種低成本的簡單驅動控制技術,在工業(yè)現(xiàn)場有著廣泛的應用。今天這篇文章,我們就和大家一起聊聊USS通信協(xié)議的基本內容。
發(fā)表于 01-19 06:45
物聯(lián)網(wǎng)常見通信協(xié)議 精選資料分享
1概述在上一篇文章《物聯(lián)網(wǎng)常見通信協(xié)議與通訊協(xié)議梳理【上】-通訊協(xié)議》中,對物聯(lián)網(wǎng)常用通信協(xié)議和通訊協(xié)議
發(fā)表于 07-30 06:02
常見的物聯(lián)網(wǎng)通信協(xié)議藍牙簡單對比
@TOC淺析物聯(lián)網(wǎng)(智能家居)無線通信協(xié)議物聯(lián)網(wǎng)無線傳輸方案產(chǎn)品開發(fā),通信協(xié)議(生態(tài))選擇至關重要,簡單對比一下常見的物聯(lián)網(wǎng)通信協(xié)議藍牙(Bluetooth), Wi-Fi,Zigbe
發(fā)表于 01-11 07:24
常見的無線通信協(xié)議有哪些
隨著物聯(lián)網(wǎng)技術的愈發(fā)成熟,適用于智能家居無線通信協(xié)議的種類也日益增多。目前,最常見的無線通信協(xié)議有WiFi、藍牙、Zigbee、Z-wave、RF等,前三者在智能家居應用中更為廣泛。W
發(fā)表于 02-15 06:54
一個簡單的基礎通信協(xié)議的設計與實現(xiàn)
之間的數(shù)據(jù)的穩(wěn)定傳輸,通信協(xié)議的設計需要考慮很多的問題。當然應對不同的應用場景,可以有針對性的設計不同的通信協(xié)議。一種常見的通信協(xié)議格式這是
發(fā)表于 12-14 18:38
?10次下載
通信協(xié)議內容與功能
通信協(xié)議不難理解,就是兩個(或多個)設備之間進行通信,必須要遵循的一種協(xié)議。通信協(xié)議是指雙方實體完成通信或服務所必須遵循的規(guī)則和約定。通過
發(fā)表于 05-06 14:43
?4146次閱讀
通信協(xié)議的作用、類型及優(yōu)缺點
通信協(xié)議是指在計算機網(wǎng)絡中,為了使不同的計算機或網(wǎng)絡設備之間能夠相互通信而制定的一套規(guī)則和標準。通信協(xié)議規(guī)定了數(shù)據(jù)傳輸?shù)母袷健鬏斔俾?、傳輸控制、錯誤檢測和糾正等方面的內容,以確保
發(fā)表于 05-10 16:09
?6991次閱讀
簡單認識UART通信協(xié)議
模塊等多種設備之間的數(shù)據(jù)傳輸。以下是對UART通訊協(xié)議的詳細簡述,內容將涵蓋其基本原理、工作方式、配置參數(shù)、常見應用以及與其他通信協(xié)議的比較。
常見串口通信協(xié)議 如何設置串口參數(shù)
串口通信是一種常見的通信方式,廣泛應用于計算機、嵌入式系統(tǒng)和各種電子設備之間。串口通信協(xié)議主要是指在串行通信中,數(shù)據(jù)傳輸?shù)母袷胶鸵?guī)則。
通信協(xié)議常見內容有哪些
評論