国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

自定義串口通信協議,如何實現?

網絡 網絡管理
通信協議是指雙方實體完成通信或服務所必須遵循的規則和約定。通過通信信道和設備互連起來的多個不同地理位置的數據通信系統,要使其能協同工作實現信息交換和資源共享,它們之間必須具有共同的語言。交流什么、怎樣交流及何時交流,都必須遵循某種互相都能接受的規則。這個規則就是通信協議。

[[402368]]

本文轉載自微信公眾號「strongerHuang」,作者strongerHuang。轉載本文請聯系strongerHuang公眾號。

有一些初學者總覺得通信協議是一個很復雜的知識,把它想的很高深,導致不知道該怎么學。

同時,偶爾有讀者問關于串口自定義通信協議相關的問題,今天就來寫寫串口通信協議,并不是你想想中的那么難?

一.什么通信協議?

通信協議不難理解,就是兩個(或多個)設備之間進行通信,必須要遵循的一種協議。

百度百科的解釋:

通信協議是指雙方實體完成通信或服務所必須遵循的規則和約定。通過通信信道和設備互連起來的多個不同地理位置的數據通信系統,要使其能協同工作實現信息交換和資源共享,它們之間必須具有共同的語言。交流什么、怎樣交流及何時交流,都必須遵循某種互相都能接受的規則。這個規則就是通信協議。

相應該有很多讀者都買過一些基于串口通信的模塊,市面上很多基于串口通信的模塊都是自定義通信協議,有的比較簡單,有的相對復雜一點。

舉一個很簡單的串口通信協議的例子:比如只傳輸一個溫度值,只有三個字節的通信協議:

幀頭 溫度值 幀尾
5A 一字節數值 3B

這種看起來是不是很簡單?它也是一種通信協議。

只是說這種通信協議應用的場合相對比較簡單(一對一兩個設備之間),同時,它存在很多弊端。

二.過于簡單的通信協議引發的問題

上面那種只有三個字節的通信協議,相信大家都看明白了。雖然它也能通信,也能傳輸數據,但它存在一系列的問題。

比如:多個設備連接在一條總線(比如485)上,怎么判斷傳輸給誰?(沒有設備信息)

還比如:處于一個干擾環境,你能保障傳輸數據正確嗎?(沒有校驗信息)

再比如:我想傳輸多個不確定長度的數據,該怎么辦?(沒有長度信息)。

上面這一系列問題,相信做過自定義通信的朋友都了解。

所以,在通信協議里面要約定更多的“協議信息”,這樣才能保證通信的完整。

三.通信協議常見內容

基于串口的通信協議通常不能太復雜,因為串口通信速率、抗干擾能力以及其他各方面原因,相對于TCP/IP這種通信協議,是一種很輕量級的通信協議。

所以,基于串口的通信,除了一些通用的通信協議(比如:Modubs、MAVLink)之外,很多時候,工程師都會根據自己項目情況,自定義通信協議。

下面簡單描述下常見自定義通信協議的一些要點內容。

(這是一些常見的協議內容,可能不同情況,其協議內容不同)

1.幀頭

幀頭,就是一幀通信數據的開頭。

有的通信協議幀頭只有一個,有的有兩個,比如:5A、A5作為幀頭。

2.設備地址/類型

設備地址或者設備類型,通常是用于多種設備之間,為了方便區分不同設備。

這種情況,需要在協議或者附錄中要描述各種設備類型信息,方便開發者編碼查詢。

當然,有些固定的兩種設備之間通信,可能沒有這個選項。

3.命令/指令

命令/指令比較常見,一般是不同的操作,用不同的命令來區分。

舉例:溫度:0x01;濕度:0x02;

4.命令類型/功能碼

這個選項對命令進一步補充。比如:讀、寫操作。

舉例:讀Flash:0x01; 寫Flash:0x02;

5.數據長度

數據長度這個選項,可能有的協議會把該選項提到前面設備地址位置,把命令這些信息算在“長度”里面。

這個主要是方便協議(接收)解析的時候,統計接收數據長度。

比如:有時候傳輸一個有效數據,有時候要傳輸多個有效數據,甚至傳輸一個數組的數據。這個時候,傳輸的一幀數據就是不定長數據,就必須要有【數據長度】來約束。

有的長度是一個字節,其范圍:0x01 ~ 0xFF,有的可能要求一次性傳輸更多,就用兩個字節表示,其范圍0x0001 ~ 0xFFFFF。

當然,有的通信長度是固定的長度(比如固定只傳輸、溫度、濕度這兩個數據),其協議可能沒有這個選項。

6.數據

