串口初始化配置

在基于AT91的嵌入式linux中接收串口数据时,发现对于接收的数据经常出现接收不完整的现象。一帧的数据可能会被当做两帧接收,导致对于一帧数据接收出现问题。虽然这种情况在一般情况下,并不是经常出现,但是只要数据量稍微大一些,情况就会出现。

于是仔细看了程序中关于串口配置这一块的程序,

[cpp] view plaincopy
  1. int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)  
  2. {  
  3. struct termios oldtio;  
  4.     struct termios newtio;  
  5.   
  6.     if( tcgetattr(fd, &oldtio)  !=  0) {  
  7.         perror("SetupSerial 1");  
  8.         return -1;  
  9.     }  
  10.       
  11.     bzero( &newtio, sizeof( newtio ));  
  12.     newtio.c_cflag  |=  CLOCAL | CREAD;  
  13.     newtio.c_cflag &= ~CSIZE;  
  14.   
  15.     switch( nBits )  
  16.     {  
  17.         case 7:  
  18.             newtio.c_cflag |= CS7;  
  19.         break;  
  20.         case 8:  
  21.             newtio.c_cflag |= CS8;  
  22.         break;  
  23.     }  
  24.   
  25.     switch( nEvent )  
  26.     {  
  27.         case 'O':  
  28.         newtio.c_cflag |= PARENB;  
  29.         newtio.c_cflag |= PARODD;  
  30.         newtio.c_iflag |= INPCK ;  
  31.         break;  
  32.         case 'E':  
  33.         newtio.c_iflag |= INPCK ;  
  34.         newtio.c_cflag |= PARENB;  
  35.         newtio.c_cflag &= ~PARODD;  
  36.         break;  
  37.         case 'N':   
  38.         newtio.c_cflag &= ~PARENB;  
  39.         break;  
  40.     }  
  41.   
  42.     switch( nSpeed )  
  43.     {  
  44.         case 2400:  
  45.             cfsetispeed(&newtio, B2400);  
  46.             cfsetospeed(&newtio, B2400);  
  47.         break;  
  48.         case 4800:  
  49.             cfsetispeed(&newtio, B4800);  
  50.             cfsetospeed(&newtio, B4800);  
  51.         break;  
  52.         case 9600:  
  53.             cfsetispeed(&newtio, B9600);  
  54.             cfsetospeed(&newtio, B9600);  
  55.         break;  
  56.         case 115200:  
  57.             cfsetispeed(&newtio, B115200);  
  58.             cfsetospeed(&newtio, B115200);  
  59.         break;  
  60.         case 460800:  
  61.             cfsetispeed(&newtio, B460800);  
  62.             cfsetospeed(&newtio, B460800);  
  63.         break;  
  64.         default:  
  65.             cfsetispeed(&newtio, B9600);  
  66.             cfsetospeed(&newtio, B9600);  
  67.         break;  
  68.     }  
  69.     if( nStop == 1 )  
  70.         newtio.c_cflag &=  ~CSTOPB;  
  71.     else if ( nStop == 2 )  
  72.         newtio.c_cflag |=  CSTOPB;  
  73.   
  74.     newtio.c_cc[VTIME]  = 0;  
  75.     newtio.c_cc[VMIN] = 1;  
  76.       
  77.     tcflush(fd,TCIFLUSH);  
  78.     if((tcsetattr(fd,TCSANOW,&newtio))!=0)  
  79.     {  
  80.         perror("com set error");  
  81.         return -1;  
  82. }  
  83.       
  84.     tcflush(fd,TCIFLUSH);//The `tcflush' function is used to clear the input and/or output  
  85.     return 0;   
  86. }  


在分析完程序后发现可能导致出问题的地方:在使用oldtio读取串口配置后,却没有将其复制给newtio,并且将newtio清零,这造成下边的设置操作,修改了一些原来的设置。

根据程序修改VTIMEVMIN可推知这里要使用非规范方式,

根据APUE可知由VTIMEVMIN的设置共可以有四种选择

AVTIME > 0, VMIN > 0

B: VTIME = 0, VMIN > 0

C: VTIME > 0, VMIN = 0

D: VTIME = 0, VMIN = 0

