文档章节

Linux串口通信

wangjian19
 wangjian19
发布于 2014/02/21 15:22
字数 2343
阅读 63
收藏 0
点赞 0
评论 0

1.打开串口

       与其他的关于设备编程的方法一样,在Linux下,操作、控制串口也是通过操作起设备文件进行的。在Linux下,串口的设备文件是/dev/ttyS0或/dev/ttyS1等。因此要读写串口,我们首先要打开串口:

       char *dev  = "/dev/ttyS0"; //串口1

       int    fd = open( dev, O_RDWR );

        //| O_NOCTTY | O_NDELAY      

       if (-1 == fd)   

       {                  

              perror("Can't Open Serial Port");

              return -1;       

       }    

       else 

              return fd;

      

2.         设置串口速度

       打开串口成功后,我们就可以对其进行读写了。首先要设置串口的波特率:

       int speed_arr[] = { B38400, B19200, B9600, B4800, B2400, B1200, B300,

                      B38400, B19200, B9600, B4800, B2400, B1200, B300, };

int name_arr[] = {38400,  19200,  9600,  4800,  2400,  1200,  300, 38400, 

                                   19200,  9600, 4800, 2400, 1200,  300, };

void set_speed(int fd, int speed){

       int   i;

       int   status;

       struct termios   Opt;

       tcgetattr(fd, &Opt);

       for ( i= 0;  i < sizeof(speed_arr) / sizeof(int);  i++) {

              if  (speed == name_arr[i]) {    

                     tcflush(fd, TCIOFLUSH);    

                     cfsetispeed(&Opt, speed_arr[i]); 

                     cfsetospeed(&Opt, speed_arr[i]);  

                     status = tcsetattr(fd, TCSANOW, &Opt); 

                     if  (status != 0) {       

                            perror("tcsetattr fd"); 

                            return;    

                     }   

                     tcflush(fd,TCIOFLUSH);  

              } 

       }

}

3.         设置串口信息

这主要包括:数据位、停止位、奇偶校验位这些主要的信息。

      /**

*@brief     设置串口数据位,停止位和效验位

*@param  fd     类型  int  打开的串口文件句柄

*@param  databits 类型  int 数据位   取值 为 7 或者8

*@param  stopbits 类型  int 停止位   取值为 1 或者2

*@param  parity  类型  int  效验类型 取值为N,E,O,,S

*/

int set_Parity(int fd,int databits,int stopbits,int parity)

{

       struct termios options;

       if  ( tcgetattr( fd,&options)  !=  0) {

              perror("SetupSerial 1");    

              return(FALSE); 

       }

       options.c_cflag &= ~CSIZE;

       options.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);  /*Input*/

       options.c_oflag  &= ~OPOST;   /*Output*/

 

       switch (databits) /*设置数据位数*/

       {  

       case 7:          

              options.c_cflag |= CS7;

              break;

       case 8:    

              options.c_cflag |= CS8;

              break;  

       default:   

              fprintf(stderr,"Unsupported data size\n"); return (FALSE); 

       }

switch (parity)

{  

       case 'n':

       case 'N':   

              options.c_cflag &= ~PARENB;   /* Clear parity enable */

              options.c_iflag &= ~INPCK;     /* Enable parity checking */

              break; 

       case 'o':  

       case 'O':    

              options.c_cflag |= (PARODD | PARENB); /* 设置为奇效验*/ 

              options.c_iflag |= INPCK;             /* Disnable parity checking */

              break; 

       case 'e': 

       case 'E':  

              options.c_cflag |= PARENB;     /* Enable parity */   

              options.c_cflag &= ~PARODD;   /* 转换为偶效验*/    

              options.c_iflag |= INPCK;       /* Disnable parity checking */

              break;

       case 'S':

       case 's':  /*as no parity*/  

           options.c_cflag &= ~PARENB;

              options.c_cflag &= ~CSTOPB;break; 

       default:  

              fprintf(stderr,"Unsupported parity\n");   

              return (FALSE); 

       } 

/* 设置停止位*/ 

switch (stopbits)

{  

       case 1:   

              options.c_cflag &= ~CSTOPB; 

              break; 

       case 2:   

              options.c_cflag |= CSTOPB; 

          break;

       default:   

               fprintf(stderr,"Unsupported stop bits\n"); 

               return (FALSE);

}

/* Set input parity option */

if (parity != 'n')  

       options.c_iflag |= INPCK;

tcflush(fd,TCIFLUSH);

options.c_cc[VTIME] = 0; /* 设置超时0 seconds*/  