數據就不用描述了,就是你傳輸的實實在在的數據,比如溫度:25℃。

7.幀尾

有些協議可能沒有幀尾,這個應該是可有可無的一個選項。

8.校驗碼

校驗碼是一個比較重要的內容,一般正規一點的通信協議都有這個選項,原因很簡單,通信很容易受到干擾,或者其他原因,導致傳輸數據出錯。

如果有校驗碼,就能比較有效避免數據傳輸出錯的的情況。

校驗碼的方式有很多,校驗和、CRC校驗算是比較常見的,用于自定義協議中的校驗方式。

還有一點,有的協議可能把校驗碼放在倒數第二,幀尾放在最后位置。

四.通信協議代碼實現

自定義通信協議,代碼實現的方式有很多種,怎么說呢,“條條大路通羅馬”你只需要按照你協議要寫實現代碼就行。

當然,實現的同時,需要考慮你項目實際情況,比如通信數據比較多,要用消息隊列(FIFO),還比如,如果協議復雜,最好封裝結構體等。

下面分享一些以前用到的代碼,可能沒有描述更多細節,但一些思想可以借鑒。

1.消息數據發送

a.通過串口直接發送每一個字節

這種對于新手來說都能理解,這里分享一個之前DGUS串口屏的例子:

  1. #define DGUS_FRAME_HEAD1          0xA5                     //DGUS屏幀頭1 
  2. #define DGUS_FRAME_HEAD2          0x5A                     //DGUS屏幀頭2 
  3.  
  4. #define DGUS_CMD_W_REG            0x80                     //DGUS寫寄存器指令 
  5. #define DGUS_CMD_R_REG            0x81                     //DGUS讀寄存器指令 
  6. #define DGUS_CMD_W_DATA           0x82                     //DGUS寫數據指令 
  7. #define DGUS_CMD_R_DATA           0x83                     //DGUS讀數據指令 
  8. #define DGUS_CMD_W_CURVE          0x85                     //DGUS寫曲線指令 
  9.  
  10. /* DGUS寄存器地址 */ 
  11. #define DGUS_REG_VERSION          0x00                     //DGUS版本 
  12. #define DGUS_REG_LED_NOW          0x01                     //LED背光亮度 
  13. #define DGUS_REG_BZ_TIME          0x02                     //蜂鳴器時長 
  14. #define DGUS_REG_PIC_ID           0x03                     //顯示頁面ID 
  15. #define DGUS_REG_TP_FLAG          0x05                     //觸摸坐標更新標志 
  16. #define DGUS_REG_TP_STATUS        0x06                     //坐標狀態 
  17. #define DGUS_REG_TP_POSITION      0x07                     //坐標位置 
  18. #define DGUS_REG_TPC_ENABLE       0x0B                     //觸控使能 
  19. #define DGUS_REG_RTC_NOW          0x20                     //當前RTCS 
  20.  
  21. //往DGDS屏指定寄存器寫一字節數據 
  22. void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data) 
  23.   DGUS_SendByte(DGUS_FRAME_HEAD1); 
  24.   DGUS_SendByte(DGUS_FRAME_HEAD2); 
  25.   DGUS_SendByte(0x04); 
  26.  
  27.   DGUS_SendByte(DGUS_CMD_W_REG);                 //指令 
  28.   DGUS_SendByte(RegAddr);                        //地址 
  29.  
  30.   DGUS_SendByte((uint8_t)(Data>>8));             //數據 
  31.   DGUS_SendByte((uint8_t)(Data&0xFF)); 
  32.  
  33. //往DGDS屏指定地址寫一字節數據 
  34. void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data) 
  35.   DGUS_SendByte(DGUS_FRAME_HEAD1); 
  36.   DGUS_SendByte(DGUS_FRAME_HEAD2); 
  37.   DGUS_SendByte(0x05); 
  38.  
  39.   DGUS_SendByte(DGUS_CMD_W_DATA);                //指令 
  40.  
  41.   DGUS_SendByte((uint8_t)(DataAddr>>8));         //地址 
  42.   DGUS_SendByte((uint8_t)(DataAddr&0xFF)); 
  43.  
  44.   DGUS_SendByte((uint8_t)(Data>>8));             //數據 
  45.   DGUS_SendByte((uint8_t)(Data&0xFF)); 

b.通過消息隊列發送