由程序修改的值

 newtio.c_cc[VTIME]  = 0;
 newtio.c_cc[VMIN]   = 1;

可知这里要设置为第二种方式:只有接收到MIN个字节数据,read才返回;否则,read将阻塞。

因为这里没有将oldtio复制给newtio所以这里的ICANON标识一定没有设置,所以是处于非规范模式下。这里的VTIME, VMIN对这里的设置也是有效的。

但是这样难免修改一些我们没有注意的选项,根据这里的设置,结合APUE中的示例程序,发现APUE中将终端设置为原始模式(raw modle)与这里的设置较为相似,于是想采用APUE中的部分参数设置,来修改此处的程序。

==================APUE中 put terminal into a raw modle ===========================

[cpp] view plaincopy
  1. int  
  2. tty_raw(int fd) /* put terminal into a raw mode */  
  3. {  
  4.  int err;  
  5.  struct termios buf;  
  6.   
  7.   
  8.  if (ttystate != RESET) {  
  9.  errno = EINVAL;  
  10.  return(-1);  
  11.  }  
  12.  if (tcgetattr(fd, &buf) < 0)  
  13.  return(-1);  
  14.  save_termios = buf; /* structure copy */  
  15.   
  16.  /* 
  17.   * Echo off, canonical mode off, extended input 
  18.   * processing off, signal chars off. 
  19.   */  
  20.  buf.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);  
  21.   
  22.  /* 
  23.   * No SIGINT on BREAK, CR-to-NL off, input parity 
  24.   * check off, don't strip 8th bit on input, output 
  25.   * flow control off. 
  26.   */  
  27.  buf.c_iflag &= ~(BRKINT | ICRNL | INPCK | ISTRIP | IXON);  
  28.   
  29.  /* 
  30.   * Clear size bits, parity checking off. 
  31.   */  
  32.  buf.c_cflag &= ~(CSIZE | PARENB);  
  33.   
  34.  /* 
  35.   * Set 8 bits/char. 
  36.   */  
  37.  buf.c_cflag |= CS8;  
  38.   
  39.  /* 
  40.   * Output processing off. 
  41.   */  
  42.  buf.c_oflag &= ~(OPOST);  
  43.   
  44.  /* 
  45.   * Case B: 1 byte at a time, no timer. 
  46.   */  
  47.  buf.c_cc[VMIN] = 1;  
  48.  buf.c_cc[VTIME] = 0;  
  49.  if (tcsetattr(fd, TCSAFLUSH, &buf) < 0)  
  50.  return(-1);  
  51.   
  52.  /* 
  53.   * Verify that the changes stuck.  tcsetattr can return 0 on 
  54.   * partial success. 
  55.   */  
  56.  if (tcgetattr(fd, &buf) < 0) {  
  57.  err = errno;  
  58.  tcsetattr(fd, TCSAFLUSH, &save_termios);  
  59.  errno = err;  
  60.  return(-1);  
  61.  }  
  62.  if ((buf.c_lflag & (ECHO | ICANON | IEXTEN | ISIG)) ||  
  63.    (buf.c_iflag & (BRKINT | ICRNL | INPCK | ISTRIP | IXON)) ||  
  64.    (buf.c_cflag & (CSIZE | PARENB | CS8)) != CS8 ||  
  65.    (buf.c_oflag & OPOST) || buf.c_cc[VMIN] != 1 ||  
  66.    buf.c_cc[VTIME] != 0) {  
  67.  /* 
  68.   * Only some of the changes were made.  Restore the 
  69.   * original settings. 
  70.   */  
  71.  tcsetattr(fd, TCSAFLUSH, &save_termios);  
  72.  errno = EINVAL;  
  73.  return(-1);  
  74.  }  
  75.   
  76.  ttystate = RAW;  
  77.  ttysavefd = fd;  
  78.  return(0);  
  79. }  

从英文注释,可以了解到,各个参数的具体意义,

同时参考博客http://blog.csdn.net/awei_xu/article/details/3725329 中的红色标记部分设置来补充,

