单片机串口接收状态机STM32

单片机串口接收状态机stm32

前言

项目的芯片stm32转国产,国产芯片的串口DMA接收功能测试不通过,所以要由原本很容易配置的串口空闲中断触发DMA接收数据的方式转为串口逐字节接收的状态机接收数据

两种方式各有优劣,不过我的芯片已经主频跑到72m,对于接收115200波特率的数据,绰绰有余。

给一张图,接收状态机就是设置串口每次接收1byte的数据就触发一次中断,在中断函数里面逐次统计数据,最后把有用的数据包放进缓存区给到处理区

协议图.drawio

1.首先现在cubemx的串口的dma接收关闭,具体配置看截图,然后导出下代码

image

image

image

2.开始配置代码,这里我用的代码编辑器是vscode,个人认为vscode编辑代码+keil调试的方式是最舒服的,有兴趣的可以试试,绝对提一档。

image

先定义一个uint8_t的变量用来接收每次收到的数据,一定注意,只要你是用的cubemx生成的project,在生成的文件下,都有把你改的代码写到/* USER CODE BEGIN 0 */​下面,不然等你下次在cubemx更改底层配置的时候会把你改好的代码擦洗掉。

uint8_t USART1_rxdata = 0;

image

往下滑,找到这个串口1的初始化函数,加这个串口接收中断启动函数

HAL_UART_Receive_IT(&huart1, &USART1_rxdata, 1);

然后我们来到中断c文件下,在串口1中断里增加我们的接收状态机

image

  rec_buff_scan(USART1_rxdata);   HAL_UART_Receive_IT(&huart1, &USART1_rxdata, 1); 

这里的rec_buff_scan就是我们定义的函数,因为我的习惯是在我自己创建的文件下写代码,避免更改cube生成的东西,所以文件名可能就不一样,这里就只看代码就好了

