首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >嵌入式开发中自定义协议的解析与组包

嵌入式开发中自定义协议的解析与组包

作者头像
用户2366192
发布2021-10-13 15:06:07
9210
发布2021-10-13 15:06:07
举报
文章被收录于专栏:TopSemic嵌入式TopSemic嵌入式

在嵌入式产品开发中,经常会遇到两个设备之间的通信、设备与服务器的通信、设备和上位机的通信等,很多时候通信协议都是自定义的,所以就涉及到自定义协议的解析和组包问题。

比如针对下面的这样一个协议:

帧头1

帧头2

字段1

字段2

校验

固定值:0x55

固定值:0xAA

设备ID

电压值

前面所有数据异或值

char

char

short

float

char

1字节

1字节

2字节

4字节

1字节

数据在发送时涉及到一个大小端的概念,大小端是针对多字节数据的传输,比如上述协议中字段1,假设两字节内容为0x0001,先发送0x01后发送0x00,称为小端模式;先发送0x00后发送0x01,称为大端模式。

假设字段1内容为0x001,字段2内容为0x40533333 (对应为3.3)

假设按照小端方式发送,下面是帧数据:

55 AA 01 00 33 33 53 40 ED

下面来看看如何解析,

若干年前,在第一次面对这种问题时,用的如下傻瓜式的代码方式实现:

#include <stdio.h>

int main()
{
    unsigned char Rxbuf[9] = {0x55,0xAA,0x01,0x00,0x33,0x33,0x53,0x40,0xED};

    short DeviceId;
    float Voltage;

    unsigned char check = 0;
    int i;

    for(i=0;i<8;i++)
    {
        check ^= Rxbuf[i];
    }

    if(Rxbuf[0]==0x55 && Rxbuf[1]==0xAA && Rxbuf[8]==check )
    {
        DeviceId=(Rxbuf[3]<<8)|Rxbuf[2];
        Voltage= *((float *)&Rxbuf[4]);

        printf("DeviceId:%d\n",DeviceId);
        printf("Voltage:%f\n",Voltage);
    }

    return 0;
}

简单来说就是硬来,按照数组的先后顺序逐个重组解析,如果协议比较长,代码里会充斥着很多的数组下标,一不小心就数错了。而且如果更改协议的话,代码要改动很多地方。

后来有人告诉我可以定义个结构体,然后使用memcpy函数直接复制过去就完事了,

#include <stdio.h>
#include <string.h>
#pragma pack(1)
struct RxFrame
{
    unsigned char header1;   
    unsigned char header2;   
    short deviceId;  
    float voltage;    
    unsigned char check;
};


int main()
{
    unsigned char Rxbuf[9] = {0x55,0xAA,0x01,0x00,0x33,0x33,0x53,0x40,0xED};

    struct RxFrame RxData;

    unsigned char check = 0;
    int i;

    for(i=0;i<8;i++)
    {
        check ^= Rxbuf[i];
    }

    memcpy(&RxData,Rxbuf,sizeof(Rxbuf));

    if(Rxbuf[0]==0x55 && Rxbuf[1]==0xAA && RxData.check==check )
    {
        printf("DeviceId:%d\n",RxData.deviceId);
        printf("Voltage:%f\n",RxData.voltage);
    }

    return 0;
}

嗯,的确是方便了很多。不过该方式仅适合小端传输方式。

再后来,又见到有人用如下代码实现,

#include <stdio.h>
#include "convert.h"

int main()
{
    unsigned char Rxbuf[9] = {0x55,0xAA,0x01,0x00,0x33,0x33,0x53,0x40,0xED};

    short DeviceId;
    float Voltage;

    unsigned char check = 0;
    int i;
    int index = 0;

    for(i=0;i<8;i++)
    {
        check ^= Rxbuf[i];
    }

    if(Rxbuf[0]==0x55 && Rxbuf[1]==0xAA && Rxbuf[8]==check )
    {
        index += 2;
        ByteToShort(Rxbuf, &index, &DeviceId);
        ByteToFloat(Rxbuf, &index, &Voltage);

        printf("DeviceId:%d\n",DeviceId);
        printf("Voltage:%f\n",Voltage);
    }

    return 0;
}