[cpp] view plaincopy
  1.  if((fd = open(dev,O_RDWR | O_NOCTTY | O_NDELAY)) == -1)   
  2.   /*----------------------  重要----------------------*/   
  3.     //保证本程序不会成为端口的所有者,从而妨碍控制工作和挂起信号.  
  4.     opt.c_cflag     |= (CLOCAL | CREAD);  
  5. //选择原始输入方式: 原始输入方式是不经处理的.  
  6. opt.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);   
  7. //输出不经处理  
  8. opt.c_oflag  &= ~OPOST;    
  9.    //取消软件流控制(不设置可能存在丢码)  
  10.    opt.c_iflag &= ~(IXON | IXOFF | IXANY);  /*----------------------------------------------------*/  

修改后的程序:

[cpp] view plaincopy
  1. int set_opt(int fd,int nSpeed, int nBits, char nEvent, int nStop)  
  2. {  
  3.     struct termios oldtio;  
  4. struct termios newtio;  
  5.     if( tcgetattr(fd, &oldtio)  !=  0) {  
  6.         perror("SetupSerial 1");  
  7.         return -1;  
  8.     }  
  9.     newtio = oldtio;  
  10. /*************Debug*******************/  
  11.     newtio.c_cflag  |=  CLOCAL | CREAD;  
  12.     newtio.c_cflag  &= ~(CSIZE | PARENB);  
  13.     newtio.c_lflag  &= ~(ICANON | ISIG | ECHO | ECHOE | IEXTEN);  
  14.     newtio.c_oflag  &= ~OPOST;  
  15.     newtio.c_iflag  &= ~(IXON | IXOFF | IXANY | ICRNL | BRKINT  
  16.                        | INPCK | ISTRIP);  
  17. /***************************************/  
  18.     switch( nBits )  
  19.     {  
  20.         case 7:  
  21.             newtio.c_cflag |= CS7;  
  22.             break;  
  23.         case 8:  
  24.             newtio.c_cflag |= CS8;  
  25.             break;  
  26.     }  
  27.     switch( nEvent )  
  28.     {  
  29.         case 'O':  
  30.             /*odd parity*/  
  31.             newtio.c_iflag |= INPCK ;  
  32.             newtio.c_cflag |= PARENB;  
  33.             newtio.c_cflag |= PARODD;  
  34.             break;  
  35.         case 'E':  
  36.             /*even parity*/  
  37.             newtio.c_iflag |= INPCK ;  
  38.             newtio.c_cflag |= PARENB;  
  39.             newtio.c_cflag &= ~PARODD;  
  40.             break;  
  41.         case 'N':  
  42.             /*no parity*/  
  43.             newtio.c_cflag &= ~PARENB;  
  44.             break;  
  45.     }  
  46.     switch( nSpeed )  
  47.     {  
  48.         case 2400:  
  49.             cfsetispeed(&newtio, B2400);  
  50.             cfsetospeed(&newtio, B2400);  
  51.             break;  
  52.         case 4800:  
  53.             cfsetispeed(&newtio, B4800);  
  54.             cfsetospeed(&newtio, B4800);  
  55.             break;  
  56.         case 9600:  
  57.             cfsetispeed(&newtio, B9600);  
  58.             cfsetospeed(&newtio, B9600);  
  59.             break;  
  60.         case 115200:  
  61.             cfsetispeed(&newtio, B115200);  
  62.             cfsetospeed(&newtio, B115200);  
  63.             break;  
  64.         case 460800:  
  65.             cfsetispeed(&newtio, B460800);  
  66.             cfsetospeed(&newtio, B460800);  
  67.             break;  
  68.         default:  
  69.             cfsetispeed(&newtio, B9600);  
  70.             cfsetospeed(&newtio, B9600);  
  71.             break;  
  72.     }  
  73.       
  74.     if( nStop == 1 )  
  75.         newtio.c_cflag &=  ~CSTOPB;  
  76.     else if ( nStop == 2 )  
  77.         newtio.c_cflag |=  CSTOPB;  
  78.     newtio.c_cc[VTIME]  = 0;  
  79.     newtio.c_cc[VMIN]   = 1;  
  80.       
  81.     tcflush(fd,TCIFLUSH);  
  82.     if((tcsetattr(fd,TCSANOW,&newtio))!=0)  
  83.     {  
  84.         perror("com set error");  
  85.         return -1;  
  86.     }  
  87. tcflush(fd,TCIFLUSH);  
  88. return 0;  
  89. }  