options.c_cc[VMIN] = 13; /* define the minimum bytes data to be readed*/

if (tcsetattr(fd,TCSANOW,&options) != 0)  

{

       perror("SetupSerial 3");  

       return (FALSE); 

}

return (TRUE); 

}

在上述代码中,有两句话特别重要:

options.c_cc[VTIME] = 0; /* 设置超时0 seconds*/  

options.c_cc[VMIN] = 13; /* define the minimum bytes data to be readed*/

这两句话决定了对串口读取的函数read()的一些功能。我将着重介绍一下他们对read()函数的影响。

       对串口操作的结构体是

Struct{

       tcflag_t   c_iflag;    /*输入模式标记*/

       tcflag_t   c_oflag;   /*输出模式标记*/

       tcflag_t   c_cflag;   /*控制模式标记*/

       tcflag_t   c_lflag;    /*本地模式标记*/

       cc_t        c_line;     /*线路规程*/

       cc_t        c_cc[NCCS];  /*控制符号*/

};

其中cc_t       c_line只有在一些特殊的系统程序(比如,设置通过tty设备来通信的网络协议)中才会用。在数组c_cc中有两个下标(VTIME和VMIN)对应 的元素不是控制符,并且只是在原始模式下有效。只有在原始模式下,他们决定了read()函数在什么时候返回。在标准模式下,除非设置了 O_NONBLOCK选项,否则只有当遇到文件结束符或各行的字符都已经编辑完毕后才返回。

控制符VTIME和VMIN之间有着复杂的关系。VTIME定义要求等待的零到几百毫秒的时间量(通常是一个8位的unsigned char变量,取值不能大于cc_t)。VMIN定义了要求等待的最小字节数(不是要求读的字节数——read()的第三个参数才是指定要求读的最大字节 数),这个字节数可能是0。

l         如果VTIME取0,VMIN定义了要求等待读取的最小字节数。函数read()只有在读取了VMIN个字节的数据或者收到一个信号的时候才返回。

l         如果VMIN取0,VTIME定义了即使没有数据可以读取,read()函数返回前也要等待几百毫秒的时间量。这时,read()函数不需要像其通常情况那样要遇到一个文件结束标志才返回0。

l         如果VTIME和VMIN都不取0,VTIME定义的是当接收到第一个字节的数据后开始计算等待的时间量。如果当调用read函数时可以得到数据,计时器 马上开始计时。如果当调用read函数时还没有任何数据可读,则等接收到第一个字节的数据后,计时器开始计时。函数read可能会在读取到VMIN个字节 的数据后返回,也可能在计时完毕后返回,这主要取决于哪个条件首先实现。不过函数至少会读取到一个字节的数据,因为计时器是在读取到第一个数据时开始计时 的。

l         如果VTIME和VMIN都取0,即使读取不到任何数据,函数read也会立即返回。同时,返回值0表示read函数不需要等待文件结束标志就返回了。

这就是这两个变量对read函数的影响。我使用的读卡器每次传送的数据是13个字节,一开始,我把它们设置成

options.c_cc[VTIME] = 150

options.c_cc[VMIN] = 0;

结果,每次读取的信息只有8个字节,剩下的5个字节要等到下一次打卡时才能收到。就是由于这个原因造成的。根据上面规则的第一条,我把VTIME取 0,VMIN=13,也就是正好等于一次需要接收的字节数。这样就实现了一次读取13个字节值。同时,得出这样的结论,如果读卡器送出的数据为n个字节, 那么就把VMIN=n,这样一次读取的信息正好为读卡器送出的信息,并且读取的时候不需要进行循环读取。

 

4.         读取数据

有了上面的函数后,我设置了串口的基本信息,根据我们自己的实际情况,设置了相应的参数,就可以读取数据了。

void getcardinfo(char *buff){

         int fd;

         int nread,count=0;

         char tempbuff[13];

         char *dev  = "/dev/ttyS0"; //串口1

         fd = OpenDev(dev);

         set_speed(fd,9600);

         if (set_Parity(fd,8,1,'N') == FALSE)  {

                   printf("Set Parity Error\n");

                   //return -1;

         }

         while (1) //循环读取数据

         {  

                   count=0;

                   //sleep(5000);

                   while(1)

                   {

                            if((nread = read(fd, tempbuff, 13))>0)

                            {

                            //printf("\nLen %d\n",nread);

                                     memcpy(&buff[count],tempbuff,nread);

                                     count+=nread;

                            }

                            if(count==13)

                            {

                                     buff[count+1] = '\0';  

                            //printf( "\n%s", buff);

                                     break;

                            }

                   }

                   //break;

         }

         //return buff;

         close(fd);

         pthread_exit(NULL);

         //close(fd); 

         // exit (0);

}