在上面基礎上,用一個buf裝下消息,然后“打包”到消息隊列,通過消息隊列的方式(FIFO)發送出去。

  1. static uint8_t  sDGUS_SendBuf[DGUS_PACKAGE_LEN]; 
  2.  
  3. //往DGDS屏指定寄存器寫一字節數據 
  4. void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data) 
  5.   sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1;           //幀頭 
  6.   sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2; 
  7.   sDGUS_SendBuf[2] = 0x06;                       //長度 
  8.   sDGUS_SendBuf[3] = DGUS_CMD_W_CTRL;            //指令 
  9.   sDGUS_SendBuf[4] = RegAddr;                    //地址 
  10.   sDGUS_SendBuf[5] = (uint8_t)(Data>>8);         //數據 
  11.   sDGUS_SendBuf[6] = (uint8_t)(Data&0xFF); 
  12.  
  13.   DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L); 
  14.   sDGUS_SendBuf[7] = sDGUS_CRC_H;                //校驗 
  15.   sDGUS_SendBuf[8] = sDGUS_CRC_L; 
  16.  
  17.   DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3); 
  18.  
  19. //往DGDS屏指定地址寫一字節數據 
  20. void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data) 
  21.   sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1;           //幀頭 
  22.   sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2; 
  23.   sDGUS_SendBuf[2] = 0x07;                       //長度 
  24.   sDGUS_SendBuf[3] = DGUS_CMD_W_DATA;            //指令 
  25.   sDGUS_SendBuf[4] = (uint8_t)(DataAddr>>8);     //地址 
  26.   sDGUS_SendBuf[5] = (uint8_t)(DataAddr&0xFF); 
  27.   sDGUS_SendBuf[6] = (uint8_t)(Data>>8);         //數據 
  28.   sDGUS_SendBuf[7] = (uint8_t)(Data&0xFF); 
  29.  
  30.   DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L); 
  31.   sDGUS_SendBuf[8] = sDGUS_CRC_H;                //校驗 
  32.   sDGUS_SendBuf[9] = sDGUS_CRC_L; 
  33.  
  34.   DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3); 

c.用“結構體”代替“數組SendBuf”方式

結構體對數組更方便引用,也方便管理,所以,結構體方式相比數組buf更高級,也更實用。(當然,如果成員比較多,如果用臨時變量方式也會導致占用過多堆棧的情況)

比如:

  1. typedef struct 
  2.   uint8_t  Head1;                 //幀頭1 
  3.   uint8_t  Head2;                 //幀頭2 
  4.   uint8_t  Len;                   //長度 
  5.   uint8_t  Cmd;                   //命令 
  6.   uint8_t  Data[DGUS_DATA_LEN];   //數據 
  7.   uint16_t CRC16;                 //CRC校驗 
  8. }DGUS_PACKAGE_TypeDef; 

d.其他更多

串口發送數據的方式有很多,比如用DMA的方式替代消息隊列的方式。

2.消息數據接收

串口消息接收,通常串口中斷接收的方式居多,當然,也有很少情況用輪詢的方式接收數據。

a.常規中斷接收

還是以DGUS串口屏為例,描述一種簡單又常見的中斷接收方式:

  1. void DGUS_ISRHandler(uint8_t Data) 
  2.   static uint8_t sDgus_RxNum = 0;                //數量 
  3.   static uint8_t sDgus_RxBuf[DGUS_PACKAGE_LEN]; 
  4.   static portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE; 
  5.  
  6.   sDgus_RxBuf[gDGUS_RxCnt] = Data; 
  7.   gDGUS_RxCnt++; 
  8.  
  9.   /* 判斷幀頭 */ 
  10.   if(sDgus_RxBuf[0] != DGUS_FRAME_HEAD1)       //接收到幀頭1 
  11.   { 
  12.     gDGUS_RxCnt = 0; 
  13.     return
  14.   } 
  15.   if((2 == gDGUS_RxCnt) && (sDgus_RxBuf[1] != DGUS_FRAME_HEAD2)) 
  16.   { 
  17.     gDGUS_RxCnt = 0; 
  18.     return
  19.   } 
  20.  
  21.   /* 確定一幀數據長度 */ 
  22.   if(gDGUS_RxCnt == 3) 
  23.   { 
  24.     sDgus_RxNum = sDgus_RxBuf[2] + 3; 
  25.   } 
  26.  
  27.   /* 接收完一幀數據 */ 
  28.   if((6 <= gDGUS_RxCnt) && (sDgus_RxNum <= gDGUS_RxCnt)) 
  29.   { 
  30.     gDGUS_RxCnt = 0; 
  31.  
  32.     if(xDGUSRcvQueue != NULL)                    //解析成功, 加入隊列 
  33.     { 
  34.       xQueueSendFromISR(xDGUSRcvQueue, &sDgus_RxBuf[0], &xHigherPriorityTaskWoken); 
  35.       portEND_SWITCHING_ISR(xHigherPriorityTaskWoken); 
  36.     } 
  37.   } 