其中convert.h如下:

#ifndef CONVERT_H
#define CONVERT_H

void  ShortToByte(unsigned char* dest, int* index, short value);
void  FloatToByte(char* dest, int* index, float value);

#endif // CONVERT_H

convert.c如下:

#include "convert.h"
#include <string.h>
#include <stdbool.h>

static bool Endianflag = 0;

void ByteToShort(const unsigned char* source, int* index, short* result)
{
    int i, len = sizeof(short);
    char p[len];
    memset(p, 0, len);

    if(Endianflag == 1 )
    {
        for( i = 0; i < len; i++ )
            *(p+i) = *(source + *index + len - i - 1);
    }
    else
    {
        for( i = 0; i < len; i++ )
            *(p+i) = *(source + *index + i);
    }


    *result = *((short*)p);

    *index += len;
}

void ByteToFloat(unsigned char* source, int* index, float* result)
{
    int i, len = sizeof(float);
    char p[len];
    memset(p, 0, len);


    if(Endianflag == 1 )
    {
        for( i = 0; i < len; i++ )
            *(p+i) = *(source + *index + len - i - 1);
    }
    else
    {
        for( i = 0; i < len; i++ )
            *(p+i) = *(source + *index + i);
    }


    *result = *((float*)p);

    *index += len;
}

该方法既可以支持小端模式,也可以支持大端模式,使用起来也是比较方便。

除了上述2个函数,完整的转换包含以下函数,就是将Bytes转换为不同的数据类型,以及将不同的数据类型转换为Bytes。

#ifndef CONVERT_H
#define CONVERT_H

void  ByteToShort(const unsigned char* source, int* index, short* result);
void  ByteToInt(unsigned char* source, int* index, int* result);
void  ByteToLong(char* source, int* index, long long* result);
void  ByteToFloat(unsigned char* source, int* index, float* result);
void  ByteToDouble(unsigned char* source, int* index, double* result);
void  ByteToString(unsigned char* source, int* index, char* result, int length);


void  ShortToByte(unsigned char* dest, int* index, short value);
void  IntToByte(char* dest, int* index, int value);
void  LongToByte(char* dest, int* index, long long value);
void  FloatToByte(char* dest, int* index, float value);
void  DoubleToByte(unsigned char* dest, int* index, double value);
void  StringToByte(char* dest, int* index, int length, char* value);

#endif // CONVERT_H

组包的过程和解析的过程正好相反,这里不再赘述。你在开发中遇到这种问题,是如何处理的呢?欢迎留言讨论

2021年9月27-29日,ELEXCON深圳国际电子展暨嵌入式系统展即将在深圳国际会展中心(宝安)盛大开幕!届时展会以“嵌入式智能系统,加速中国AIoT技术商用落地”为主题,云集数百家嵌入式系统厂商、AIoT技术与解决方案厂商、MCU/SOC厂商、RISC-V厂商、存储厂商、嵌入式工控板厂商、工业显示/电源厂商、AI芯片与FPGA厂商展示前沿技术、新品及方案。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-09-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 TopSemic嵌入式 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
物联网
腾讯连连是腾讯云物联网全新商业品牌,它涵盖一站式物联网平台 IoT Explorer,连连官方微信小程序和配套的小程序 SDK、插件和开源 App,并整合腾讯云内优势产品能力,如大数据、音视频、AI等。同时,它打通腾讯系 C 端内容资源,如QQ音乐、微信支付、微保、微众银行、医疗健康等生态应用入口。提供覆盖“云-管-边-端”的物联网基础设施,面向“消费物联”和 “产业物联”两大赛道提供全方位的物联网产品和解决方案,助力企业高效实现数字化转型。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档