这是我原来的程序,其实把VMIN设置以后,可以改成:

void getcardinfo(char *buff){

       int fd;

       int nread,count=0;

       char tempbuff[13];

       char *dev  = "/dev/ttyS0"; //串口1

       fd = OpenDev(dev);

       set_speed(fd,9600);

       if (set_Parity(fd,8,1,'N') == FALSE)  {

              printf("Set Parity Error\n");

              //return -1;

       }

       nread = read(fd, buff, 13)

       close(fd);

}

 

5.         程序完整代码:

#include     <stdio.h>      /*标准输入输出定义*/

#include     <stdlib.h>     /*标准函数库定义*/

#include     <unistd.h>     /*Unix 标准函数定义*/

#include     <sys/types.h> 

#include     <sys/stat.h>  

#include     <fcntl.h>      /*文件控制定义*/

#include     <termios.h>    /*PPSIX 终端控制定义*/

#include     <errno.h>      /*错误号定义*/

 

#define FALSE  -1

#define TRUE   0

/**

*@brief    设置串口通信速率

*@param  fd     类型 int  打开串口的文件句柄

*@param  speed  类型 int  串口速度

*@return    void

*/

int speed_arr[] = { B38400, B19200, B9600, B4800, B2400, B1200, B300,

                   B38400, B19200, B9600, B4800, B2400, B1200, B300, };

int name_arr[] = {38400,  19200,  9600,  4800,  2400,  1200,  300, 38400, 

                            19200,  9600, 4800, 2400, 1200,  300, };

void set_speed(int fd, int speed){

         int   i;

         int   status;

         struct termios   Opt;

         tcgetattr(fd, &Opt);

         for ( i= 0;  i < sizeof(speed_arr) / sizeof(int);  i++) {

                   if  (speed == name_arr[i]) {    

                            tcflush(fd, TCIOFLUSH);    

                            cfsetispeed(&Opt, speed_arr[i]); 

                            cfsetospeed(&Opt, speed_arr[i]);  

                            status = tcsetattr(fd, TCSANOW, &Opt); 

                            if  (status != 0) {       

                                     perror("tcsetattr fd"); 

                                     return;    

                            }   

                            tcflush(fd,TCIOFLUSH);  

                   } 

         }

}

/**

*@brief     设置串口数据位,停止位和效验位

*@param  fd     类型  int  打开的串口文件句柄

*@param  databits 类型  int 数据位   取值 为 7 或者8

*@param  stopbits 类型  int 停止位   取值为 1 或者2

*@param  parity  类型  int  效验类型 取值为N,E,O,,S

*/

int set_Parity(int fd,int databits,int stopbits,int parity)

{

         struct termios options;

         if  ( tcgetattr( fd,&options)  !=  0) {

                   perror("SetupSerial 1");    

                   return(FALSE); 

         }

         options.c_cflag &= ~CSIZE;

         options.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);  /*Input*/

         options.c_oflag  &= ~OPOST;   /*Output*/

 

         switch (databits) /*设置数据位数*/

         {  

         case 7:                

                   options.c_cflag |= CS7;

                   break;

         case 8:    

                   options.c_cflag |= CS8;

                   break;  

         default:   

                   fprintf(stderr,"Unsupported data size\n"); return (FALSE); 

         }

switch (parity)

{  

         case 'n':

         case 'N':   

                   options.c_cflag &= ~PARENB;   /* Clear parity enable */

                   options.c_iflag &= ~INPCK;     /* Enable parity checking */

                   break; 

         case 'o':  

         case 'O':    

                   options.c_cflag |= (PARODD | PARENB); /* 设置为奇效验*/ 

                   options.c_iflag |= INPCK;             /* Disnable parity checking */

                   break; 

         case 'e': 

         case 'E':  

                   options.c_cflag |= PARENB;     /* Enable parity */   

                   options.c_cflag &= ~PARODD;   /* 转换为偶效验*/    

                   options.c_iflag |= INPCK;       /* Disnable parity checking */

                   break;

         case 'S':

         case 's':  /*as no parity*/  

             options.c_cflag &= ~PARENB;

                   options.c_cflag &= ~CSTOPB;break; 

         default:  

                   fprintf(stderr,"Unsupported parity\n");   

                   return (FALSE); 

         } 

