`

自定义通信协议-网摘

 
阅读更多

现在大部分的仪器设备都要求能过通过上位机软件来操作,这样方便调试,利于操作。其中就涉及到通信的过程。在实际制作的几个设备中,笔者总结出了通信程序的通用写法,包括上位机端和下位机端等。

1.自定义数据通信协议

  这里所说的数据协议是建立在物理层之上的通信数据包格式。所谓通信的物理层就是指我们通常所用到的RS232、RS485、红外、光纤、无线等等通信方式。在这个层面上,底层软件提供两个基本的操作函数:发送一个字节数据、接收一个字节数据。所有的数据协议全部建立在这两个操作方法之上。

通信中的数据往往以数据包的形式进行传送的,我们把这样的一个数据包称作为一帧数据。类似于网络通信中的TCPIP协议一般,比较可靠的通信协议往往包含有以下几个组成部分:帧头、地址信息、数据类型、数据长度、数据块、校验码、帧尾

  帧头和帧尾用于数据包完整性的判别,通常选择一定长度的固定字节组成,要求是在整个数据链中判别数据包的误码率越低越好。减小固定字节数据的匹配机会,也就是说使帧头和帧尾的特征字节在整个数据链中能够匹配的机会最小。通常有两种做法,一、减小特征字节的匹配几率。二、增加特征字节的长度。通常选取第一种方法的情况是整个数据链路中的数据不具有随即性,数据可预测,可以通过人为选择帧头和帧尾的特征字来避开,从而减小特征字节的匹配几率。使用第二种方法的情况更加通用,适合于数据随即的场合。通过增加特征字节的长度减小匹配几率,虽然不能够完全的避免匹配的情况,但可以使匹配几率大大减小,如果碰到匹配的情况也可以由校验码来进行检测,因此这种情况在绝大多说情况下比较可靠。

  地址信息主要用于多机通信中,通过地址信息的不同来识别不同的通信终端。在一对多的通信系统中,可以只包含目的地址信息。同时包含源地址和目的地址则适用于多对多的通信系统。 

  数据类型、数据长度和数据块是主要的数据部分。数据类型可以标识后面紧接着的是命令还是数据。数据长度用于指示有效数据的个数。

  校验码则用来检验数据的完整性和正确性。通常对数据类型、数据长度和数据块三个部分进行相关的运算得到。最简单的做法可是对数据段作累加和,复杂的也可以对数据进行CRC运算等等,可以根据运算速度、容错度等要求来选取。

 

2.上位机和下位机中的数据发送

  物理通信层中提供了两个基本的操作函数,发送一个字节数据则为数据发送的基础。数据包的发送即把数据包中的左右字节按照顺序一个一个的发送数据而已。当然发送的方法也有不同。

  在单片机系统中,比较常用的方法是直接调用串口发送单个字节数据的函数。这种方法的缺点是需要处理器在发送过程中全程参与,优点是所要发送的数据能够立即的出现在通信线路上,能够立即被接收端接收到。另外一种方法是采用中断发送的方式,所有需要发送的数据被送入一个缓冲区,利用发送中断将缓冲区中的数据发送出去。这种方法的优点是占用处理器资源小,但是可能出现需要发送的数据不能立即被发送的情况,不过这种时延相当的小。对于51系列单片机,比较倾向于采用直接发送的方式,采用中断发送的方式比较占用RAM资源,而且对比直接发送来说也没有太多的优点。以下是51系列单片机中发送单个字节的函数。

void SendByte(unsigned char ch)

{

     SBUF = ch;

     while(TI == 0);

     TI = 0;

}

  上位机中关于串口通信的方式也有多种,这种方式不是指数据有没有缓冲的问题,而是操作串口的方式不同,因为PC上数据发送基本上都会被缓冲后再发送。对于编程来说操作串口有三种方式,一、使用windows系统中自带的串口通信控件,这种方式使用起来比较简单,需要注意的是接收时的阻塞处理和线程机制。二、使用系统的API直接进行串口数据的读取,在windows和linux系统中,设备被虚拟为文件,只需要利用系统提供的API函数即可进行串口数据的发送和读取。三、使用串口类进行串口操作。在此只介绍windows环境下利用串口类编程的方式。

  CSerialPort是比较好用的串口类。它提供如下的串口操作方法:

  void WriteToPort(char* string, int len);

  串口初始化成功后,调用此函数即可向串口发送数据。为了避免串口缓冲所带来的延时,可以开启串口的冲刷机制。

 

3.下位机中的数据接收和协议解析

  下位机接收数据也有两种方式,一、等待接收,处理器一直查询串口状态,来判断是否接收到数据。二、中断接收。两种方法的优缺点在此前的一篇关于串口通信的文章中详细讨论过。得出的结论是采用中断接收的方法比较好。

  数据包的解析过程可以设置到不同的位置。如果协议比较简单,整个系统只是处理一些简单的命令,那么可以直接把数据包的解析过程放入到中断处理函数中,当收到正确的数据包的时候,置位相应的标志,在主程序中再对命令进行处理。如果协议稍微复杂,比较好的方式是将接收的数据存放于缓冲区中,主程序读取数据后进行解析。也有两种方式交叉使用的,比如一对多的系统中,首先在接收中断中解析“连接”命令,连接命令接收到后主程序进入设置状态,采用查询的方式来解析其余的协议。

  以下给出具体的实例。在这个系统中,串口的命令非常简单。所有的协议全部在串口中断中进行。数据包的格式如下:

  0x55, 0xAA, 0x7E, 0x12, 0xF0, 0x02, 0x23, 0x45, SUM, XOR, 0x0D

  其中0x55, 0xAA, 0x7E为数据帧的帧头,0x0D为帧尾,0x12为设备的目的地址,0xF0为源地址,0x02为数据长度,后面接着两个数据0x23, 0x45,从目的地址开始结算累加、异或校验和,到数据的最后一位结束。

        协议解析的目的,首先判断数据包的完整性,正确性,然后提取数据类型,数据等数据,存放起来用于主程序处理。代码如下:

 

if(state_machine == 0)     // 协议解析状态机

{

    if(rcvdat == 0x55)     // 接收到帧头第一个数据

        state_machine = 1;

    else

        state_machine = 0;    // 状态机复位

}

else if(state_machine == 1)

{

    if(rcvdat == 0xAA)     // 接收到帧头第二个数据

        state_machine = 2;

    else

        state_machine = 0;    // 状态机复位

}

else if(state_machine == 2)

{

    if(rcvdat == 0x7E)     // 接收到帧头第三个数据

        state_machine = 3;

     else

        state_machine = 0;    // 状态机复位

}

else if(state_machine == 3)

{

    sumchkm = rcvdat;     // 开始计算累加、异或校验和

    xorchkm = rcvdat;

    if(rcvdat == m_SrcAdr)    // 判断目的地址是否正确

        state_machine = 4;

    else

        state_machine = 0;

}

else if(state_machine == 4)

{

    sumchkm += rcvdat;

    xorchkm ^= rcvdat;

    if(rcvdat == m_DstAdr)    // 判断源地址是否正确

        state_machine = 5;

    else

        state_machine = 0;

   }

else if(state_machine == 5)

{

    lencnt = 0;        // 接收数据计数器

    rcvcount = rcvdat;      // 接收数据长度

    sumchkm += rcvdat;

    xorchkm ^= rcvdat;

    state_machine = 6;

}

else if(state _machine == 6 || state _machine == 7)

{

    m_ucData[lencnt++] = rcvdat;     // 数据保存

    sumchkm += rcvdat;

    xorchkm ^= rcvdat;

    if(lencnt == rcvcount)    // 判断数据是否接收完毕

        state_machine = 8;

    else

        state_machine = 7;

}

else if(state_machine == 8)

{

    if(sumchkm == rcvdat)    // 判断累加和是否相等

        state_machine = 9;

    else

        state_machine = 0;

}

else if(state_machine == 9)

{

    if(xorchkm == rcvdat)    // 判断异或校验和是否相等

        state_machine = 10;

    else

        state_machine = 0;

}

else if(state_machine == 10)

{

    if(0x0D == rcvdat)     // 判断是否接收到帧尾结束符

    {

        retval = 0xaa;    // 置标志,表示一个数据包接收到

    }

    state_machine = 0;     // 复位状态机

}

 

  此过程中,使用了一个变量state_machine作为协议状态机的转换状态,用于确定当前字节处于一帧数据中的那个部位,同时在接收过程中自动对接收数据进行校验和处理,在数据包接收完的同时也进行了校验的比较。因此当帧尾结束符接收到的时候,则表示一帧数据已经接收完毕,并且通过了校验,关键数据也保存到了缓冲去中。主程序即可通过retval的标志位来进行协议的解析处理。

  接收过程中,只要哪一步收到的数据不是预期值,则直接将状态机复位,用于下一帧数据的判断,因此系统出现状态死锁的情况非常少,系统比较稳定,如果出现丢失数据包的情况也可由上位机进行命令的补发,不过这种情况笔者还没有碰到。

  对于主程序中进行协议处理的过程与此类似,主程序循环中不断的读取串口缓冲区的数据,此数据即参与到主循环中的协议处理过程中,代码与上面所述完全一样。

 

4.上位机中的数据接收和命令处理 

  上位机中数据接收的过程与下位机可以做到完全一致,不过针对不同的串口操作方法有所不同。对于阻赛式的串口读函数,例如直接进行API操作或者调用windows的串口通信控件,最好能够开启一个线程专门用于监视串口的数据接收,每接收到一个数据可以向系统发送一个消息。笔者常用的CSerialPort类中就是这样的处理过程。CSerialPort打开串口后开启线程监视串口的数据接收,将接收的数据保存到缓冲区,并向父进程发送接收数据的消息,数据将随消息一起发送到父进程。父进程中开启此消息的处理函数,从中获取串口数据后就可以把以上的代码拷贝过来使用。 

  CSerialPort向父类发送的消息号如下:

  #define WM_COMM_RXCHAR WM_USER+7 // A character was received and placed in the input buffer.

  因此需要手动添加此消息的响应函数:

  afx_msg LONG OnCommunication(WPARAM ch, LPARAM port);

  ON_MESSAGE(WM_COMM_RXCHAR, OnCommunication)

  响应函数的具体代码如下:

LONG CWellInfoView::OnCommunication(WPARAM ch, LPARAM port)

{

     int retval = 0;

     rcvdat = (BYTE)ch;

     if(state_machine == 0)     // 协议解析状态机

    {

      if(rcvdat == 0x55)     // 接收到帧头第一个数据

          state_machine = 1;

      else

          state_machine = 0;    // 状态机复位

    }

    else if(state_machine == 1)

    {

      if(rcvdat == 0xAA)     // 接收到帧头第二个数据

          state_machine = 2;

      else

          state_machine = 0;    // 状态机复位

    ......

 

5.总结

  以上给出的是通信系统运作的基本雏形,虽然简单,但是可行。实际的通信系统中协议比这个要复杂,而且涉及到数据包响应、命令错误、延时等等一系列的问题,在这样的一个基础上可以克服这些困难并且实现出较为稳定可靠的系统

分享到:
评论

相关推荐

    Jquery合并单元格--网摘

    在网页开发中,jQuery 是一个非常流行的 JavaScript 库,它简化了 DOM 操作、事件处理以及动画效果。本文将探讨如何使用 jQuery 来合并表格中相同文本的相邻单元格,这是在展示数据时提高可读性和美观度的一种常见...

    百亿网摘完整无错版

    百亿网摘整站 后台功能 --------------------- 基本信息 管理站点的基本信息 栏目管理 栏目分类 网摘管理 已推荐网摘 推荐网摘 首页生成 重新生成页面 审核网摘 管理网摘 模版管理 模版管理 用户管理 管理员修改 ...

    网摘集合网摘集合网摘集合网摘集合

    网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合网摘集合...

    网摘精灵软件包1.5

    网摘自动提交工具 网摘精灵教程:网摘自动提交工具。 尊敬的站长:你好。 我们注意到您的网站放置了很多网摘提交代码。 你希望通过提交网摘来获取流量,对吗? 靠这种原始的手工提交,效率很低,对吗? 如果有...

    PHP免费网摘程序

    2、修改数据库配置文件,conn.php.修改网摘配置文件 include/setup.php。 3、后台路径admin/default.php 默认帐号密码 是 admin admin . 4、生成所有网摘静态页面程序为data.php。生成所有新网摘的程序为 datanew....

    加入网摘插件 v1.0 通用版

    JavaScript负责处理用户的交互,如触发下拉菜单的显示和隐藏,以及与服务器的通信,将页面信息提交到网摘站点。而CSS则用于控制插件的样式,使其与网站的现有设计保持一致,提供良好的用户体验。 源代码的提供对于...

    天空网摘 V120606.rar

    目标 利用网摘建站,打造最便捷的资讯服务!最适合于做小型SEO网站。 适用范围 新闻娱乐、专业学术、地方门户、行业资讯等相关网站建设。 参考范例:电子书藏家 初始化 注册用户:天空 密码:admin 登录:Login....

    php网摘程序

    在互联网信息爆炸的时代,网摘程序为用户提供了方便的工具,帮助他们保存感兴趣的内容,便于日后查阅和分享。这种程序通常包含用户登录系统、内容收藏功能、分类管理以及搜索功能等,旨在提升用户的网络浏览体验。 ...

    天空网摘 v120606

    网摘建站具有技术自主容易、运营维护简单等性价比的优势,并能高效快捷的整合网络诸如论坛、博客、资讯、图片、视频、下载等各类资源来。当然,网摘应用也具有内容原创性不强,互动交流性不够的特点,这就需要在网站...

    PHP免费网摘程序(源代码)

    2、修改数据库配置文件,conn.php.修改网摘配置文件 include/setup.php。 3、后台路径admin/default.php 默认帐号密码 是 admin admin . 4、生成所有网摘静态页面程序为data.php。生成所有新网摘的程序为 datanew....

    PW广告添加方式网摘插件

    "PW广告添加方式网摘插件"是一款专为网站管理者设计的工具,旨在方便地在网页上添加广告,尤其适用于那些希望通过广告展示来增加收入或推广产品的网站。这款插件的特点在于其简单易用的安装过程和灵活的广告投放位置...

    天空网摘 v20110624.rar

    程序说明网摘等级:删除 私有 验证 公开 高级 推荐 精华用户等级:删除 锁定 验证 普通 高级 贵宾 操作员 管理员各个等级的用户分别对应拥有各个等级的网摘权限,其中操作员可以添加“图片”类型的网摘,管理员可以...

    咖啡屋网摘联盟程序 v6.0 终极无错误版

    《咖啡屋网摘联盟程序 v6.0 终极无错误版》正是在这样的背景下应运而生,它不仅为用户提供了一个便捷的网摘分享平台,更是将社交化功能融入到知识管理之中,让用户在收集和整理网页资源的同时,也能享受到交流互动的...

    天空网摘 v120606-ASP源码.zip

    4. **灵活性和可扩展性**: 通过引入自定义的ActiveX组件或.NET Framework组件,可以增强ASP的功能,实现更复杂的业务逻辑。 在使用【天空网摘 v120606-ASP源码.zip】时,需要注意以下几点: 1. **解压密码**: 提供...

    ASP实例开发源码-麦布简易网摘源码 asp版.zip

    【ASP实例开发源码-麦布简易网摘源码 ASP版】是一个针对初学者和有一定经验的ASP开发者设计的学习资源,它提供了完整的源代码,帮助开发者理解和实践ASP(Active Server Pages)技术。ASP是一种服务器端脚本环境,常...

    CSDN网摘右链添加

    总的来说,"CSDN网摘右链添加"是一个为了提升用户体验而进行的小型自定义设置。通过导入`.reg`文件,用户能够方便地在浏览器中实现CSDN网摘的快速存档和管理,这体现了Windows系统高度的个性化定制能力。同时,这也...

    23个收藏到网摘的16x16图标及代码

    标题中的“23个收藏到网摘的16x16图标及代码”指的是一个包含23个16像素乘16...开发者可以下载这个资源包,根据提供的代码示例,自定义并集成到自己的网站中,让用户能够轻松地将感兴趣的内容保存到他们的网摘服务。

    23个收藏到网摘的16x16图标及代码.rar

    标题中的“23个收藏到网摘的16x16图标及代码.rar”表明这是一个包含23个图标以及对应代码的压缩文件,这些图标尺寸为16像素乘以16像素,通常用于网页设计或者软件开发中。网摘,也被称为书签服务或社交书签,是用户...

Global site tag (gtag.js) - Google Analytics