博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
经过验证的串口通信协议及其问题概述
阅读量:6619 次
发布时间:2019-06-25

本文共 2345 字,大约阅读时间需要 7 分钟。

  hot3.png

输入图片说明

串口是我的最爱,成本低,容易实现,连接简单方便。在我设计的硬件中,无一不配置一个串口,在主要功能完成之后,我会在计算机上再编写一个终端软件。这个软件可不只用来监控,我用这个软件完全控制硬件的所有功能,直到能够监测到硬件尽可能多的状态信息,只要一看这些信息,我就能知道硬件的工作状态如何,故障可能发生在哪里。 近来发现论坛里提问频率较高、问题也五花八门的问题之一就是串口通信问题,因此想在这里写点心得,也只是一点心得,对于网上很容易查到的信息,这里就不谈了。

** 问题一:缓冲区大小的问题。**

现在绝大多数CPU都包含串口功能,同时也为串口配置了FIFO缓冲区,对于FIFO缓冲区的使用存在着些误区。单就串口来说,通信也分为很多方式,缓冲区大小影响较大的有两种:突发通信和连续的数据流通信。突发通信的主要特点是:数据量小、持续时间短、通信发生时间不确定,通常的设备状态监控就属于这一类型。相对的数据流通信的特点是:数据量大、持续时间长等,如图像通信、音频采集等。对于突发通信,应该不使用FIFO缓冲区,或者把FIFO大小设置为1。因为在如果把缓冲区大小设置为应该接收的数据的大小,那么发生通信错误如丢失数据时就不会产生中断,接收方会认为未发生通信事件,而发送方只能靠超时认为通信中断。这和接收到数据并判断出通信数据发生错误的性质是不同的,如果接收到数据我们就可以判断出发生错误的原因,如果接收不数据,那么怎么判断,硬件连接?FIFO设置?软件错误?软件存在隐藏的bug?所以对于突发通信不应该使用FIFO。对于数据流通信,由于数据量大,如果还是用单字节产生中断的方式来接收,那CPU就会疲于处理串口中断事件,而且这种情况下单字节是无法作数据处理的,而使用FIFO缓冲区,一次接收多个字节后再产生中断就会节约CPU时间用于数据处理,这里就不多说了。

** 问题二:串口通信软件的编写。**

好的硬件配上好的软件才能有好的效果,我发现有些编写了几年软件的人串口通信软件也写不好,我这里就把这几年编写串口通信软件的一点小小的心得拿出来分享。我们常用的TCP/IP协议是一种分层的协议,在编写串口通信软件时也可以借鉴这种分层协议的编程思想。另外对于协议的使用,不同的人有不同的看法,区别较大,这里只介绍相当于物理层和协议层之间的数据链路层程序。由于没有使用FIFO功能,就要在内存中划出一块内存作为发送缓冲区,由发送中断服务程序把数据发送到接收方。代码如下:
// 定义发送缓冲区大小,可以根据系统RAM的大小的数据量来定义#define U0TX_BUF_SIZE 1024// 定义发送缓冲区char u0tx_buf[U0TX_BUF_SIZE];// 定义串口发射忙标志,当串口正在发送数据且发送缓冲区不空时为1char u0tx_busy_bit;// 缓冲区中数据长度unsigned int u0tx_length;// 发送缓冲区数据指针,指向下一个发送的数据unsigned int u0tx_buf_pointer;// 准备发送数据// data是要发送的数据// length数据的长度char u0tx_send_packet( char *data, int length ){ int i; // u0tx_busy_bit为1时表示串口正在使用,正在使用则退出, // 避免后续的数据覆盖正在发送的数据 if( u0tx_busy_bit==1 ) {  // 返回0x01表示未能进入串口发送操作  return 0x01; } // 把要发送的数据转移到发送缓冲区 for( i=0; i
=U0_RX_BUFF_SIZE ) u0rx_in_pointer = 0; // 标志置1,表示接收缓冲区中有数据 RX0_Flag = 1;}// 从串口接收缓冲区中读取一个字节数据char u0rx_get_data(void){ // 临时变量,保存从接收缓冲区中读取的数据 char c; // 如果出队列指针不等于入队列指针且有效,表示接收缓冲区在有数据,否则为空 if( (u0rx_out_pointer != u0rx_in_pointer) && (u0rx_out_pointer < U0RX_BUFF_SIZE) ) { // 读取一个字节的数据 c = u0rx_buf[u0rx_out_pointer]; // 出队列指针加1,指向下一个存储单元 u0rx_out_pointer ++; // 如果出队列指针超出了缓冲区尾地址,则指向首地址 if( u0rx_out_pointer>=U0_RX_BUFF_SIZE ) u0rx_out_pointer = 0; // 数据有效标志置1 u0rx_data_valid_flag = 1; // 如果出队列指针等于入队列指针,表示接收缓冲区已空 if( u0rx_out_pointer==u0rx_in_pointer ) u0rx_Flag = 0; // 返回读取的数据 return c; } else { // 标志置1,表示接收缓冲区为空 u0rx_Flag = 0; // 数据有效标志清零 u0rx_data_valid_flag = 0; // 返回0xff,无意义 return 0xff; }}
这部分代码主要是针对无操作系统的单片机软件编写的,但我用的是流通信协议,所以在C#中也经常这样使用。

转载于:https://my.oschina.net/boonya/blog/827412

你可能感兴趣的文章
IT讲师韩顺平:创业不易,尚硅谷延续教育初心
查看>>
IntelliJ IDEA 插件 阿里巴巴Java开发手册
查看>>
利用nmap对Mongodb Redis未授权访问测试
查看>>
CakePHP
查看>>
我的友情链接
查看>>
编译mysql5.6.27
查看>>
搭建centos6.7网站服务器记录
查看>>
Release版本调用ffmpeg av_register_all程序崩溃
查看>>
Referenced management pack not found
查看>>
jquery中data函数的用法示例
查看>>
巧用strtotime函数计算日期
查看>>
JVM中java对象的生命周期
查看>>
mysql 查看连接数,状态
查看>>
JFinal集成YUI Compressor压缩合并JS和CSS
查看>>
windows下的Oracle卸载
查看>>
sqlserver查看死锁的存储过程
查看>>
在VirtualBox中的CentOS 6.3下安装VirtualBox增强包(GuestAd...
查看>>
Java开发中的23种设计模式详解(转)
查看>>
Tomcat配置日志生产功能
查看>>
移植Qt与Tslib到X210开发板的体会
查看>>