/* 设置停止位*/ 

switch (stopbits)

{  

         case 1:   

                   options.c_cflag &= ~CSTOPB; 

                   break; 

         case 2:   

                   options.c_cflag |= CSTOPB; 

            break;

         default:   

                    fprintf(stderr,"Unsupported stop bits\n"); 

                    return (FALSE);

}

/* Set input parity option */

if (parity != 'n')  

         options.c_iflag |= INPCK;

tcflush(fd,TCIFLUSH);

options.c_cc[VTIME] = 0; /* 设置超时15 seconds*/  

options.c_cc[VMIN] = 13; /* define the minimum bytes data to be readed*/

if (tcsetattr(fd,TCSANOW,&options) != 0)  

{

         perror("SetupSerial 3");  

         return (FALSE); 

}

return (TRUE); 

}

/**********************************************************************

代码说明:使用串口一测试的,发送的数据是字符,

但是没有发送字符串结束符号,所以接收到后,后面加上了结束符号

**********************************************************************/

 

/*********************************************************************/

int OpenDev(char *Dev)

{

         int     fd = open( Dev, O_RDWR );

        //| O_NOCTTY | O_NDELAY         

         if (-1 == fd)        

         {                        

                   perror("Can't Open Serial Port");

                   return -1;            

         }      

         else  

                   return fd;

}

void getcardinfo(char *buff){

         int fd;

         int nread,count=0;

         char tempbuff[13];

         char *dev  = "/dev/ttyS0"; //串口1

         fd = OpenDev(dev);

         set_speed(fd,9600);

         if (set_Parity(fd,8,1,'N') == FALSE)  {

                   printf("Set Parity Error\n");

                   //return -1;

         }

         while (1) //循环读取数据

         {  

                   count=0;

                   //sleep(5000);

                   while(1)

                   {

                            if((nread = read(fd, tempbuff, 13))>0)

                            {

                            //printf("\nLen %d\n",nread);

                                     memcpy(&buff[count],tempbuff,nread);

                                     count+=nread;

                            }

                            if(count==13)

                            {

                                     buff[count+1] = '\0';  

                            //printf( "\n%s", buff);

                                     break;

                            }

                   }

                   //break;

         }

         //return buff;

         close(fd);

         pthread_exit(NULL);

         //close(fd); 

         // exit (0);

}


© 著作权归作者所有

共有 人打赏支持
wangjian19
粉丝 15
博文 60
码字总数 101397
作品 0
深圳
高级程序员
Linux 下的串口编程(一)

Linux下串口编程要知道的那些事 --------------------------------------------------------- Author :tiger-john WebSite :blog.csdn.net/tigerjb Email :jibo.tiger@gmail.com Tiger声明:......

YACHE
2011/07/04
0
0
xp下用虚拟串口与linux的minicom通信

