原创 基于51串口通讯编程软件架构剖析(请大家就本文相关内容发表评论,来探讨一个高效可靠的通讯架构)

2008-4-26 12:00 3354 4 7 分类: MCU/ 嵌入式

 


前言:
串口通讯对于所有的嵌入式工程师十分常见,对于一个与外界交互的系统必须依赖一些手段,比如串口、USB、红外、GPRS之类的数据通讯传输方式。而串口作为一种廉价的短距离可靠的通讯方式得到了广泛应用。
废话少说了,就此打住,进入正题。
本文主要从软件结构上讲解如何在资源比较缺乏的系统上实现通讯协议的串口通讯编程,以及如何优化程序效率,从而使系统更快、更稳定运行。


正文:
      我们以51单片机为例。51中一般针对串口通讯编程,通常采取中断接受查询发送的方式。中断函数在接受数据到达时被重复调用,其实是个重复入栈的过程,所以不宜将函数写的太长,函数太长一般会导致栈太深占用系统资源,二是处理时间过长,可能导致通讯出错。为了防止在处理数据过程中不受干扰,通常在处理接受数据前关闭中断,处理完后再开。
通常的的编程方式如下:
static void UartInterruptService(void) interrupt 4
{
    ES = 0;
    RI = 0;
    uart_process(SBUF);
    ES="1";
}
下面重点介绍数据处理函数 uart_process(SBUF);
其实很多时候,对于通讯传输的数据处理才是关键,尤其对于设计通讯协议而言。笔者在刚刚做的一个系统上就碰到这样的问题,当系统庞大了,资源十分有限的情况下,数据处理一旦占用资源太多,效率太低将导致系统崩溃而无法运行。
到了这里,很多工程师可能会考虑开个大的缓冲区FIFO将接收到的数据保存在缓冲区,然后对其进行解析、判断进行下一步程序编写,当然这在系统资源比较丰富的情况下是没有问题的,ARM上采取的就是这样的方式。但如何系统庞大呢,留给的资源缺乏则不行。这样做的一个很大缺点必须是将数据帧接收完了才能够判断,降低了效率和运行速度。
其实还有另外的方式,可以采取在每接收一个字节就对其解析,解析完判断转到下一个状态,并将其中的有用数据存储在相应的数据结构中去,可以采取状态机实现。


将状态机设计为两个控制状态,一是串口状态——uart_state ,一是命令类型状态——cmd_state .
(1)状态机开始状态:串口状态为CMD_NO
(2)接受到STX_CMD,状态变为CMD_START.
(3)接下来将自动进入接受命令帧的状态,再开启命令状态的状态机,对发送来的有用数据进行解析,保存,校验等。处理完毕后将uart_state设为CMD_END状态进行下一步的接受完毕判断,将cmd_state设置为初始的NO_CMD状态。
(4)最后进行ETX_CMD判断,判断数据接收是否完毕。



void uart_process(U8 u8)
{
     if(uart_state == CMD_NO)
     {
    if(u8 == STX_CMD)
      {
        uart_state = CMD_START;
      }
       
       }
    else if(uart_state == CMD_START)
    {
        switch(cmd_state)
        {
           case NO_CMD:
              cmd_state = u8;
              break;
               
           case COST_CMD:
                    //解析存储有用数据到相应数据结构中
                    //进行CRC校验
                    ……
                       uart_state = CMD_END;
                       cmd_state = NO_CMD;
                       CRC = 0;
                        break;
                        ……
                   }
              ……
            }
        else if(uart_state == CMD_END)
         {
            uart_state = CMD_NO;
            if(u8 == ETX_CMD)
            {
              //接受完毕
              //可以考虑抛出一个消息main函数循环中进行响应处理。
             }
        
           }
}
                         


接下来我们要讨论解析后我们数据存储的问题,其实在资源比较足够的情况下或者能够挤出data区的情况下可以考虑用结构体,我们构造好相应结构体,将接收到的数据存储进去,要应用的时候就十分方便。但这也有个矛盾,一般c51定义的结构体都被存储在data区,一般通讯的字节量大空间必然不够,存在一个矛盾,可以采用联合体union进行存储效果会好一点。当然也可以在保存数据时采用定义在xdata区(片外)的buffer来存储。这样在一定程序上优化了程序的执行效率,在程序处理立即抛出消息处理,提高了通讯数据的处理速度。对于通常资源比较丰富的系统,比如ARM上一般采取的做法是这样的,将数据存在缓冲区,接收完一帧数据后再转换成相应的数据结构,再进行分析、校验。   


总体来说,这种采取状态机实时解析串口通讯数据的方式在一定程序提高了程序运行效率,使软件架构清晰明了,程序可扩展性大,有利于后续开发。以上是笔者的一点愚见,欢迎指教。       

PARTNER CONTENT

文章评论3条评论)

登录后参与讨论

用户144403 2008-4-26 16:36

如果外部数据存储区够大,还是喜欢用分层的结构采用环型缓冲区,中断只负责收数到缓冲区以及从发送缓冲区中取字节数据发送。这种方式可以将程序分成物理层,链路层,(传输层),应用层等多个独立的层面,方便重用和修改。

用户144403 2008-4-26 16:36

全部用case跳转更好一些,因为当时MCU资源紧张,所以在解析的时候对stx和etx并没有保存…… 直接用一个状态机实现是更好的……

用户144403 2008-4-26 16:35

接收数据的同时,进行数据帧的解析,思路可取数据接收时,根据数据包的格式进行解析,完全正确时,将有用的数据体存放到待处理的数据区,遇到错误时跳出解析过程,从头开始并将出错信息反馈到上位机。楼主的程序用Case作跳转也许会更直观些,总是if else看多了就晕了。
相关推荐阅读
用户144403 2008-04-26 16:56
有限状态机
有限状态机(finite state machine)是一个数学概念,如果把它运用于程序中,可以发挥很大的作用。它是一种协议,用于有限数量的子程序("状态")的发展变化。每个子程序进行一些处理并选择下...
用户144403 2008-04-26 16:32
环形缓冲区
环形缓冲区在通信程序中,经常使用环形缓冲区作为数据结构来存放通信中发送和接收的数据。环形缓冲区是一个先进先出的循环缓冲区,可以向通信程序提供对缓冲区的互斥访问。1、环形缓冲区的实现原理<?xml...
用户144403 2008-04-08 14:55
C运行时库详解
         运行时库是程序在运行时所需要的库文件,通常运行时库是以LIB或DLL形式提供的。C运行时库诞生于20世纪70年代,当时的程序世界还很单纯,应用程序都是单线程的,多任务或多线程机制在此...
用户144403 2008-04-03 08:59
函数的定义,声明与原型
         对函数的“定义”和“声明”不是一回事。“定义”是指对函数功能的确立,包括指定函数名,函数值类型、形参类型、函数体等,它是一个完整的、独立的函数单位。而“声明”的作用则是把函数的名字、...
用户144403 2008-03-27 16:16
最支持台独的厂商,支持台独艺人
        昨天的台湾《联合报》发了一篇报道“大陆拒挺扁两台商入境…台湾的第一金融控股公司发言人蔡哲三昨天证实,该公司董事长谢寿夫、总经理蔡哲雄原定本月初赴北京,与中国银行签订策略联盟合约,但因大...
EE直播间
更多
我要评论
3
4
关闭 站长推荐上一条 /3 下一条