1.

打开串口

与其他的关于设备编程的方法一样,在

Linux

下,操作、控制串口也是通过操作起设备文件进行的。在

Linux

下,串口的设备文件是

/dev/ttyS0

/dev/ttyS1

等。因此要读些串口,我们首先要打开串口:

char *dev= "/dev/ttyS0"; //

串口

1

intfd = 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){

inti;

intstatus;

struct termiosOpt;

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

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

*@paramfd

类型

int

打开的串口文件句柄

*@paramdatabits

类型

int

数据位

取值

7

或者

8

*@paramstopbits

类型

int

停止位

取值为

1

或者

2

*@paramparity

类型

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_tc_line;/*

线路规程

*/

cc_tc_cc[NCCS];/*

控制符号

*/

}

其中

cc_tc_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/*

标准输入输出定义

*/

#include/*

标准函数库定义

*/

#include/*Unix

标准函数定义

*/

#include

#include

#include/*

文件控制定义

*/

#include/*PPSIX

终端控制定义

*/

#include/*

错误号定义

*/

#define FALSE-1

#define TRUE0

/**

*@brief

设置串口通信速率

*@paramfd

类型

int

打开串口的文件句柄

*@paramspeed

类型

int

串口速度

*@returnvoid

*/

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){

inti;

intstatus;

struct termiosOpt;

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

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

*@paramfd

类型

int

打开的串口文件句柄

*@paramdatabits

类型

int

数据位

取值

7

或者

8

*@paramstopbits

类型

int

停止位

取值为

1

或者

2

*@paramparity

类型

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)

{

intfd = 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);

}

linux下串口通信程序,关于Linux下串口通信的一点心得相关推荐

  1. 用VS2012或VS2013在win7下编写的程序在XP下运行就出现“不是有效的win32应用程序

    问题描述: 用VC2013编译了一个程序,在Windows 8.Windows 7(64位.32位)下都能正常运行.但在Win XP,Win2003下运行时,却报错不能运行,具体错误信息为" ...

  2. 将asp.net1.1的应用程序升级到asp.net2.0的一点心得

    将asp.net1.1的应用程序升级到asp.net2.0的一点心得 费了好一袋烟工夫把CommunityServer升级到了Asp.Net2.0平台,一点心得: vs2005可以很方便的帮我们把vs ...

  3. linux下启动应用程序,RedHat Linux 9下启动应用程序

    RedHat Linux 9下启动应用程序 发布时间:2005-09-02 14:07:29来源:红联作者:zz123 我安装了一个从互联网上下载的应用程序,一切似乎都正常,但在我键入它的名称后,却总 ...

  4. Linux从头开始学--学习笔记9知识点补充-ubuntu,centos;在linux上创建c程序;linux基础命令,shell命令,vi命令,man帮助手册

    这是我从头开始学习Linux的学习笔记,后续还会更新. 记录自己的技术成长,也希望和大家分享交流,欢迎关注~ 本笔记为coursera网站课程<Linux for Developers>的 ...

  5. win下搭建小程序服务器,win下搭建小程序服务器

    win下搭建小程序服务器 内容精选 换一换 云硬盘挂载至云服务器后,需要登录云服务器初始化云硬盘,即格式化云硬盘,之后云硬盘才可以正常使用.云耀云服务器磁盘初始化的操作方法与ECS相同,本节操作介绍使 ...

  6. linux下怎么查看程序异常,linux程序莫名异常怎么查

    内存异常经常导致程序出现莫名其妙的错误,往往很难查证,本文介绍在linux下的各种常见内存异常的查证工具和方法. 1 访问空指针/未初始化指针/重复释放内存 对于像访问空指针.未初始化指针(非法地址) ...

  7. linux怎么卸载桌面程序吗,Linux下怎么卸载软件

    Linux软件的安装和卸载一直是困扰许多新用户的难题.在Windows中,我们可以使用软件自带的安装卸载程序或在控制面板中的"添加/删除程序"来实现.与其相类似,在Linux下有一 ...

  8. 在linux上安装QQ程序,在Linux系统下使用QQ(wine方法)

    前言:未加设置直接wine必然导致QQ不断错误重启 原因推断:键盘保护功能导致 解决方法: 1.机器上要有wine(Debian系列:apt-get install wine) 2.wine QQ安装 ...

  9. 在linux中查找运行程序句柄,Linux下查看句柄

    查看各个分区的句柄数 df -ih 查看句柄最大数 ulimit -n 查看23711进程连接句柄数 lsof -n|awk '{print $2}'|sort|uniq -c|sort -nr|mo ...

最新文章

  1. pap和chap交叉认证
  2. 如何使用Navicat恢复数据库脚本
  3. Java中WeakReference,SoftReference,PhantomReference和Strong Reference之间的区别
  4. SpringBoot使用@Cacheable实现最简单的Redis缓存
  5. 爬虫模拟登陆手机验证码_爬虫入门到精通-headers的详细讲解(模拟登录知乎)...
  6. 较简单的date转化成格式化的timeString
  7. Android 退出app,后台推送的服务也停止了,怎么可以做到不停止后台服务呢?
  8. ICP算法学习笔记(原理加公式理解)
  9. matlab求系统根轨迹代码_如何绘制变参数根轨迹(针对复杂情况,无法分离出开环增益k*时)...
  10. 做绿色数据中心基础设施建维服务认证有哪些好处?
  11. 饮食、生物钟、肠道菌群的“三角恋”
  12. python调用sendcloud模板发送带附件的邮件
  13. 开传奇大概需要什么条件
  14. UI设计师必备10大工具
  15. 单体测试使用Assert.assertThat(expected,Matcher matcher)来对比结果和预期
  16. 如何安装和使用vicuna
  17. 标准化金额:加小数点和逗号 / 加逗号 / 加小数点 / 纯数字
  18. python glob函数_Python glob()函数
  19. 该爬破解验证码,爬企信宝必须破解滑块验证
  20. javascript的ES6笔记整理

热门文章

  1. CentOS7添加中文输入法
  2. 供SAPI中TTS功能用使的Win8.1语言包安装
  3. 2019牛客暑期多校训练营(第九场) E All men are brothers
  4. 28-Interview-面试
  5. 20145317 《网络对抗技术》免杀原理与实践
  6. 自动加载 autoload
  7. oracle获取时间毫秒数
  8. 民政部部长李纪恒:适龄人口生育意愿偏低,总和生育率破警戒线
  9. Pthread多线程编程之查看Pthread版本的方法
  10. 从难以普及的数据增强技术,看AI的性价比时代