b.增加超時檢測

接收數據有可能存在接收了一半,中斷因為某種原因中斷了,這時候,超時檢測也很有必要。

比如:用多余的MCU定時器做一個超時計數的處理,接收到一個數據,開始計時,超過1ms沒有接收到下一個數據,就丟掉這一包(前面接收的)數據。

  1. static void DGUS_TimingAndUpdate(uint16_t Nms) 
  2.   sDGUSTiming_Nms_Num = Nms; 
  3.   TIM_SetCounter(DGUS_TIM, 0);                   //設置計數值為0 
  4.   TIM_Cmd(DGUS_TIM, ENABLE);                     //啟動定時器 
  5.  
  6. void DGUS_COM_IRQHandler(void) 
  7.   if((DGUS_COM->SR & USART_FLAG_RXNE) == USART_FLAG_RXNE) 
  8.   { 
  9.     DGUS_TimingAndUpdate(5);                     //更新定時(防止超時) 
  10.     DGUS_ISRHandler((uint8_t)USART_ReceiveData(DGUS_COM)); 
  11.   } 

c.更多

接收和發送一樣,實現方法有很多種,比如接收同樣也可以用結構體方式。但有一點,都需要結合你實際需求來編碼。

五.最后

以上自定義協議內容僅供參考,最終用哪些、占用幾個字節都與你實際需求有關。

基于串口的自定義通信協議,有千差萬別,比如:MCU處理能力、設備多少、通信內容等都與你自定義協議有關。

有的可能只需要很簡單的通信協議就能滿足要求。有的可能需要更復雜的協議才能滿足。

最后強調兩點:

1.以上舉例并不是完整的代碼(有些細節沒有描述出來),主要是供大家學習這種編程思想,或者實現方式。

2.一份好的通信協議代碼,必定有一定容錯處理,比如:發送完成檢測、接收超時檢測、數據出錯檢測等等。所以說,以上代碼并不是完整的代碼。

 

責任編輯:武曉燕 來源: strongerHuang
相關推薦

2021-03-04 09:50:23

Redis網絡通信Redis服務器

2023-10-12 19:37:50

通信協議HTTP

2022-12-02 14:42:37

2010-06-11 14:31:08

通信協議

2009-09-07 22:00:15

LINQ自定義

2010-07-06 17:14:03

網關通信協議

2019-05-27 06:05:20

物聯網協議物聯網IOT

2010-06-11 14:25:08

通信協議

2010-06-25 14:43:46

通信協議

2023-01-03 07:40:27

自定義滑塊組件

2009-09-03 13:34:03

.NET自定義控件

2010-06-09 10:43:54

廣義網協議

2019-08-23 12:49:18

USB通信協議

2009-12-22 09:37:47

網關設置通信協議

2019-04-29 10:26:49

TCP網絡協議網絡通信

2013-01-10 09:36:19

NagiosNagios插件

2024-02-20 19:53:57

網絡通信協議

2020-02-20 22:44:01

通信協議物聯網終端設備

2023-04-27 17:49:38

物聯網通信協議

2024-01-23 12:47:27

點贊
收藏

51CTO技術棧公眾號

