【嵌入式项目应用】__UART自定义通信协议代码实现方法

目录

前言

一、什么是通信协议

二、简单通信协议的问题

三、通信协议的常见内容

1. 帧头

2. 设备地址/类型

3. 命令/指令

4. 命令类型/功能码

5. 数据长度

 6. 数据

7.帧尾

8.校验码

四、通信协议代码实现

1. 消息数据发送

a. 通过串口直接发送每一个字节

b. 通过消息队列发送

c. 用“结构体”代替“数组SendBuf”方式 

d. 其他更多

2. 消息数据接收

a. 常规中断接收

b. 增加超时检测

c. 更多

三、结束语

(* ̄︶ ̄)创作不易!期待你们的 点赞、收藏和评论喔。


前言

我们学习单片机,首先接触的可能是点灯(GPIO),再次就是串口(UART)。

串口是常用的一种通信接口,也是学嵌入式必备掌握的一项知识,但我发现有很多小伙伴只知道用串口输出或者打印一些数据,却不知道如何用串口进行数据传输和通信。

这里就给大家分享一下串口通信协议、自定义通信协议,以及实现的原理。

一、什么是通信协议

通信协议不难理解,就是两个(或多个)设备之间进行通信,必须要遵循的一种协议。

百度百科的解释:

通信协议:是指双方实体完成通信或服务所必须遵循的规则和约定。通过通信信道和设备互连起来的多个不同地理位置的数据通信系统,要使其能协同工作实现信息交换和资源共享,它们之间必须具有共同的语言。

交流什么、怎样交流及何时交流,都必须遵循某种互相都能接受的规则。这个规则就是通信协议。

应该有很多读者都买过一些基于串口通信的模块,市面上很多基于串口通信的模块都是自定义通信协议,有的比较简单,有的相对复杂一点。

举一个很简单的串口通信协议的例子:比如只传输一个温度值,只有三个字节的通信协议:

  1. 帧头
  2. 温度值
  3. 帧尾
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串口屏的例子:


#define DGUS_FRAME_HEAD1          0xA5                     //DGUS屏帧头1
#define DGUS_FRAME_HEAD2          0x5A                     //DGUS屏帧头2#define DGUS_CMD_W_REG            0x80                     //DGUS写寄存器指令
#define DGUS_CMD_R_REG            0x81                     //DGUS读寄存器指令
#define DGUS_CMD_W_DATA           0x82                     //DGUS写数据指令
#define DGUS_CMD_R_DATA           0x83                     //DGUS读数据指令
#define DGUS_CMD_W_CURVE          0x85                     //DGUS写曲线指令/* DGUS寄存器地址 */
#define DGUS_REG_VERSION          0x00                     //DGUS版本
#define DGUS_REG_LED_NOW          0x01                     //LED背光亮度
#define DGUS_REG_BZ_TIME          0x02                     //蜂鸣器时长
#define DGUS_REG_PIC_ID           0x03                     //显示页面ID
#define DGUS_REG_TP_FLAG          0x05                     //触摸坐标更新标志
#define DGUS_REG_TP_STATUS        0x06                     //坐标状态
#define DGUS_REG_TP_POSITION      0x07                     //坐标位置
#define DGUS_REG_TPC_ENABLE       0x0B                     //触控使能
#define DGUS_REG_RTC_NOW          0x20                     //当前RTCS//往DGDS屏指定寄存器写一字节数据
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));             //数据DGUS_SendByte((uint8_t)(Data&0xFF));
}//往DGDS屏指定地址写一字节数据
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));             //数据DGUS_SendByte((uint8_t)(Data&0xFF));
}

b. 通过消息队列发送

在上面基础上,用一个buf装下消息,然后“打包”到消息队列,通过消息队列的方式(FIFO)发送出去。

static uint8_t  sDGUS_SendBuf[DGUS_PACKAGE_LEN];//往DGDS屏指定寄存器写一字节数据
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);         //数据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屏指定地址写一字节数据
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);         //数据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);
}

c. 用“结构体代替数组SendBuf”方式 

结构体对数组更方便引用,也方便管理,所以,结构体方式相比数组buf更高级,也更实用。

(当然,如果成员比较多,如果用临时变量方式也会导致占用过多堆栈的情况)

比如:

typedef struct
{uint8_t  Head1;                 //帧头1uint8_t  Head2;                 //帧头2uint8_t  Len;                   //长度uint8_t  Cmd;                   //命令uint8_t  Data[DGUS_DATA_LEN];   //数据uint16_t CRC16;                 //CRC校验
}DGUS_PACKAGE_TypeDef;

d. 其他更多

串口发送数据的方式有很多,比如用DMA的方式替代消息队列的方式。

2. 消息数据接收

串口消息接收,通常串口中断接收的方式居多,当然,也有很少情况用轮询的方式接收数据。

a. 常规中断接收

还是以DGUS串口屏为例,描述一种简单又常见的中断接收方式:

void DGUS_ISRHandler(uint8_t Data)
{static uint8_t sDgus_RxNum = 0;                //数量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;}/* 确定一帧数据长度 */if(gDGUS_RxCnt == 3){sDgus_RxNum = sDgus_RxBuf[2] + 3;}/* 接收完一帧数据 */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. 增加超时检测

接收数据有可能存在接收了一半,中断因为某种原因中断了,这时候,超时检测也很有必要。

比如:用多余的MCU定时器做一个超时计数的处理,接收到一个数据,开始计时,超过1ms没有接收到下一个数据,就丢掉这一包(前面接收的)数据。

static void DGUS_TimingAndUpdate(uint16_t Nms)
{sDGUSTiming_Nms_Num = Nms;TIM_SetCounter(DGUS_TIM, 0);                   //设置计数值为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. 更多

接收和发送一样,实现方法有很多种,比如接收同样也可以用结构体方式。但有一点,都需要结合你实际需求来编码。

三、结束语

以上自定义协议内容仅供参考,最终用哪些、占用几个字节都与你实际需求有关。

基于串口的自定义通信协议,有千差万别,比如:MCU处理能力、设备多少、通信内容等都与你自定义协议有关。

有的可能只需要很简单的通信协议就能满足要求。有的可能需要更复杂的协议才能满足。

注意的是:

  1. 以上举例并不是完整的代码(有些细节没有描述出来),主要是供大家学习这种编程思想,或者实现方式。
  2. 一份好的通信协议代码,必定有一定容错处理,比如:发送完成检测、接收超时检测、数据出错检测等等。所以说,以上代码并不是完整的代码。

 


(* ̄︶ ̄)创作不易!期待你们的 点赞收藏评论喔。

本文来源网络,免费分享知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/127403.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

图像处理:图片二值化学习,以及代码中如何实现

目录 1、了解下图片二值化的含义 2、进行图像二值化处理的方法 3、如何选择合适的阈值进行二值化 4、实现图片二值化&#xff08;代码&#xff09; &#xff08;1&#xff09;是使用C和OpenCV库实现&#xff1a; &#xff08;2&#xff09;纯C代码实现&#xff0c;不要借…

<windows>win11输入法无法设置UI,变为了win10样子,解决方法

1 问题 升级win11版本后&#xff0c;输入法可能会变为旧版&#xff0c;如下图。如何恢复WIN11新版样子&#xff1f; 2 解决 点击开始–>设置–>搜索”语言“–>选择中文简体右侧语言选项。 下拉到最后选择微软拼音右侧三点&#xff0c;选择键盘选项–>选择常规…

Unity从入门到主程学习路线(内含学习资料)干货超全

写在最前 很多小伙伴想进阶Unity主程&#xff0c;进阶Unity架构师&#xff0c;不知道要学哪些知识&#xff0c;今天给大家分享一下比较完整的知识体系&#xff0c;Unity学习路线&#xff0c;介绍一些有干货的博主与教程,给大家做参考。 不管你是已经工作了&#xff0c;还是正…

Ubuntu:使用apache2部署Vue开发的网站

作者:CSDN @ _乐多_ 本文记录了Vue项目打包到部署到ubuntu系统的全部步骤。 文章目录 一、代码打包二、安装 Apache2三、开启/关闭apache23.1 开启3.2 关闭四、部署Vue应用到Apache24.1 首次部署4.2 更新部署五、全部操作截图一、代码打包 首先,确保您已经在本地开发环境中…

【C语言初阶】之函数

【C语言初阶】之函数 1. 函数是什么2. C语言中的函数2.1 库函数2.2.1 利用文档学习库函数 2.2 自定义函数 3. 函数参数3.1 实际参数(实参)3.2 形式参数(形参) 4. 函数调用4.1 传值调用4.2 传址调用4.3 练习 5. 函数的嵌套调用和链式访问5.1 嵌套调用5.2 链式访问 6. 函数的声明…

微服务架构之路1,服务如何拆分?使用微服务的注意事项?

目录 一、前言二、单体服务的弊端三、微服务化四、服务如何拆分&#xff1f;五、使用微服务的注意事项1、服务如何定义2、服务如何发布和订阅3、服务如何监控4、服务如何治理5、故障如何定位 大家好&#xff0c;我是哪吒。 一、前言 微服务已经是Java开发的必备技能&#xff…

Python画图之动态爱心

Python画出动态爱心&#xff08;有趣小游戏&#xff09; 一、效果图二、Python代码 一、效果图 二、Python代码 import random from math import sin, cos, pi, log from tkinter import *CANVAS_WIDTH 640 # 画布的宽 CANVAS_HEIGHT 480 # 画布的高 CANVAS_CENTER_X CANV…

线段树 区间赋值 + 区间加减 + 求区间最值

线段树好题&#xff1a;P1253 扶苏的问题 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 区间赋值 区间加减 求区间最大。 对于区间赋值和区间加减来说&#xff0c;需要两个懒标记&#xff0c;一个表示赋值cover&#xff0c;一个表示加减add。 区间赋值的优先级大于区间加…

【论文阅读】Towards Globally Optimal Normal Orientations for Large Point Clouds

文章目录 声明相关文章核心思想归纳主要贡献点主要流程核心公式机器翻译声明 本帖更新中如有问题,望批评指正!如果有人觉得帖子质量差,希望在评论中给出建议,谢谢!相关文章 这里列出一些相关的文章,方便对比其原理 Parallel Globally Consistent Normal Orientation of …

大麦协议开发

1. 用户抢购请求处理&#xff1a; - 后端实现&#xff1a;在后端&#xff0c;您可以创建一个用于处理用户抢购请求的接口。当用户发起抢购请求时&#xff0c;后端会根据一定的算法和逻辑来处理请求。例如&#xff0c;可以使用分布式锁来保证只有一个用户能够成功抢购。示例后端…

02.Oracle的启动过程

Oracle的启动过程 一、Oracle数据库的四种状态二、Oracle的启动过程 一、Oracle数据库的四种状态 Oracle数据库有四种状态&#xff1a;SHUTDOWN、NOMOUNT、MOUNT、OPEN. 1.SHUTDOWN状态 数据库没有启动 2.NOMOUNT状态 启动了instance&#xff08;数据库实例&#xff09;启动…

VS2017制作安装包如何将整个文件夹添加进依赖项中

找到安装项目右键view-文件系统 找到Application Folder 右键Add-Folder 如Python38 选中创建的 Python38 在右侧的空白处粘贴要复制的文件即可。文件多&#xff0c;等待时间较长

玩转硬件之Micro:bit的玩法(三)——计步器

随着技术的发展&#xff0c;现在智能手机和智能手表已经走进千家万户&#xff0c;所以大家对于计步器可能不陌生&#xff0c;计步器是一种用于计算行走步数的装置。它通常是一个小型电子设备&#xff0c;可以佩戴在身体上&#xff0c;如腕带、腰带或口袋中。计步器通过感应人体…

树结构及其算法-二叉查找树

目录 树结构及其算法-二叉查找树 C代码 树结构及其算法-二叉查找树 二叉树在建立的过程中是根据“左子树 < 树根 < 右子树”的原则建立的&#xff0c;因此只需从树根出发比较键值即可&#xff0c;如果比树根大就往右&#xff0c;否则往左而下&#xff0c;直到相等就找…

浅谈安科瑞无线测温产品在南非某变电站的应用

摘要&#xff1a;随着电力工业的发展&#xff0c;对设备的安全性、可靠性要求越来越高。在这种条件下&#xff0c;高压设备的无线测温系统应运而生。这种技术是将内置电池或电流感应和无线发射模块的测温传感器安装于各测温点&#xff0c;由于其体积小&#xff0c;且无需任何接…

CWE(Common Weakness Enumeration,通用缺陷枚举)

参考链接&#xff1a;https://cwe.mitre.org/ CWE&#xff08;Common Weakness Enumeration&#xff0c;通用缺陷枚举&#xff09;和CVE&#xff08;Common Vulnerabilities & Exposures&#xff0c;通用漏洞和风险&#xff09;都是在计算机软件安全领域中非常重要的公开数…

WordPress外链页面安全跳转插件

老白博客我参照csdn和腾讯云的外链跳转页面&#xff0c;写了一个WordPress外链安全跳转插件&#xff1a;给网站所有第三方链接添加nofollow标签和重定向功能&#xff0c;提高网站安全性。插件包括两个样式&#xff0c;由于涉及到的css不太一样&#xff0c;所以分别写了两个版本…

矢量图形编辑软件illustrator 2023 mac中文软件特点

illustrator 2023 mac是一款矢量图形编辑软件&#xff0c;用于创建和编辑排版、图标、标志、插图和其他类型的矢量图形。 illustrator 2023 mac软件特点 矢量图形&#xff1a;illustrator创建的图形是矢量图形&#xff0c;可以无限放大而不失真&#xff0c;这与像素图形编辑软…

赋能制造业高质量发展,释放采购数字化新活力——企企通亮相武汉2023国际智能制造创新论坛

摘要 “为应对成本上升、供应端不稳定、供应链上下游协同困难、决策无数据依据等问题&#xff0c;利用数字化手段降本增效、降低潜在风险十分关键。在AI等先进技术发展、供应链协同效应和降本诉求等机遇的驱动下&#xff0c;采购供应链数字化、协同化成为企业激烈竞争的优先选…

几种常见的接地类型详解

接地作为一种应用最为广泛的电气安全措施&#xff0c;是指电力系统和电气装置的中性点、电气设备的外露导电部分和装置外导电部分经由导体与大地相连。接地的作用主要是防止人身遭受电击、设备和线路遭受损坏、预防火灾和防止雷击、防止静电损害和保障电力系统正常运行。按其功…