目的 linux下的串口通信程序,笔记本没有串口,台式机居然也没有,没有USB转串,调试个P啊,肿么办?虚拟个串口先~ 交代下环境 windowsXP SP3 vmware 7.1.4 Red Hat 4.1.2-48 (cat /proc/v...

bbdlg
2013/01/09
0
0
学习 ARM 系列 -- FS2410 开发板上的串口通信编程

学习 ARM 系列 -- FS2410 开发板上的串口通信编程 一、目的 串口通信我们并不陌生,我们经常用串口来进行数据传输,可并不清楚它是如何工作 的。那这一节我们就来揭开 ARM S3c2410 UART(Uni...

YACHE
2011/07/04
0
0
ubuntu下配置交叉编译环境

操作系统平台 华邦公司建议使用RedHat6.x 以上linux版本,和至少800M的硬盘,这里我们使用Ubuntu 10.04.4LTS(Lucid Lynx)linux操作系统,20G的文件系统空间。 我们这里使用华邦提供的软件包...

tanxunlishi
2012/11/16
0
0
海思3516D 调试串口做通信串口 应用层修改

1.修改/etc/inittab # Example of how to put a getty on a serial line (for a terminal) #::respawn:/sbin/getty -L ttyS000 115200 vt100 -n root -I "Auto login as root ..." #::respaw......

ma_ya_dong
05/08
0
0
激光雷达学习笔记(六)ARM-Linux平台算法移植

在写第一篇激光雷达的日志我就说,激光雷达的算法开发的最终的目标是移植到嵌入式设备上, 因为在机器人定位时,不可能用一个很大很重的设备,因此最理想的还是ARM平台,(工控机也可以 考虑...

dddxxxx
05/11
0
0
嵌入式Linux学习基础规划篇

嵌入式的学习是需要日积月累的,是通过一点一滴的积累才能成为大神。下面来介绍一下嵌入式linux学习基础规划,目标是达到适应嵌入式应用软件开发、嵌入式系统开发或嵌入式驱动开发的基本素质...

创客学院
04/10
0
0
ubuntu下安装minicom

今天看了看嵌入式系统开发,结合之前在tinyos实验中做过的一些串口通信的实验,这里看到linux下也有相关的串口通信软件。所以就在我的ubuntu下安装了minicom,其实这个软件并没有图形化界面,...

demose
2014/11/09
0
0
Linux环境下使用 USB转串口驱动(二)

linux下USB转串口识别 2012年03月30日 星期五 11:24 minicom是linux下串口通信的软件,它的使用完全依靠键盘的操作,虽然没有“超级终端”那么易用,但是使用习惯之后读者将会体会到它的高效...

乐学为上
2012/05/24
0
1
UART0串口编程系列(三)

UART0串口编程之在UC/OS—II中遭遇的危机 一.潜在的危机 1.在uc/os操作系统中设计串口编程时,由于ISR和多个任务并发执行,情况比较复杂。尤其是接收状态为被动状态时,只能靠串行口中断来接...

YACHE
2011/07/04
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Spring Cloud云服务 - HongHu架构common-service 项目构建过程

上一篇我们介绍了《整合spring cloud云服务架构 - HongHu云架构common-service代码结构分析》,本节我们将对common-service整个项目进行剖析,将整个构建的流程给记录下来,让更多的关注者来...

itcloud
5分钟前
0
0
Connection reset

在使用HttpClient调用后台resetful服务时,“Connection reset”是一个比较常见的问题,有同学跟我私信说被这个问题困扰很久了,今天就来分析下,希望能帮到大家。例如我们线上的网关日志就会...

夜黑人模糊灬
9分钟前
0
0
如何写PHP规范注释

所有的文档标记都是在每一行的 * 后面以@开头。如果在一段话的中间出来@的标记,这个标记将会被当做普通内容而被忽略掉。 @access 该标记用于指明关键字的存取权限:private、public或prote...

度_
9分钟前
0
0
influxDB Ppostgis

PostGis 1.需要安装postgreSQL,postgis作为插件嵌入到postgreSQL中; 2.使用zip包直接安装,需要修改 makepostgisdb_using_extensions.bat文件中的路径,用户名,密码,然后直接运行; 3.没有...

courtzjl
13分钟前
0
0
多线程Thread-多线程顺序执行

需求:现在有两个任务,任务1和任务2,任务1中有多个线程,并且任务2必须等任务1完成后才能执行。 namespace TThread{ class Program { static void Main(string[] ar...

kaixinguo314
17分钟前
1
0
直播总结

https://blog.csdn.net/weiyuefei/article/details/70257616 视频: 上传服务,转码服务,视频一致性MD5 视频转码技术及转码实现详解 https://wenku.baidu.com/view/32702108844769eae009eda...

素雷
19分钟前
0
0
Linux系统-tcpdump常用抓包命令

序言 单独总结tcpdump抓包的常用命令 主要语法 过滤主机/IP: tcpdump -i eth1 host 172.16.7.206 抓取所有经过网卡1,目的IP为172.16.7.206的网络数据 过滤端口: tcpdump -i eth1 dst port...

MrBoyce
20分钟前
0
0
274. H-Index - LeetCode

Question 274. H-Index Solution 题目大意: 论文里的 h 因子判定,题目的意思可能有点晦涩。h 因子是评判学术成就的一种重要方法,h 因子越高越好,h 因子兼顾研究学术人员的学术产出数量与...

yysue
24分钟前
0
0
 Unable to read class [com.mrp.action.BaseAction]

struts2启动时,出现的 Unable to read class [com.mrp.action.BaseAction] java.lang.NoClassDefFoundError: com/opensymphony/xwork2/util/finder/DefaultClassFinder$InfoBuildingVisito......

uug
26分钟前
0
0
分发系统介绍&expect脚本远程登录&expect脚本远程执行命令&expect脚本传递参数

20.27 分发系统介绍 应用场景 企业中随之业务的逐渐增大,后端所使用的编程语言是php,系统为LAMP/LNMP架构,需要将代码上传到服务器中;代码会不断的迭代,这就需要在业务服务器上更新代码,...

影夜Linux
27分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部