精品av一区二区| 天堂影视av| aa亚洲一区一区三区| 亚洲国产日韩精品| 无码 制服 丝袜 国产 另类| 久久精品凹凸全集| 精品美女一区二区| 亚洲夫妻av| 97aⅴ精品视频一二三区| 国产精品v欧美精品v日韩| 日韩毛片网站| 精品少妇一区二区三区视频免付费 | 国产精品女主播| 成人a在线观看高清电影| 欧美视频一二三区| 国产污污在线观看| ●精品国产综合乱码久久久久 | 欧美一区二区三区电影在线观看| 青青一区二区| 色婷婷综合久久久久| 91福利区在线观看| 91精品欧美久久久久久动漫| 久久白虎精品| 亚洲欧洲综合另类| 成人拍拍拍在线观看| 久久夜色精品国产噜噜av| 四虎一区二区| 国产手机视频一区二区| 国产精品传媒毛片三区| 9999国产精品| 国产激情999| 偷拍亚洲色图| 欧美一级成年大片在线观看| 伊色综合久久之综合久久| 日韩视频在线观看免费| 免费成人动漫| 一区二区三区视频免费| 天天综合网站| 日韩视频―中文字幕| 日韩国产大片| 成年人精品视频| 看片一区二区| 搡老女人一区二区三区视频tv| 国产精品99| 久久久国产一区二区| 亚洲开心激情| 久久频这里精品99香蕉| 成人高潮视频| 国产成人久久精品| 国产精品久久久久久久久妇女| 亚洲一区二区三区乱码aⅴ蜜桃女| 久久精品影视| 免费看国产精品一二区视频| 水野朝阳av一区二区三区| 欧美在线日韩精品| 国产精品综合久久| 日本www在线视频| 国产精品沙发午睡系列990531| 妞干网在线视频| 一二三区精品福利视频| 青青草免费观看免费视频在线| 欧美亚洲精品一区| 性感女国产在线| 久久久91精品国产| 国产探花在线精品一区二区| 99国产超薄肉色丝袜交足的后果| 亚洲深夜激情| 成人国产在线看| 久久免费国产精品 | 蜜桃久久久久久久| 无码av天堂一区二区三区| 国产精品视频线看| 日中文字幕在线| 欧美成人女星排名| 日韩精品一区二区三区中文字幕 | 91国内精品| 成人国产亚洲精品a区天堂华泰| 狠狠色丁香久久综合频道| 蜜桃成人免费视频| 国产.精品.日韩.另类.中文.在线.播放 | 九色在线播放| 亚洲大胆美女视频| 国产精品3区| 成人网在线免费观看| 蜜桃av噜噜一区二区三区小说| 99精品视频在线看| 在线精品视频一区二区| 992tv国产精品成人影院| 国产精品扒开腿做爽爽爽的视频| 亚洲激情精品| 欧美日韩一道本| 色婷婷综合久久久久中文一区二区 | 中文字幕在线观看第三页| 黑人巨大精品欧美一区免费视频| 欧美wwww| 日韩av片免费在线观看| 久久在线91| 人人在线97| 日韩av在线一区二区| 成人精品影院| 免费视频爱爱太爽了| 欧美日韩精品在线| 99精品在免费线偷拍| 国产精品久久九九| 久久久久久久综合狠狠综合| 色开心亚洲综合| 91av在线国产| 国产高清在线观看免费不卡| 亚洲美女欧洲| 欧美老少配视频| h片视频在线观看| 麻豆av一区| 丁香网亚洲国际| 精品动漫3d一区二区三区免费| 成人黄色电影在线| 国产精品第100页| 国产精品福利一区二区| 在线观看免费国产小视频| 亚洲精品福利在线观看| 国产高清欧美| 黄色在线视频网| 日韩av最新在线| 欧美日韩天堂| 日本aⅴ写真网站免费| 日韩在线免费av| 亚洲h色精品| 777视频在线| 在线日韩日本国产亚洲| 性一交一乱一区二区洋洋av| caoporn超碰国产公开| 一区二区三区回区在观看免费视频| 午夜日韩激情| 国产激情视频网址| 久久天天躁狠狠躁夜夜av| 久久国内精品视频| 免费av网站在线看| 亚洲字幕一区二区| 婷婷综合五月天| 伊人久久大香线蕉综合网站| 成熟老妇女视频| 亚洲毛片在线免费观看| 久久综合婷婷| 日韩av网站大全| 欧美性xxxx| 日韩精品福利一区二区三区| 欧美成人在线网站| 欧洲亚洲免费在线| 欧美xfplay| 国产精品午夜免费| 欧美一级视频| 91精品入口| 日韩中文在线播放| 里番在线观看网站| 国产激情小视频在线| 久久久欧美一区二区| 国产精品123区| 竹内纱里奈兽皇系列在线观看| 欧美日韩亚洲一区二区三区在线观看| 欧美日韩国产一区中文午夜| 国产乱人伦精品一区| 国产91丝袜在线观看| 日韩av影院在线观看| 销魂美女一区二区三区视频在线| 国产综合在线观看| 亚洲成人精品一区| 欧美成人一级视频| 日韩天天综合| 狠狠噜天天噜日日噜| 亚洲高清精品中出| 成人欧美一区二区三区在线| 99re资源| 久久精品成人一区二区三区蜜臀| 亚洲国产高清国产精品| 国产精品xxxxx| 热99在线视频| 欧美大尺度激情区在线播放| 九九热hot精品视频在线播放| 欧美精品一区免费| 日韩在线视频免费观看高清中文| 99久久国产免费看| 日韩成人久久| 91se在线观看| 成人av资源网| 欧美一区二区久久| 美女一区二区三区在线观看| 日韩av首页| 成人黄色网页| 国产精品国产精品| 亚洲欧美日韩天堂| 一区在线观看视频| 一区在线播放| 91精品影视| 男人免费av| 日本亚洲自拍| 久99久在线视频| 欧美色播在线播放| 国产一区二区三区不卡在线观看| 女仆av观看一区| 欧美三级电影一区二区三区| 警花观音坐莲激情销魂小说|