[STM32] 串口数据帧处理(第一弹)_如何用define定义数据帧-程序员宅基地

技术标签: stm32  笔记  STM32  物联网  整理  


1 串口使用的常用场景

使用串口的主要目的是实现数据的交互,数据的交互的方法脱身于常用的场景。这里描述一个比较典型的场景:

MCU作为主控制器通过串口和外部的设备或者人进行交互。对于单片机端的设备往往存在一系列的指令。对于控制一个电机而言,他可以具备如下的行为,停止、转动、速度设置、位置设置等等。进而,对于多个微控制器存在的系统而言,各个控制器间也需要进行一定的信息交互,比如,从机向主机定时的发送心跳包来告知主机我的工作状态良好等。因此,一套遵从统一规则的串口帧协议十分重要,可以较好对命令集合进行处理,同时,可以对常见的问题进行合理的调整。

2 字节帧处理

制定串口协议的方式有很多,这里说一下我处理字节帧的一点经验。
以做过的一个项目为例,具体的帧结构如下图所示:

在这里插入图片描述

  • 数据帧头:AA 55
  • ID编号为固定的 0x10
  • CH: 输入 1 – 4 选通指定通道, 5:失能所有通道 6:使能所有通道
  • NULL: 空字节 可填入任意值
  • SUM: SUM = ID + CH + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL + NULL

帧定义了帧头、数据、帧尾。这里使用的帧属于定长帧,定长帧的优势在于处理的方式较为简便。缺点也显而易见,对于数据位较少的帧,是一种时间上的浪费(问题不大)。
解帧思路:同时获取到两个帧头,获取ID,通过ID来判断数据的长度,进而获取完整数据,再者检查校验码。校验码正确则消费数据,否则丢弃。

好了,直接上代码(有优化的可以评论留言):

#define FRAME_HEAD1         0xAA
#define FRAME_HEAD2         0x55
#define FRAME_DATA_LEN      20

/*
***********************************************************************************************
*	函 数 名: CheckSum
*	功能说明: 校验和判断
*	形    参:无
*	返 回 值: true :校验正确  false:校验失败
***********************************************************************************************
*/
static uint8_t CheckSum(uint8_t *pValueBuff, uint8_t u8BuffLen, uint8_t u8CheckNum)
{
    
    uint8_t u8Sum = 0;
    
    if (u8BuffLen > FRAME_DATA_LEN)
    {
    
        return FALSE;
    }
    
    for (uint8_t i = 0; i < u8BuffLen; i++)
    {
    
        u8Sum += *pValueBuff;
        pValueBuff++;
    }
    
    return u8CheckNum == u8Sum ? TRUE : FALSE;
}


/*
***********************************************************************************************
*	函 数 名: DealCmd
*	功能说明: 命令处理函数
*	形    参:数据帧
*	返 回 值: 无
***********************************************************************************************
*/
void DealCmd(uint8_t *pFrame)
{
    

    switch (*pFrame)
    {
    
        case EM_CYCLIC_ID:
        {
    
            DealCvFrame(pFrame);
        }
        break;  
        case EM_PULSE_ID:
        {
          
            DealDpvFrame(pFrame);
        }
        break;

        case EM_TEST_ID:
        {
          
            DealTestFrame(pFrame);
        }
        break; 
        default:
        {
    
            print(PERROR, " 无效ID\n ");
        }
        break;
    }
}


/*
***********************************************************************************************
*	函 数 名: GetConfigCmd
*	功能说明: 数据帧处理
*	形    参:无
*	返 回 值: 无
***********************************************************************************************
*/
void GetConfigCmd(void)
{
    
    uint8_t u8Temp = 0;
    static uint8_t u8index = 0;
    
    static uint8_t u8HeadFlag = 0;
    static uint8_t u8BuffTemp[FRAME_DATA_LEN + 1] = {
    0};
    
    
    while (comGetChar(CONNECT_COM, &u8Temp))
    {
    
        if (FRAME_HEAD1 == u8Temp  &&  FALSE == u8HeadFlag)
        {
    
            u8HeadFlag = 1;
        }
        
        if (FRAME_HEAD2 == u8Temp  &&  1 == u8HeadFlag)
        {
    
            u8HeadFlag = 2;
            continue;
        }

        if (2 == u8HeadFlag)
        {
    
            u8BuffTemp[u8index++] = u8Temp;
            if (u8index > FRAME_DATA_LEN)
            {
    
                if (CheckSum(u8BuffTemp, FRAME_DATA_LEN, u8BuffTemp[FRAME_DATA_LEN]))
                {
    
                    DealCmd(u8BuffTemp);    //消费数据    
                }
                u8HeadFlag = FALSE;
                u8index = 0;
                u8BuffTemp[0] = EM_MAX_ID;
                    
            }
        }
    }
}

