linux 串口读取数据被截断,怎样一次全部接收?
打开串口后,用read读取串口数据,串口发来的
数据为20,而用read读取到的数据却是分多次得到的?
有时只读了8个数据便被截断了,这是为什么??
怎样一次全部接收?? 谢谢
附代码:
一般地串口的读写模式有直接模式和缓存模式,在直接模式下,串口的读写都是单字节的,也就是说一次的read或write只能操作一个字节;另外大部份串口芯片都支持缓存模式,缓存模式一般同时支持中断聚合和超时机制,也就是说在有数据时,当缓存满或者超时时间到时,都会触发读或写中断。写的时候可以将要操作的数据先搬到缓存里,然后启动写操作,芯片会自动将一连串的数据写出,在读的时候类似,一次读到的是串口芯片缓存里的数据。串口设备的缓存一般有限,一次能read到的最大字节数就是缓存的容量。
所以你串口芯片的缓存容量决定了你一次能收到的字节数。
楼主,你可以一个字节一读,使用一个全局计数器来保证收到8个字节的数据。
可以使用select/poll来避免超时接收。
从通讯的角度来说,接受方必须自己解决如何识别一个祯的问题。
操作串口相当于操作物理层,OSI/ISO模型中的第一层,解决祯同步问题是第二层的任务,所以你需要自己搭一个第二层。
简单的说,你需要通过定义通讯协议,规定数据的内容自己分析什么时候收完了一次需要的数据。因为通讯过程中无法保证一次发送的数据肯定是一次接收的。
打开串口后,用read读取串口数据,串口发来的
数据为20,而用read读取到的数据却是分多次得到的?
有时只读了8个数据便被截断了,这是为什么??
怎样一次全部接收?? 谢谢
附代码:
#include <stdio.h>
#include <unistd.h> /*Unix标准函数定义*/
#include <sys/types.h> /**/
#include <sys/stat.h> /**/
#include <fcntl.h> /*文件控制定义*/
#include <termios.h> /*PPSIX终端控制定义*/
#include <errno.h> /*错误号定义*/
#include <stdlib.h>
#include <string.h>
int main()
{
int fd,readnum;
char *dev="/dev/ttyS0",recv[100];
struct termios opt,oldopt;
fd = open( dev, O_RDWR );
tcgetattr( fd,&oldopt);
tcgetattr( fd,&opt);
cfsetispeed(&opt, B9600);//9600
cfsetospeed(&opt, B9600);
opt.c_cflag &= ~PARENB; //N
opt.c_cflag &= ~INPCK;
opt.c_cflag &= ~CSTOPB;//1
opt.c_cflag &= ~CSIZE;
opt.c_cflag |= CS8; //8
opt.c_iflag &= ~(IXON | IXOFF | IXANY);
opt.c_cc[VTIME] = 0;
opt.c_cc[VMIN] = 0;
opt.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); /*Input*/
opt.c_oflag &= ~OPOST; /*Output*/
tcflush(fd, TCIOFLUSH);
tcsetattr(fd, TCSANOW, &opt);
while(1)
{
bzero(recv,100);
while((readnum = read(fd,recv,100))>0)
{
printf("%s\n",recv);
}
}
tcflush(fd, TCIOFLUSH);
tcsetattr(fd, TCSANOW, &oldopt);
close(fd);
}
#include <unistd.h> /*Unix标准函数定义*/
#include <sys/types.h> /**/
#include <sys/stat.h> /**/
#include <fcntl.h> /*文件控制定义*/
#include <termios.h> /*PPSIX终端控制定义*/
#include <errno.h> /*错误号定义*/
#include <stdlib.h>
#include <string.h>
int main()
{
int fd,readnum;
char *dev="/dev/ttyS0",recv[100];
struct termios opt,oldopt;
fd = open( dev, O_RDWR );
tcgetattr( fd,&oldopt);
tcgetattr( fd,&opt);
cfsetispeed(&opt, B9600);//9600
cfsetospeed(&opt, B9600);
opt.c_cflag &= ~PARENB; //N
opt.c_cflag &= ~INPCK;
opt.c_cflag &= ~CSTOPB;//1
opt.c_cflag &= ~CSIZE;
opt.c_cflag |= CS8; //8
opt.c_iflag &= ~(IXON | IXOFF | IXANY);
opt.c_cc[VTIME] = 0;
opt.c_cc[VMIN] = 0;
opt.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG); /*Input*/
opt.c_oflag &= ~OPOST; /*Output*/
tcflush(fd, TCIOFLUSH);
tcsetattr(fd, TCSANOW, &opt);
while(1)
{
bzero(recv,100);
while((readnum = read(fd,recv,100))>0)
{
printf("%s\n",recv);
}
}
tcflush(fd, TCIOFLUSH);
tcsetattr(fd, TCSANOW, &oldopt);
close(fd);
}
一般地串口的读写模式有直接模式和缓存模式,在直接模式下,串口的读写都是单字节的,也就是说一次的read或write只能操作一个字节;另外大部份串口芯片都支持缓存模式,缓存模式一般同时支持中断聚合和超时机制,也就是说在有数据时,当缓存满或者超时时间到时,都会触发读或写中断。写的时候可以将要操作的数据先搬到缓存里,然后启动写操作,芯片会自动将一连串的数据写出,在读的时候类似,一次读到的是串口芯片缓存里的数据。串口设备的缓存一般有限,一次能read到的最大字节数就是缓存的容量。
所以你串口芯片的缓存容量决定了你一次能收到的字节数。
楼主,你可以一个字节一读,使用一个全局计数器来保证收到8个字节的数据。
可以使用select/poll来避免超时接收。
从通讯的角度来说,接受方必须自己解决如何识别一个祯的问题。
操作串口相当于操作物理层,OSI/ISO模型中的第一层,解决祯同步问题是第二层的任务,所以你需要自己搭一个第二层。
简单的说,你需要通过定义通讯协议,规定数据的内容自己分析什么时候收完了一次需要的数据。因为通讯过程中无法保证一次发送的数据肯定是一次接收的。
作者:jackxiang@向东博客 专注WEB应用 构架之美 --- 构架之美,在于尽态极妍 | 应用之美,在于药到病除
地址:http://jackxiang.com/post/2856/
版权所有。转载时必须以链接形式注明作者和原始出处及本声明!
评论列表