通信设计中考虑协议的灵活性,经常把协议设计成“不定长度”。一个实例如下图:锐米LoRa终端的通信协议帧。
如果一个系统接收上述“不定长度”的协议帧,将会有一个挑战--如何高效接收与解析。
为简化系统设计,我们强烈建议您采用“状态机”来解析UART数据帧,并且把解析工作放在ISR(中断服务程序)完成,仅当接收到最后一个字节(0x0D)时,再将整个数据帧提交给进程处理。该解析状态机的原理如下图所示:
那么ISR处理这个状态机来得及吗?答案是:so easy!因为它只有3个动作,运算量十分小:
比较接收数据 -> 更新状态变量 -> 存储接收数据,C语言仅3条语句,翻译成机器指令也不超过10条。
代码清单如下:
/** * @brief Status of received communication frame */ typedef enum { STATUS_IDLE = (uint8_t)0, STATUS_HEAD, /* Rx Head=0x3C */ STATUS_TYPE, /* Rx Type */ STATUS_DATA, /* Data filed */ STATUS_TAIL, /* Tail=0x0D */ STATUS_END, /* End of this frame */ } COMM_TRM_STATUS_TypeDef; /** * @brief Data object for received communication frame */ typedef struct { uint8_t byCnt; /* Count of 1 field */ uint8_t byDataLen; /* Length of data field */ uint8_t byFrameLen; /* Length of frame */ COMM_TRM_STATUS_TypeDef eRxStatus; uint8_t a_byRxBuf[MAX_LEN_COMM_TRM_DATA]; } COMM_TRM_DATA; /** * @brief Data object for received communication frame. * @note Prevent race condition that accessed by both ISR and process. */ static COMM_TRM_DATA s_stComm2TrmData; /** * @brief Put a data that received by UART into buffer. * @note Prevent race condition this called by ISR. * @param uint8_t byData: the data received by UART. * @retval None */ void comm2trm_RxUartData(uint8_t byData) { /* Update status according to the received data */ switch (s_stComm2TrmData.eRxStatus) { case STATUS_IDLE: if (COMM_TRM_HEAD == byData) /* Is Head */ { s_stComm2TrmData.eRxStatus = STATUS_HEAD; } else { goto rx_exception; } break; case STATUS_HEAD: if (TYPE_INVALID_MIN < byData && byData < TYPE_INVALID_MAX) /* Valid type */ { s_stComm2TrmData.eRxStatus = STATUS_TYPE; } else { goto rx_exception; } break; case STATUS_TYPE: if (byData <= MAX_LEN_UART_FRAME_DATA) /* Valid data size */ { s_stComm2TrmData.eRxStatus = STATUS_DATA; s_stComm2TrmData.byDataLen = byData; } else { goto rx_exception; } break; case STATUS_DATA: if (s_stComm2TrmData.byCnt < s_stComm2TrmData.byDataLen) { ++s_stComm2TrmData.byCnt; } else { s_stComm2TrmData.eRxStatus = STATUS_TAIL; } break; case STATUS_TAIL: if (COMM_TRM_TAIL == byData) { /* We received a frame of data, now tell process to deal with it! */ process_poll(&Comm2TrmProcess); } else { goto rx_exception; } break; default: ASSERT(!"Error: Bad status of comm2trm_RxUartData(). "); break; } /* Save the received data */ s_stComm2TrmData.a_byRxBuf[s_stComm2TrmData.byFrameLen++] = byData; return; rx_exception: ClearCommFrame(); return; }
审核编辑:汤梓红
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
通信协议
+关注
关注
28文章
881浏览量
40299 -
状态机
+关注
关注
2文章
492浏览量
27529 -
通信设计
+关注
关注
0文章
25浏览量
10507 -
LoRa
+关注
关注
349文章
1689浏览量
231917
原文标题:如何高效解析不定长度的协议帧?
文章出处:【微信号:c-stm32,微信公众号:STM32嵌入式开发】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
用串口DMA传输不定长度包的方式
经常看到有人在问用DMA接收不定长包的问题,由于STM32DMA的特殊性,使其对接收不定长的数据不太方便,很多人都在为此伤脑筋,也有不少牛人想了不少的办法,例如用协议或开定时器判别等,其实这些方法都
发表于 12-26 08:48
STM32单片机接收不定长度字节数据的方法解析相关资料推荐
//m.obk20.com/article/899756.html今天说一下STM32单片机的接收不定长度字节数据的方法。由于STM32单片机带IDLE中断,所以利用这个中断,可以
发表于 07-01 09:41
STM32单片机的接收不定长度字节数据的方法
STM32串口使用IDLE中断接收不定长数据原理与源程序转载 2016年07月16日 18:20:24原文来自 今天说一下STM32单片机的接收不定长度字节数据的方法。由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长
发表于 08-04 09:05
stm32串口是如何实现接收不定长度数据的呢
stm32串口是如何实现接收不定长度数据的呢?串口接收数据一般会采用串口中断方式自动接收,要想接收不定长度数据,就需要让单片机在接收完成一帧数据之后,自动告知系统数据已经接收完成了,这个过程其实都是
发表于 08-11 08:18
STM32串口接收不定长数据帧
STM32串口接收不定长数据帧->链表数据帧说明二级目录三级目录数据帧说明STM32数据寄存器为USARTx->DR寄存器二级目录三级目录
发表于 12-06 07:05
STM32单片机的接收不定长度字节数据的方法
来说一下STM32单片机的接收不定长度字节数据的方法。由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长字节的数据,由于STM32属于ARM单片机,所以这篇文章的方法也适合其他
发表于 01-06 08:04
利用STM32F1的串口空闲中断实现不定长的数据断帧
常见的数据帧断帧方式有两种,一种是通过固定的协议格式断帧,另外一个是通过两帧数据的时间断帧,下面介绍最后一种,利用STM32F1的串口空闲中
发表于 02-17 07:55
STM32串口通信 (采用链表接收不定长数据帧)
STM32串口接收不定长数据帧->链表数据帧说明二级目录三级目录数据帧说明STM32数据寄存器为USARTx->DR寄存器二级目录三级目录
发表于 11-23 18:07
•30次下载
stm32 串口接收不定长度数据及黏包处理 + 串口DMA接收
,那么stm32串口是如何实现接收不定长度数据的呢? 串口接收数据一般会采用串口中断方式自动接收,要想接收不定长度数据,就需要让单片机在接收完成一帧数据之后,自动告知系统数据已经接收完成了,这个过程其实
发表于 12-23 19:09
•27次下载
单片机的通信协议该如何设计
通信设计中考虑协议的灵活性,经常把协议设计成“不定长度”。 一个实例如下图:锐米LoRa终端的通信协议帧。 如果一个系统接收
评论