0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何实现自定义串口通信协议?

strongerHuang 来源:嵌入式专栏 作者:strongerHuang 2021-06-01 10:01 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

有一些初学者总觉得通信协议是一个很复杂的知识,把它想的很高深,导致不知道该怎么学。 同时,偶尔有读者问关于串口自定义通信协议相关的问题,今天就来写写串口通信协议,并不是你想想中的那么难?

1什么通信协议?

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

通信协议是指双方实体完成通信或服务所必须遵循的规则和约定。通过通信信道和设备互连起来的多个不同地理位置的数据通信系统,要使其能协同工作实现信息交换和资源共享,它们之间必须具有共同的语言。交流什么、怎样交流及何时交流,都必须遵循某种互相都能接受的规则。这个规则就是通信协议。

相应该有很多读者都买过一些基于串口通信的模块,市面上很多基于串口通信的模块都是自定义通信协议,有的比较简单,有的相对复杂一点。 举一个很简单的串口通信协议的例子:比如只传输一个温度值,只有三个字节的通信协议:

帧头温度值帧尾

5A一字节数值3B

这种看起来是不是很简单?它也是一种通信协议。 只是说这种通信协议应用的场合相对比较简单(一对一两个设备之间),同时,它存在很多弊端。

2过于简单的通信协议引发的问题

上面那种只有三个字节的通信协议,相信大家都看明白了。虽然它也能通信,也能传输数据,但它存在一系列的问题。

比如:多个设备连接在一条总线(比如485)上,怎么判断传输给谁?(没有设备信息) 还比如:处于一个干扰环境,你能保障传输数据正确吗?(没有校验信息) 再比如:我想传输多个不确定长度的数据,该怎么办?(没有长度信息)。

上面这一系列问题,相信做过自定义通信的朋友都了解。 所以,在通信协议里面要约定更多的“协议信息”,这样才能保证通信的完整。

3通信协议常见内容

基于串口的通信协议通常不能太复杂,因为串口通信速率、抗干扰能力以及其他各方面原因,相对于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校验算是比较常见的,用于自定义协议中的校验方式。 还有一点,有的协议可能把校验码放在倒数第二,帧尾放在最后位置。

4通信协议代码实现