修改完的初步测试中发现有些地方还有问题,

待下一步测试,找到原因所在。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/401795.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Angularjs 通过asp.net web api认证登录

Angularjs 通过asp.net web api认证登录 Angularjs利用asp.net mvc提供的asp.net identity&#xff0c;membership实现居于数据库的用户名/密码的认证登录 环境 Vs.net 2013 Asp.net mvc web api Individual user accounts Angularjs Underscore 新建一个asp.net mvc web api …

PANIC: Unreachable code reached.

为什么80%的码农都做不了架构师&#xff1f;>>> Caused by: java.lang.RuntimeException: PANIC: Unreachable code reached.at cryptix.jce.provider.cipher.BlockCipher.engineGetParameters(BlockCipher.java:244)at javax.crypto.Cipher.checkCryptoPerm(Ciphe…

Linux VTIME VMIN的作用以及使用有效的前提

前提条件 1、fdcom open(ptty, O_RDWR | O_NOCTTY); //other attributions default /*Canonical Input*/ //termios_new.c_lflag | (ICANON | ECHO | ECHOE); 2、/*Raw Input*/ //termios_new.c_lflag & ~(ICANON | ECHO | ECHOE | ISIG); 下面解释&#xff1a; op…

php中使用httpclient

下载HttpClient.phphttp://yunpan.cn/QiD93DMb6vsEH &#xff08;提取码&#xff1a;ec47&#xff09; 下载之后&#xff0c;将该文件放到项目文件目录下新建class目录下 然后在php中使用: 1 <?php2 require_once class/HttpClient.php;3 $params array(4 coords>114.3…

Hi3520d uboot uImage rootfs 移植与升级

安装、升级hi3520DDEMO板开发开发环境 # 如果您使用的hi3520D的DEMO板&#xff0c;可以按照以下步骤烧写u-boot&#xff0c;内核以及文件系统&#xff0c;以下步骤均使用网络来更新。 # 通常&#xff0c;您拿到的单板中已经有u-boot&#xff0c;如果没有的话&#xff0…

面向切面编程(转)

面向切面编程&#xff08;AOP是Aspect Oriented Program的首字母缩写&#xff09; &#xff0c;我们知道&#xff0c;面向对象的特点是继承、多态和封装。而封装就要求将功能分散到不同的对象中去&#xff0c;这在软件设计中往往称为职责分配。实际上也就是说&#xff0c;让不同…

Hi3520d uImage制作 uboot制作 rootfs制作

首次安装SDK 1、hi3520D SDK包位置 在"hi3520D_V100R001***/01.software/board"目录下&#xff0c;您可以看到一个 hi3520D_SDK_Vx.x.x.x.tgz 的文件&#xff0c; 该文件就是hi3520D的软件开发包。 2、解压缩SDK包 在linux服务器上&#xff08;或者一台装有…

[LeetCode Online Judge]系列-求二维平面内在一条直线上的最大点数

