新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > 嵌入式硬件通信接口協議-UART(五)數據包設計與解析

        嵌入式硬件通信接口協議-UART(五)數據包設計與解析

        作者: 時間:2019-03-20 來源:網絡 收藏

          上一節講到起止式SST(Start-Stop-Type)幀結構協議,該協議利用幀頭、長度、校驗構建幀結構,基于幀結構能實現對數據包的可靠、準確傳輸。

        本文引用地址:http://www.104case.com/article/201903/398679.htm

          應用層數據包設計思路

          回到工程本身,幀結構中的數據包才是應用程序最終需要解析使用的,且與具體的業務需求有關。

          這篇文章將簡單介紹,在數據包里如何設計應用層的交互指令,從而實現具體的業務需求。分享個思路,就當拋磚引玉了。

          類似于幀結構,在設計數據包時,根據交互邏輯的具體需求,同樣采用逐字節組成字段,字段組成數據包,從而完成指令交互。

          具體到項目中,一般地有目標地址、源地址、指令類型、傳輸方向、級聯序號、參數ID、參數值等等。

          字段的定義因項目需求而定,以上提及的字段可能存在且不限于此。

          以下介紹在具體項目中,對數據包設計與解析思路。工程實踐中方法眾多,相信很多經驗嫻熟的老工程師肯定都有各自巧妙的編程思路,歡迎在本頁留言交流。

          項目案例

          基于nRF51822的BLE終端設備,與上位機使用通信,物理線路使用USB轉

          數據包定義

          類型定義

          參數名&參數值定義

          根據以上定義,可以為應用程序設計指令解析的結構體,結構體中所定義的類型type和參數名para,使用枚舉類型定義:

          


          常規解析過程

          解析函數,一般地會把輸入參數的 *indata,利用一個新的結構體指針指向該輸入參數,之后的解析使用結構體指針來對數據處理,增強代碼可讀性!

          


          上述截圖中的定義方式出現了警告,這里需要做個如下的強制轉換:

          


          常規的判斷處理,多采用switch(){case :}聯合if(...){;}else(...){;}判斷邏輯,這個模式的判斷處理架構如下:

          


          以上的做法,依次去判斷類型type、參數名para,然后直接處理。當這兩個字段的枚舉成員數量少,倒還可以這么判斷;但是如果工程需要擴展、業務有了新的需求,那么if(...){;}else(...){;}的逐一判斷將會使得解析函數里的代碼量巨大!

          總結有這幾個缺點:

          1.業務需求有多少個類型或者其他分支,就需要多少個這樣的判斷邏輯,對于編寫代碼變成個體力活;

          2.在代碼查看、維護時,面對的還是羅列了一大堆的switch(){case :}和if(...){;}else(...){;}語句;

          3.增刪功能時,需要找到代碼中具體的判斷位置,然后小心翼翼給注釋或者修改掉。

          這里已經沒有任何的技術含量,基本上就是復制粘貼判斷語句、修改判斷對象,說到底也就是個查表的過程!

          構建查表方式解析

          既然要查表,當然是有個while()循環,然后遞增某一變量來查表的過程。在這里,數據包結構體中定義的類型type、參數名para,都可以作為查表的對象,該如何選擇?

          假設:

          1.以類型作為查表對象,假如查表后類型等于查詢參數,那么參數名仍然是個多個分支的情況,要么繼續查表要么繼續采用switch(){case :}或者if(...){;}else(...){;}來判斷眾多不同的參數名;

          2.以參數名作為查表對象,假如查表后參數名等于設備運行狀態,那么類型需要做最多三種判斷:查詢、設置、其他。

          對比以上兩種,必然是第2個更能提高編程效率、縷清邏輯框架。

          要查表就要建表,建表的結構體,以參數名para作為被查對象,并且以回調函數的形式執行查表結果。建表如下:

          


          說是建表,其實就是定義一個結構體數組,數組的每個元素都是結構體類型,這里的結構體,主要由數據包協議的參數名和回調函數組成,定義如下:

          


          在執行數據包解析的時候,查表的思路是:

          1.先創建一個表結構的指針*ptable指向表的開始位置,也就是指向數組內第一個元素{ECHO, dcapp_dev_echo}

          2.再創建一個數據包結構的指針*pbuf指向輸入數據首地址

          3.通過遞增ptable指針,對ptable與pbuf的參數名成員進行比對

          4.最后執行ptable指針對應回調函數

          


          以上的思路,放到代碼中,僅僅數行就可以實現對輸入數據包參數名的解析!高效、清晰!

          另外,建表時,把無效參數名對應的值和對應的回調函數放在最后,這樣做的好處是查完整個表,無需區分是否找到對應的參數名,而直接執行指針對應的回調函數即可。

          這樣即使是未找到參數名,也會執行表中最后一個元素,就是錯誤解析的回調函數dcapp_parser_err()。

          有了這樣一個查表的處理方式,增刪指令功能就變得簡單太多了!增加功能,只需要在表中添加參數名和對應的回調函數,刪除某功能,也是回到表中找到對應的參數名和回調函數即可!

          總結一下,雖然查表方式非常清晰,但是對應的回調函數內部,需要獨自處理和實現,并且每個參數名都需要單獨處理。相比于采用switch(){case :}聯合if(...){;}else(...){;}判斷邏輯,確實清晰很多。

          以上的查表思路,來源于經歷的項目,同時還參考了

          《STM32CubeExpansion_MEMSMIC1_V1.1》

          這個ST官方的數字麥克風開源項目示例,作為USB音頻設備時,類似的回調函數方式:

          


          調試截圖

          正確解析了數據包的參數名之后,對應的函數執行結果是打印輸出調試信息,如下截圖:

          


          以上是初步的解析效果,可以通過回調函數,正確地跳轉到對應的函數執行。具體的處理仍需要針對項目的業務需求而設計,在此不做更多的延伸。



        關鍵詞: 嵌入式 UART

        評論


        相關推薦

        技術專區

        關閉
        主站蜘蛛池模板: 白水县| 镇康县| 宽城| 海林市| 瑞昌市| 曲松县| 云龙县| 蚌埠市| 岑巩县| 虎林市| 新昌县| 合肥市| 莱西市| 辉县市| 北流市| 五华县| 开阳县| 昆明市| 黄石市| 海丰县| 长泰县| 游戏| 安龙县| 象山县| 报价| 博爱县| 葫芦岛市| 本溪市| 巫山县| 花莲市| 舞阳县| 天水市| 河东区| 汪清县| 略阳县| 秀山| 积石山| 正定县| 香港| 垦利县| 青阳县|