首先先把定义全部cv下来,我们协议是[0XAA,0X55,命令包长度,…(命令包内容),校验和高位,校验和底位,的格式,如果你们的不一样就改下RecState里面的变量名

image

#define REC_BUFF_SIZE 100
uint8_t rec_buf_scan[REC_BUFF_SIZE];    // 数据接收缓冲区
uint8_t rec_index = 0;                  // 数据接收索引
extern uint8_t USART1_rxdata;                  // 存储接收的单个字节// 状态枚举
typedef enum {STATE_WAIT_FOR_HEAD1,      // 等待帧头AASTATE_WAIT_FOR_HEAD2,      // 等待帧头55STATE_WAIT_FOR_LENGTH,     // 等待命令包长度STATE_RECEIVE_DATA,        // 接收命令包数据STATE_WAIT_FOR_CHECKSUM1,  // 等待校验和字节1STATE_WAIT_FOR_CHECKSUM2   // 等待校验和字节2
} RecState;RecState rec_state = STATE_WAIT_FOR_HEAD1;  // 初始状态
uint8_t packet_length = 0;                  // 数据包长度
uint16_t checksum = 0;  

然后把状态机整段cv,状态机就是把接收部分分成一个个的状态,条件符合就会跳到下一段,最后会在校验和验证整段数据是否接收正常,正常的话就会送入数据处理的函数里面,我这里定义了一个rec_buf_scan[REC_BUFF_SIZE]是跟我之前的接收数据缓存区做兼容而已,你们可以只导入数据包的内容,不需要把头和校验和导入,这里其实就是我懒了嘻嘻

我这里的处理其实你们不用借鉴的 ,直接在STATE_WAIT_FOR_CHECKSUM2状态判断校验和成功后,把接收成功flag=true;然后在你另外的处理函数就可以处理数据包了,这里的memcpy一下,也是很有必要的,我们的处理缓存和接收缓存一定要区分开来,因为没有自锁,所以只能在成功的时候把接受的数据塞入处理缓存,至于我这里为什么塞了全部数据而不是只有数据包,主要是配合我以前的处理,无需借鉴,总之你上班就明白我的难处,要改来改去还要兼容是很麻烦的事情

image

void rec_buff_scan(uint8_t byte) {switch (rec_state) {case STATE_WAIT_FOR_HEAD1:if (byte == 0xAA) {rec_state = STATE_WAIT_FOR_HEAD2;rec_index = 0;  // 重置接收索引checksum = 0;   // 重置校验和rec_buf_scan[rec_index++] = byte;//可以不用导入}break;case STATE_WAIT_FOR_HEAD2:if (byte == 0x55) {rec_state = STATE_WAIT_FOR_LENGTH;rec_buf_scan[rec_index++] = byte;//可以不用导入} else {rec_state = STATE_WAIT_FOR_HEAD1;}break;case STATE_WAIT_FOR_LENGTH:packet_length = byte;rec_buf_scan[rec_index++] = byte;//可以不用导入rec_state = STATE_RECEIVE_DATA;break;case STATE_RECEIVE_DATA:if (rec_index-3 < packet_length) {//这里if里面的条件把-3去掉 rec_buf_scan[rec_index++] = byte;}if (rec_index-3 == packet_length) {//这里if里面的条件把-3去掉 checksum = Frame_CalculationChecksum(&rec_buf_scan[3], packet_length);  // 计算校验和rec_state = STATE_WAIT_FOR_CHECKSUM1;}break;case STATE_WAIT_FOR_CHECKSUM1:if (byte == ((checksum >> 8) & 0xFF)) {rec_state = STATE_WAIT_FOR_CHECKSUM2;rec_buf_scan[rec_index++] = byte;//可以不用导入} else {rec_state = STATE_WAIT_FOR_HEAD1;}break;case STATE_WAIT_FOR_CHECKSUM2:if (byte == (checksum & 0xFF)) {//这里做你自己的处理就好了 ,程序跑到这里就已经验证通过了rec_buf_scan[rec_index++] = byte;//可以不用导入memcpy(&rec_buff[0], &rec_buf_scan[0], rec_index);Device_data.device_state = JUDGE_FLAG;Data_queue_rx.Interrupt_Len = rec_index;}// 无论校验是否通过,回到初始状态rec_state = STATE_WAIT_FOR_HEAD1;break;}
}

用到的Frame_CalculationChecksum函数是校验和计算函数,具体操作就是把接收的数据需要校验的那一段的第一个元素地址放进去,把数据包长度放进去,他会算完把结果返回,我这里就是简单的数据包加和校验

uint16_t Frame_CalculationChecksum(uint8_t *pData, uint8_t u8Length)
{uint16_t u16check_sum = 0;uint8_t i;for (i = 0; i < u8Length; i++){u16check_sum = u16check_sum + pData[i];}return u16check_sum;
}

这样基本上就把接收状态机整完了,可以仿真下试试看。

3.仿真测试

可以看到断点在第二步的时候0xaa已经存入数据缓存去了

image

同理我们直接看到接收成功这里打断点,可以看到我们在把接收成功时,已经把所有数据都塞进处理缓存区了。这里再严谨点最好是在移完数据后把接收缓存清零一下,但是不清也不影响。

image

image

4.小结

串口中断状态机是最基本的协议解析接收方式,作为一个嵌入式人员这个你必须要学会的,写法不限制,但是流程就是跟我的差不多的,具体根据协议来定的。

这种接收方式适合没有dma外设的单片机比如51单片机,国产单片机,在项目开发完成进行降本的时候也是可能会改到这种方式的,如果有不规范的地方请留言,我会更改。

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

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

相关文章

【Python编程实例】-深入理解Python线程安全

深入理解Python线程安全 文章目录 深入理解Python线程安全1、Python中的线程2、线程安全2.1 GIL 及其对线程的影响2.2 竞态条件3、同步原语3.1 线程锁3.2 信号量4、使用同步原语进行通信和协调4.1 事件用于信号通知4.2 条件变量用于条件等待4.3 协调用屏障(Barriers for Coord…

词嵌入方法(Word Embedding)

词嵌入方法&#xff08;Word Embedding&#xff09; Word Embedding是NLP中的一种技术&#xff0c;通过将单词映射到一个空间向量来表示每个单词 ✨️常见的词嵌入方法&#xff1a; &#x1f31f;Word2Vec&#xff1a;由谷歌提出的方法&#xff0c;分为CBOW&#xff08;conti…

【go从零单排】实现枚举类型(Enum)

&#x1f308;Don’t worry , just coding! 内耗与overthinking只会削弱你的精力&#xff0c;虚度你的光阴&#xff0c;每天迈出一小步&#xff0c;回头时发现已经走了很远。 &#x1f4d7;概念 在Go语言中&#xff0c;并没有内置的枚举类型&#xff08;Enum&#xff09;&…

Python爬虫如何处理验证码与登录

Python爬虫如何处理验证码与登录 Python 爬虫在抓取需要登录的网站数据时&#xff0c;通常会遇到两个主要问题&#xff1a;登录验证和验证码处理。这些机制是网站用来防止自动化程序过度抓取数据的主要手段。本文将详细讲解如何使用 Python 处理登录与验证码&#xff0c;以便进…

MOS管损坏原因

MOS管是什么&#xff1f; MOS管&#xff0c;全程就是MOSFET&#xff08;Metal-Oxide-Semiconductor Field-Effect Transistor&#xff09;&#xff0c;是一种场效应晶体管。‌ MOS管控制原理 MOS管的工作原理是通过栅极电压&#xff08;G&#xff09;来控制源极&#xff08…

「QT」QT5程序设计专栏目录

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「QT」QT5程序设计&#x1f4da;全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasolid…

论文阅读《机器人状态估计中的李群》

目录 摘要1 介绍2 微李理论2.1 李群 摘要 李群是一个古老的数学抽象对象&#xff0c;可以追溯到19世纪&#xff0c;当时数学家 Sophus Lie奠定了连续变换群理论的基础。多年后&#xff0c;它的影响已经蔓延到科学和技术的各个领域。在机器人领域&#xff0c;我们最近正在经历一…

EHOME视频平台EasyCVR视频融合平台使用OBS进行RTMP推流,WebRTC播放出现抖动、卡顿如何解决?

在现代视频监控领域&#xff0c;跨区域的网络化视频监控管理平台成为了大中型项目的首选。EHOME视频平台EasyCVR以其强大的功能和兼容性&#xff0c;成为了众多项目的核心组件。它不仅能够管理视频资源、设备、用户、运维和安全&#xff0c;还支持多种行业标准协议&#xff0c;…

go语言使用总结(持续更新)

整理后的内容如下&#xff1a; 1. 先了解函数签名&#xff0c;再了解传入参数以及调用 函数签名是函数的声明部分&#xff0c;包括函数名、参数列表和返回值列表。理解函数签名是理解函数行为的第一步&#xff0c;尤其是在了解参数类型、参数数量和返回值类型等方面。通过了解…

浮动路由:实现出口线路的负载均衡冗余备份。

浮动路由 Tip&#xff1a;浮动路由指在多条默认路由基础上加入优先级参数&#xff0c;实现出口线路冗余备份。 ip routing-table //查看路由表命令 路由优先级参数&#xff1a;越小越优 本次实验测试两条默认路由&#xff0c;其中一条默认路由添加优先级参数&#xff0c;设置…

Android CCodec Codec2 (十九)C2LinearBlock

在上一篇文章的结尾&#xff0c;我们看到fetchLinearBlock方法最终创建了一个C2LinearBlock对象。这一节&#xff0c;我们将深入了解C2LinearBlock是什么&#xff0c;它的作用是什么&#xff0c;以及它是如何被创建的。 1、_C2BlockFactory 先对上一篇文章的结尾内容做简单回顾…

Axure PR 9 多级下拉选择器 设计交互

​ 大家好&#xff0c;我是大明同学。 Axure选择器是一种在交互设计中常用的组件&#xff0c;这期内容&#xff0c;我们来探讨Axure中多级下拉选择器设计与交互技巧。 下拉列表选择输入框元件 创建选择输入框所需的元件 1.在元件库中拖出一个矩形元件。 2.选中矩形元件&…

SparkSql读取数据的方式

一、读取普通文件 方式一&#xff1a;给定读取数据源的类型和地址 spark.read.format("json").load(path) spark.read.format("csv").load(path) spark.read.format("parquet").load(path) 方式二&#xff1a;直接调用对应数据源类型的方法 …

使用特征构建进行连续变量的特征提取

特征构建(Feature Engineering)是机器学习过程中至关重要的一步,它直接影响模型的性能和准确性。通过对原始数据进行转换、处理和扩展,可以为模型提供更加丰富的信息,提升预测效果。特征构建的核心思想是利用现有的数据来生成新的特征,以便模型可以更好地捕捉潜在的规律和…

使用Python实现图像的手绘风格效果

使用Python实现图像的手绘风格效果 一、引言二、代码详细解释与示例三、完整框架流程四、运行五、结论附&#xff1a;完整代码 一、引言 在数字图像处理领域&#xff0c;模拟手绘风格是一项有趣且具有挑战性的任务。手绘风格图像通常具有独特的纹理和深浅变化&#xff0c;给人…

Oracle Select语句

SELECT语句使用方法 在Oracle中&#xff0c;表是由列和行组成。 例如&#xff0c;示例数据库中的customers表具有以下列&#xff1a;customer_id&#xff0c;name&#xff0c;address&#xff0c;website和credit_limit。customers表中这些列中也有对应的数据。 要从表的一个或…

Scala的集合。

定义&#xff1a;set表示没有重复元素的集合 特点&#xff1a;唯一&#xff0c;无序 Set有可变mutable和不可变immutable 两种类型。不可变set创建后元素不能修改&#xff0c;可变set可对元素进行添加&#xff0c;删除等操作&#xff0c;这两种类型能满足不同场景需求。 pack…

w~大模型~合集21

我自己的原文哦~ https://blog.51cto.com/whaosoft/12459590 #大模型~微调~用带反馈的自训练 面对当前微调大模型主要依赖人类生成数据的普遍做法&#xff0c;谷歌 DeepMind 探索出了一种减少这种依赖的更高效方法。大模型微调非得依赖人类数据吗&#xff1f;用带反馈的自训…

opencv 中 threshold 函数作用

在 OpenCV 中&#xff0c;threshold 函数用于将图像转换为二值图像&#xff0c;它通过设置一个阈值来将像素值分类为两类&#xff1a;低于阈值的像素设置为 0&#xff08;或黑色&#xff09;&#xff0c;高于阈值的像素设置为最大值&#xff08;通常是 255 或白色&#xff09;。…

ctfshow(316,317,318)--XSS漏洞--反射性XSS

反射型XSS相关知识 Web316 进入界面&#xff1a; 审计 显示是关于反射性XSS的题目。 思路 首先想到利用XSS平台解题&#xff0c;看其他师傅的wp提示flag是在cookie中。 当前页面的cookie是flagyou%20are%20not%20admin%20no%20flag。 但是这里我使用XSS平台&#xff0c;…