2019独角兽企业重金招聘Python工程师标准>>> Max Points on a Line Given n points on a 2D plane, find the maximum number of points that lie on the same straight line. 题目是:在2D平面内给N个点,求最多多少个点在一个直线上. 以下是AC的解决方式: /*** Defi…

Hi3520D UART2和UART3是如何加载到内核的

Hi3520D的UART驱动位于linux-3.0.y/drivers/tty/serial/amba-pl011.c 添加UART2和UART3需要修改的文件为&#xff1a;linux-3.0.y/arch/arm/mach-hi3520d/core.c和linux-3.0.y/arch/arm/mach-hi3520d/include/mach/irqs.h两个文件&#xff1b; 首先修改 core.c文件&#xff0c;…

Linux-C实现GPRS模块发送短信

“GSM模块&#xff0c;是将GSM射频芯片、基带处理芯片、存储器、功放器件等集成在一块线路板上&#xff0c;具有独立的操作系统、GSM射频处理、基带处理并提供标准接口的功能模块。GSM模块根据其提供的数据传输速率又可以分为GPRS模块、EDGE模块和纯短信模块。短信模块只支持语…

【思科】GNS3模拟静态NAT/动态NAT

实验拓扑&#xff1a;实验目的&#xff1a;利用NAT技术&#xff0c;使C1,C2能与R2连通。实验步骤&#xff1a;如图配置好IP地址&#xff0c;为C1,C2指定网关192.168.1.1。2.在R1上配置静态NAT.假定为C1配置公网地址12.0.0.3&#xff0c;来实现与R2的连通。这样&#xff0c;静态…

Text模式和PDU模式的区别

发送短消息常用Text和PDU(Protocol Data Unit&#xff0c;协议数据单元)模式。使用Text模式收发短信代码简单&#xff0c;实现起来十分容易&#xff0c;但最大的缺点是不能收发中文短信&#xff1b;而PDU模式不仅支持中文短信&#xff0c;也能发送英文短信。PDU模式收发短信…

[家里蹲大学数学杂志]第041期中山大学数计学院 2008 级数学与应用数学专业《泛函分析》期末考试试题 A...

1 ( 10 分 ) 设 $\mathcal{X}$ 是 Banach 空间, $f$ 是 $\mathcal{X}$ 上的线性泛函. 求证: $f\in \mathcal{L}(\mathcal{X})$ 的充分必要条件是 \[ N(f)\{ x\in \mathcal{X};\ f(x)0 \} \] 是 $\mathcal{X}$ 的闭线性子空间. 证明: 必要性. 设 $N(f)\ni x_n\to x$, 则 $$\bex …

PUD编码规则

共有三种方式来发送和接收SMS信息&#xff1a;Block Mode, Text Mode和PDU Mode。其中PDU Mode被所有手机支持&#xff0c;可以使用任何字符集&#xff0c;这也是手机默认的编码方式。 发送短消息常用Text和PDU(Protocol Data Unit&#xff0c;协议数据单元)模式。使用Text模式…

xml 解析

2019独角兽企业重金招聘Python工程师标准>>> 各大数据接口大全&#xff1a;http://blog.sina.com.cn/s/articlelist_2127818045_10_1.html package com.test.junit; import static org.junit.Assert.*; import java.io.ByteArrayInputStream;import java.io.InputSt…

3.1 采购管理规划

3.1.1 采购业务管理规划 通过企业采购业务管理规划&#xff0c;从而引入完全价值采购体系&#xff0c;建立企业的战略性采购或电子化采购流程&#xff0c;进行合同管理&#xff0c;收货管理和使用&#xff0c;采购结算&#xff0c;降低总体采购成本&#xff1b; 整合企业的采…

能吹是多么的重要

联合利华引进了一条香皂包装生产线&#xff0c;结果发现这条生产线有个缺陷&#xff1a;常常会有盒子里没装入香皂。总不能把空盒子卖给顾客啊&#xff0c;他们只得请了一个学自动化的博士后设计一个方案来分拣空的香皂盒。博士后拉起了一个十几人的科研攻关小组&#xff0c;综…

深入理解Linux守护进程

深入理解Linux守护进程Linux服务器在启动时需要启动很多系统服务&#xff0c;它们向本地和网络用户提供了Linux的系统功能接口&#xff0c;直接面向应用程序和用户。提供这些服务的程序是由运行在后台的守护进程&#xff08;daemons&#xff09;来执行的。守护进程是生存期长的…

【翻译】Ext JS 4——Ajax和Rest代理处理服务器端一场和消息的方法

原文&#xff1a;EXTJS4 - Handle Server-side exceptions and message from an Ajax or Rest proxy作者&#xff1a;Raja可能要处理的情况&#xff1a;success&#xff08;成功&#xff09;——Ext处理failure&#xff08;失败&#xff09;&#xff0c;由于通讯问题——Ext处理…

Apache下PHP Loaded Configuration File None 解决方法

解决方法可在apache配置文件中增加 PHPIniDir “The path to your php.ini”&#xff0c; 比如&#xff1a;PHPIniDir "/usr/local/php/etc/php.ini"重启apache。 确保PHPIniDir在loadModule php5_module之前 转载于:https://blog.51cto.com/zrer90/1421464