通信设计中考虑协议的灵活性,经常把协议设计成“不定长度”。一个实例如下图:锐米LoRa终端的通信协议帧。
如果一个系统接收上述“不定长度”的协议帧,将会有一个挑战--如何高效接收与解析。 为简化系统设计,我们强烈建议您采用“状态机”来解析UART数据帧,并且把解析工作放在ISR(中断服务程序)完成,仅当接收到最后一个字节(0x0D)时,再将整个数据帧提交给进程处理。 该解析状态机的原理如下图所示:
那么ISR处理这个状态机来得及吗?答案是:so easy!因为它只有3个动作,运算量十分小: 比较接收数据 -> 更新状态变量 -> 存储接收数据,C语言仅3条语句,翻译成机器指令也不超过10条。
代码清单如下:
/** *@briefStatusofreceivedcommunicationframe */ typedefenum { STATUS_IDLE=(uint8_t)0, STATUS_HEAD,/*RxHead=0x3C*/ STATUS_TYPE,/*RxType*/ STATUS_DATA,/*Datafiled*/ STATUS_TAIL,/*Tail=0x0D*/ STATUS_END,/*Endofthisframe*/ }COMM_TRM_STATUS_TypeDef; /** *@briefDataobjectforreceivedcommunicationframe */ typedefstruct { uint8_tbyCnt;/*Countof1field*/ uint8_tbyDataLen;/*Lengthofdatafield*/ uint8_tbyFrameLen;/*Lengthofframe*/ COMM_TRM_STATUS_TypeDefeRxStatus; uint8_ta_byRxBuf[MAX_LEN_COMM_TRM_DATA]; }COMM_TRM_DATA; /** *@briefDataobjectforreceivedcommunicationframe. *@notePreventraceconditionthataccessedbybothISRandprocess. */ staticCOMM_TRM_DATAs_stComm2TrmData; /** *@briefPutadatathatreceivedbyUARTintobuffer. *@notePreventraceconditionthiscalledbyISR. *@paramuint8_tbyData:thedatareceivedbyUART. *@retvalNone */ voidcomm2trm_RxUartData(uint8_tbyData) { /*Updatestatusaccordingtothereceiveddata*/ switch(s_stComm2TrmData.eRxStatus) { caseSTATUS_IDLE: if(COMM_TRM_HEAD==byData)/*IsHead*/ { s_stComm2TrmData.eRxStatus=STATUS_HEAD; } else { gotorx_exception; } break; caseSTATUS_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; }审核编辑:黄飞
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
状态机
+关注
关注
2文章
492浏览量
27630 -
数据帧
+关注
关注
0文章
47浏览量
6767 -
csdn
+关注
关注
2文章
17浏览量
6859
原文标题:状态机”来解析UART不定长度的协议帧
文章出处:【微信号:嵌入式情报局,微信公众号:嵌入式情报局】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
Verilog状态机+设计实例
在verilog中状态机的一种很常用的逻辑结构,学习和理解状态机的运行规律能够帮助我们更好地书写代码,同时作为一种思想方法,在别的代码设计中也会有所帮助。 一、简介 在使用过程中我们常说
什么是状态机?状态机5要素
玩单片机还可以,各个外设也都会驱动,但是如果让你完整的写一套代码时,却无逻辑与框架可言。这说明编程还处于比较低的水平,你需要学会一种好的编程框架或者一种编程思想!比如模块化编程、状态机编程、分层思想
状态模式(状态机)
以前写状态机,比较常用的方式是用 if-else 或 switch-case,高级的一点是函数指针列表。最近,看了一文章《c语言设计模式–状态模式(状态机)》(来源:embed linux
发表于 12-16 16:53
•9次下载
什么是状态机?状态机的种类与实现
状态机,又称有限状态机(Finite State Machine,FSM)或米利状态机(Mealy Machine),是一种描述系统状态变化的模型。在芯片设计中,
评论