自定义通信协议,代码实现的方式有很多种,怎么说呢,“条条大路通罗马”你只需要按照你协议要写实现代码就行。 当然,实现的同时,需要考虑你项目实际情况,比如通信数据比较多,要用消息队列(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; //帧头1 uint8_t Head2; //帧头2 uint8_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); //设置计数值为0 TIM_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.更多

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

5最后

以上自定义协议内容仅供参考,最终用哪些、占用几个字节都与你实际需求有关。 基于串口的自定义通信协议,有千差万别,比如:MCU处理能力、设备多少、通信内容等都与你自定义协议有关。 有的可能只需要很简单的通信协议就能满足要求。有的可能需要更复杂的协议才能满足。

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

编辑:jq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 通信协议
    +关注

    关注

    28

    文章

    1101

    浏览量

    42477
  • 寄存器
    +关注

    关注

    31

    文章

    5625

    浏览量

    130789
  • 函数
    +关注

    关注

    3

    文章

    4423

    浏览量

    68118
  • 代码
    +关注

    关注

    30

    文章

    4985

    浏览量

    74590

原文标题:自定义串口通信协议,如何实现?

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    使用Python/MyHDL创建自定义FPGA IP

    使用 Python/MyHDL 创建自定义 FPGA IP,与 Vivado 集成,并通过 PYNQ 进行控制——实现软件上的简单硬件设计。
    的头像 发表于 04-09 09:53 531次阅读
    使用Python/MyHDL创建<b class='flag-5'>自定义</b>FPGA IP

    极海APM32F427如何实现自定义USB HID设备与PC进行通信

    最近需要使用到 APM32F427 枚举成Custom HID设备进行用户自定义通信,但是官方的例程只有提供的HID枚举为鼠标或者键盘类型的设备。这里记录一下,怎么使用Geehy官方的USB中间件,实现
    的头像 发表于 03-30 09:30 2930次阅读
    极海APM32F427如何<b class='flag-5'>实现</b><b class='flag-5'>自定义</b>USB HID设备与PC进行<b class='flag-5'>通信</b>

    极海APM32F427移植CherryUSB实现自定义USB HID设备

    最近需要使用到APM32F427枚举成Custom HID设备进行用户自定义通信,但我又不想要使用官方的USB中间件去做一个USB Custom HID设备。了解到Cherry USB这个开源
    的头像 发表于 01-20 09:52 4277次阅读
    极海APM32F427移植CherryUSB<b class='flag-5'>实现</b><b class='flag-5'>自定义</b>USB HID设备

    电能质量在线监测装置的自定义监测时段功能有哪些应用场景?

    电能质量在线监测装置的 自定义监测时段功能 ,核心价值是通过 “按需配置监测时间、采样频率和数据策略”,实现精准监测、资源优化与数据针对性分析,其应用场景覆盖工业生产、商业运营、电网运维、特殊保障等
    的头像 发表于 12-10 14:01 611次阅读
    电能质量在线监测装置的<b class='flag-5'>自定义</b>监测时段功能有哪些应用场景?

    无图形界面模式下自定义检查工具的应用

    此前文章已介绍 ANSA 中的自定义检查工具。本文将探讨该功能在无图形界面(No-GUI)模式下的应用,旨在满足标准化工作流程的需求,适用于需要高度自动化的前处理场景。通过集成自定义检查,用户可实现工作流程的高效自动化运行。
    的头像 发表于 11-30 14:13 867次阅读
    无图形界面模式下<b class='flag-5'>自定义</b>检查工具的应用

    电能质量在线监测装置的数据推送协议是否可以自定义配置?

    现代电能质量在线监测装置普遍支持 协议参数自定义配置 和 协议组合灵活适配 ,用户可根据实际需求调整通信参数、映射数据格式,甚至通过第三方工具实现
    的头像 发表于 11-12 13:58 629次阅读
    电能质量在线监测装置的数据推送<b class='flag-5'>协议</b>是否可以<b class='flag-5'>自定义</b>配置?

    采用汇编指示符来使用自定义指令

    具体实现 1、采用.word .half .dword等汇编指示符直接插入自定义指令,这种方法需要自己指定寄存器。其中.word为插入一个字的数据即32位,.half为插入半字即16位
    发表于 10-28 06:02

    强实时运动控制内核MotionRT750(九):内置C语言的自定义机械手模型实现

    内置C语言的自定义机械手模型实现
    的头像 发表于 10-27 14:14 1045次阅读
    强实时运动控制内核MotionRT750(九):内置C语言的<b class='flag-5'>自定义</b>机械手模型<b class='flag-5'>实现</b>

    上海泗博网关ENS-317——解锁串口设备数据直通 Modbus TCP 网络

    ENS-317是一款实现通用串口与Modbus TCP协议转换网关,配备双路RS485接口,可将多个采用自定义协议
    的头像 发表于 08-29 15:56 888次阅读
    上海泗博网关ENS-317——解锁<b class='flag-5'>串口</b>设备数据直通 Modbus TCP 网络

    通用串口转Modbus TCP 网关

    通用串口转Modbus TCP 网关 在工业自动化系统中,设备协议的多样性常常导致通信障碍。许多关键设备采用独特的串口协议
    的头像 发表于 08-13 15:59 789次阅读
    通用<b class='flag-5'>串口</b>转Modbus TCP 网关

    LOTO示波器自定义解码功能—CANFD解码

    /?spm_id_from=333.1365.list.card_archive.click 。 视频中仅对串口进行解码演示,正好我们从客户手中拿到了一块USB转CAN/CANFD模块,如下图所示, 本文借此对自定义解码功能
    的头像 发表于 07-11 10:34 1288次阅读
    LOTO示波器<b class='flag-5'>自定义</b>解码功能—CANFD解码

    大彩讲堂:VisualTFT软件如何自定义圆形进度条

    VisualTFT软件如何自定义圆形进度条
    的头像 发表于 07-07 17:10 2093次阅读
    大彩讲堂:VisualTFT软件如何<b class='flag-5'>自定义</b>圆形进度条

    KiCad 中的自定义规则(KiCon 演讲)

    “  Seth Hillbrand 在 KiCon US 2025 上为大家介绍了 KiCad 的规则系统,并详细讲解了自定义规则的设计与实例。  ”   演讲主要围绕 加强 KiCad 中的自定义
    的头像 发表于 06-16 11:17 2768次阅读
    KiCad 中的<b class='flag-5'>自定义</b>规则(KiCon 演讲)

    HarmonyOS应用自定义键盘解决方案

    自定义键盘是一种替换系统默认键盘的解决方案,可实现键盘个性化交互。允许用户结合业务需求与操作习惯,对按键布局进行可视化重构、设置多功能组合键位,使输入更加便捷和舒适。在安全防护层面,自定义键盘可以
    的头像 发表于 06-05 14:19 2773次阅读

    串口网关是什么

    自定义协议)转换为网络数据(如Modbus TCP、HTTP、MQTT)。 反之亦然:将网络指令转换为串口信号,控制串
    的头像 发表于 06-05 11:31 1035次阅读