这个实际上不是重点,下面来看看如何提取数据。
我们知道我们很大概率需要传送的数据有,(有无符号)整形,浮点型(往往更具精度乘一定倍数处理成整形)。那如何将这两种类型较好的处理呢。
这里我们用到了C种的结构体,我们知道结构体存在字节对齐的问题,但我们的数据是单字节对齐的,不存在空位的,因此需要使用如下命令告知编译器不适用默认的字节对齐规则,而是单字节对齐。
第二个问题是对于ARM而言,数据的存储方式是小端模式,但是我们的串口数据在接收后存在FIFO种是大端的模式,因此我们需要进行转换。

解决第一个问题:
#pragma pack(1) 中 1 代表单字节对齐。

#pragma pack(1)
typedef struct CV_DATA_FRAME
{
    
    uint8_t  Id;
    int16_t  InitE;
    int16_t  HighE;
    int16_t  LowE;
    uint8_t  InitPN;
    uint16_t ScanRate;
    uint32_t SweepSeg;
    uint8_t  SampleInt;
    uint32_t QuietTime;    
}stCvFrame;
#pragma pack()

解决第二个问题:

#define REVERSE_16(X)		((X & 0x00FFU) << 8 | (X & 0xFF00U) >> 8)

#define REVERSE_32(X)		((X & 0x000000FFU) << 24 | (X & 0x0000FF00U) << 8 | (X & 0x00FF0000U) >> 8 | (X & 0xFF000000U) >> 24)


OK, 到此为止一切解决。现在看一下提取数据:

/*
***********************************************************************************************
*	函 数 名: DealCvFrame
*	功能说明: 命令处理函数
*	形    参:数据帧
*	返 回 值: 无
***********************************************************************************************
*/
void DealCvFrame(uint8_t *pFrame)
{
    

    stCvFrame *p_stCvDataFrame;
    
    stCvFrame stCvDataTemp;

    p_stCvDataFrame = (stCvFrame *)pFrame;

    {
    
        stCvDataTemp.InitE        = REVERSE_16(p_stCvDataFrame->InitE);
        stCvDataTemp.HighE        = REVERSE_16(p_stCvDataFrame->HighE);
        stCvDataTemp.LowE         = REVERSE_16(p_stCvDataFrame->LowE);
        stCvDataTemp.InitPN       = p_stCvDataFrame->InitPN;
        stCvDataTemp.ScanRate     = REVERSE_16(p_stCvDataFrame->ScanRate);
        stCvDataTemp.SweepSeg     = REVERSE_32(p_stCvDataFrame->SweepSeg);
        stCvDataTemp.SampleInt    = p_stCvDataFrame->SampleInt;    
    }
    

    {
    
        stCvDataTemp.QuietTime = REVERSE_32(p_stCvDataFrame->QuietTime);
        stCvDataTemp.Id = p_stCvDataFrame->Id;
        
    }
}

总结

具体的实现都写在了正文中,同时如果你有好的建议欢迎留言交流,共同学习。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/Haley__xu/article/details/120142436

智能推荐

Base64Decoder使用时找不到包_xvlyc-程序员宅基地

