[软件资料] “状态机”来解析UART不定长度的协议帧

[复制链接]
 楼主| szt1993 发表于 2024-7-22 21:00 | 显示全部楼层 |阅读模式
通信设计中考虑协议的灵活性,经常把协议设计成“不定长度”。一个实例如下图:锐米LoRa终端的通信协议帧。



1.png



如果一个系统接收上述“不定长度”的协议帧,将会有一个挑战--如何高效接收与解析。



为简化系统设计,我们强烈建议您采用“状态机”来解析UART数据帧,并且把解析工作放在ISR(中断服务程序)完成,仅当接收到最后一个字节(0x0D)时,再将整个数据帧提交给进程处理。



该解析状态机的原理如下图所示:


2.png



那么ISR处理这个状态机来得及吗?答案是:so easy!因为它只有3个动作,运算量十分小:



比较接收数据 -> 更新状态变量 -> 存储接收数据,C语言仅3条语句,翻译成机器指令也不超过10条。



代码清单如下:

  1. /**
  2. * [url=home.php?mod=space&uid=247401]@brief[/url]  Status of received communication frame
  3. */
  4. typedef enum
  5. {
  6.     STATUS_IDLE = (uint8_t)0,
  7.     STATUS_HEAD, /* Rx Head=0x3C */
  8.     STATUS_TYPE, /* Rx Type */
  9.     STATUS_DATA, /* Data filed */
  10.     STATUS_TAIL, /* Tail=0x0D */
  11.     STATUS_END, /* End of this frame */
  12. } COMM_TRM_STATUS_TypeDef;

  13. /**
  14. * @brief  Data object for received communication frame
  15. */
  16. typedef struct
  17. {
  18.     uint8_t    byCnt; /* Count of 1 field */
  19.     uint8_t    byDataLen; /* Length of data field */
  20.     uint8_t    byFrameLen; /* Length of frame */
  21.     COMM_TRM_STATUS_TypeDef    eRxStatus;
  22.     uint8_t    a_byRxBuf[MAX_LEN_COMM_TRM_DATA];
  23. } COMM_TRM_DATA;

  24. /**
  25. * @brief  Data object for received communication frame.
  26. * [url=home.php?mod=space&uid=536309]@NOTE[/url]  Prevent race condition that accessed by both ISR and process.
  27. */
  28. static COMM_TRM_DATA    s_stComm2TrmData;

  29. /**
  30.   * @brief  Put a data that received by UART into buffer.
  31.   * @note  Prevent race condition this called by ISR.
  32.   * @param  uint8_t byData: the data received by UART.
  33.   * @retval  None
  34.   */
  35. void comm2trm_RxUartData(uint8_t byData)
  36. {
  37.     /* Update status according to the received data */
  38.     switch (s_stComm2TrmData.eRxStatus)
  39.     {
  40.         case STATUS_IDLE:
  41.             if (COMM_TRM_HEAD == byData) /* Is Head */
  42.             {
  43.                 s_stComm2TrmData.eRxStatus = STATUS_HEAD;
  44.             }
  45.             else
  46.             {
  47.                 goto rx_exception;
  48.             }
  49.             break;
  50.         case STATUS_HEAD:
  51.             if (TYPE_INVALID_MIN < byData && byData < TYPE_INVALID_MAX) /* Valid type */
  52.             {
  53.                 s_stComm2TrmData.eRxStatus = STATUS_TYPE;
  54.             }
  55.             else
  56.             {
  57.                 goto rx_exception;
  58.             }
  59.             break;
  60.         case STATUS_TYPE:
  61.             if (byData <= MAX_LEN_UART_FRAME_DATA) /* Valid data size */
  62.             {
  63.                 s_stComm2TrmData.eRxStatus = STATUS_DATA;
  64.                 s_stComm2TrmData.byDataLen = byData;
  65.             }
  66.             else
  67.             {
  68.                 goto rx_exception;
  69.             }
  70.             break;
  71.         case STATUS_DATA:
  72.             if (s_stComm2TrmData.byCnt < s_stComm2TrmData.byDataLen)
  73.             {
  74.                 ++s_stComm2TrmData.byCnt;
  75.             }
  76.             else
  77.             {
  78.                 s_stComm2TrmData.eRxStatus = STATUS_TAIL;
  79.             }
  80.             break;
  81.         case STATUS_TAIL:
  82.             if (COMM_TRM_TAIL == byData)
  83.             {
  84.                 /* We received a frame of data, now tell process to deal with it! */
  85.                 process_poll(&Comm2TrmProcess);
  86.             }
  87.             else
  88.             {
  89.                 goto rx_exception;
  90.             }
  91.             break;
  92.         default:
  93.             ASSERT(!"Error: Bad status of comm2trm_RxUartData().\r\n");
  94.             break;
  95.     }

  96.     /* Save the received data */
  97.     s_stComm2TrmData.a_byRxBuf[s_stComm2TrmData.byFrameLen++] = byData;
  98.     return;

  99. rx_exception:
  100.     ClearCommFrame();
  101.     return;
  102. }

中国龙芯CDX 发表于 2024-7-26 22:54 | 显示全部楼层
比较接收数据 -> 更新状态变量 -> 存储接收数据,C语言仅3条语句,其实还是非常稳定的
 楼主| szt1993 发表于 2024-7-27 12:42 | 显示全部楼层
中国龙芯CDX 发表于 2024-7-26 22:54
比较接收数据 -> 更新状态变量 -> 存储接收数据,C语言仅3条语句,其实还是非常稳定的 ...

其实这个主要说的是代码执行效率问题
LOVEEVER 发表于 2024-8-12 14:20 | 显示全部楼层
比较接收数据 -> 更新状态变量 -> 存储接收数据这个是处理过程
 楼主| szt1993 发表于 2024-8-14 21:40 | 显示全部楼层
LOVEEVER 发表于 2024-8-12 14:20
比较接收数据 -> 更新状态变量 -> 存储接收数据这个是处理过程

过程简单了才能使得程序优化简单
小夏天的大西瓜 发表于 2024-8-21 15:04 | 显示全部楼层
C语言的精髓还是比较大的
jf101 发表于 2024-9-17 21:17 | 显示全部楼层
C语言一定要多练习才是重要的
您需要登录后才可以回帖 登录 | 注册

本版积分规则

344

主题

2836

帖子

6

粉丝
快速回复 在线客服 返回列表 返回顶部

344

主题

2836

帖子

6

粉丝
快速回复 在线客服 返回列表 返回顶部