文章浏览阅读568次。使用Base64加解密时,eclipse的设置public static void main(String[] args){String s=“H4sIAAAAAAAAAJVTPW/VMBT9K/wD2/kgbXVlyc/xo1Ff7JA4iNfFS6dKTJ3K72FDQgiQqkclGJgK”+“A1FB3ehQ9krdUCfs5MWJURey+N5zj2/uxzEsi5Uwel0JSqKEAJp8WLYyN0VOX5xg7CJbF2pRqVoP”+“JBeY+7AQTwppcqYFjXBEcI_xvlyc

Flutter中的MaterialButton不能使用全局主题颜色的真正原因_flutter textbutton 设置全局颜色不起作用-程序员宅基地

文章浏览阅读3.1k次。Flutter中的MaterialButton不能使用全局主题颜色的真正原因最近在使用flutter开发项目时遇到的问题,在页面中想设置下按钮的宽度,首先想到的是RaisedButton,主要是有阴影,并且官方推荐的,但是不能设置宽度和高度,然后选用其父类MaterialButton,可以使用minWidth这个属性,也有阴影等效果,但是新的问题,我设置的全局主题中的button颜色,不能应用..._flutter textbutton 设置全局颜色不起作用

.Net Core中的日志组件(Logging)-程序员宅基地

文章浏览阅读783次。1、介绍  Logging组件是微软实现的日志记录组件包括控制台(Console)、调试(Debug)、事件日志(EventLog)和TraceSource,但是没有实现最常用用的文件记录日志功能(可以用其他第三方的如NLog、Log4Net。之前写过NLog使用的文章)。2、默认配置  新建.Net Core Web Api项目,添加下面代码。 [Route("api/..._.net core logging

stm32软件模拟i2c通讯读取lm75a温度_float readtemperature(void){ unsigned char temp[2]-程序员宅基地

文章浏览阅读1.1w次,点赞16次,收藏105次。stm32硬件i2c有着一些bug,此外对于i2c这种通用的串行通信协议,从源头掌握和使用显然更加靠谱一些,当然,对于arm,还是直接操作寄存器来得方便的多。1、I2C协议1.1 i2c串行总线概述采用串行总线技术可以使系统的硬件设计大大简化、系统的体积减小、可靠性提高。同时,系统的更改和扩充更为容易。常用的串行扩展总线有:== I2C (Inter IC BUS)总线==、单总线(1-..._float readtemperature(void){ unsigned char temp[2]={0}; float temp_value=0

一文带你学习,动态规划算法-程序员宅基地

文章浏览阅读1.1k次,点赞4次,收藏7次。动态规划其实就是,给定一个问题,我们把它拆成一个个子问题,直到子问题可以直接解决。然后呢,把子问题答案保存起来,以减少重复计算。再根据子问题答案反推,得出原问题解的一种方法一般这些子问题很相似,可以通过函数关系式(DP方程)递推出来。动态规划就致力于解决每个子问题一次,减少重复计算。其核心思想就是:拆分子问题,记住过往,减少重复计算一个具体的例子:A : “上面等式的值是多少”B : 计算 “8”A : 在上面等式的左边写上 “1+” 呢?A : “此时等式的值为多少”_动态规划算法

android错误:Installation error: INSTALL_FAILED_VERSION_DOWNGRADE_packageinstaller 代码安装apk install_failed_version_do-程序员宅基地

文章浏览阅读546次。Installation error: INSTALL_FAILED_VERSION_DOWNGRADE,安装过一个开发的APP之后,需要把应用程序的安装包中的包文件目录修改一下,然后就出现了这个问题了,以前也出现过没有太注意,仔细查了一下资料,按其字面意思就是手机上安装的app版本比你当前想要安装的app版本低一些,所以只要提高版本号就好了,修改项目中AndroidManifest.xml文件中_packageinstaller 代码安装apk install_failed_version_downgrade

随便推点

登录及注册模块设置与流程图-程序员宅基地

文章浏览阅读6.7k次。原文地址:http://www.cocoachina.com/design/20170320/18918.html一、登录/注册模块流程图 1、电商&O2O类产品下单注册逻辑2、类似于简书&知乎等内容型社区产品二、关于注册流程1、同一页面中完成注册较适合填写信息较少(往往不包含关于用户个人信息的设置),注册流程简洁的产品,这类产品的注册所需的..._注册、登录、产品下单的流程图

python3之http.server模块_python http.server-程序员宅基地

文章浏览阅读5.7w次,点赞30次,收藏104次。有时候我们需要快速地搭建一个web服务,这时我们就可以使用python里面的http.server模块搭建http服务器。实现的方式有以下几种。BaseHTTPServer模块在Python3中已被合并到http.server1. 命令行启动pyhton -m http.server 80 #pyhton3中启动方式,开启的端口为80python -m SimpleHTTPServer 8080 #python2启动方式这种是常用的方式,很简单。2. 代_python http.server

「面试必背」TCP,UDP,Socket,Http网络编程面试题(快收藏)_socket面试题-程序员宅基地

文章浏览阅读3.7k次。网络通讯在系统交互中是必不可少的一部分,无论是职场面试还是工作中都是绕不过去的,本节我们来谈谈 Java 网络编程中的一些知识和网络编程面试题。1.先看一天面试的经验:2.什么是网络编程3.网络编程中两个主要的问题4.网络协议是什么在计算机网络要做到井井有条的交换数据,就必须遵守一些事先约定好的规则,比如交换数据的格式、是否需要发送一个应答信息。这些规则被称为网络协议。5.为什么要对网络协议分层简化问题难度和复杂度。由于各层之间独立,我们可以分割大问题为小_socket面试题

android游戏开发入门题大学题,android基础教程ppt-程序员宅基地

文章浏览阅读810次,点赞9次,收藏17次。对程序员来说,很多技术的学习都是“防御性”的。也就是说,我们是在为未来学习。我们学习新技术的目的,或是为了在新项目中应用,或仅仅是为了将来的面试。但不管怎样,一定不能“止步不前”,不能荒废掉。!文章以下内容会给出阿里与美团的面试题(答案+解析)、面试题库、Java核心知识点梳理等《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!**

前端「HTML+CSS」零基础入门学习笔记_前端【html+css零基础入门学习笔记-程序员宅基地

文章浏览阅读1.3k次,点赞29次,收藏41次。HTML+CSS入门学习,这篇就够了!_前端【html+css零基础入门学习笔记

递推法_递推法的一般步骤-程序员宅基地

文章浏览阅读1.9k次。目录设计思想典型例题一、兔子繁殖问题二、最大公约数三、猴子吃桃问题设计思想 利用所求解问题的本身具有的性质(递推关系)来求得问题解得有效方法。具体的做法是:对于一个问题,可以根据N=n之前的一步(n-1)或多步(n-1,n-2,n-3,·····)的结果推导出n时的解:f(n)=F(f(n-1),(n-2),·····) 称为递推关系式。递推算法的关键问题是得到相邻的数据项之间的关系,即递推关系。一般步骤:(1)确_递推法的一般步骤

推荐文章

热门文章

相关标签