linux 串口编程 用read函数读取数据被截断 怎样一次读完

来源:互联网 发布:腾讯企业邮箱域名设置 编辑:程序博客网 时间:2024/06/08 19:04
linux 串口读取数据被截断,怎样一次全部接收?
打开串口后,用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);
}


一般地串口的读写模式有直接模式和缓存模式,在直接模式下,串口的读写都是单字节的,也就是说一次的read或write只能操作一个字节;另外大部份串口芯片都支持缓存模式,缓存模式一般同时支持中断聚合和超时机制,也就是说在有数据时,当缓存满或者超时时间到时,都会触发读或写中断。写的时候可以将要操作的数据先搬到缓存里,然后启动写操作,芯片会自动将一连串的数据写出,在读的时候类似,一次读到的是串口芯片缓存里的数据。串口设备的缓存一般有限,一次能read到的最大字节数就是缓存的容量。
所以你串口芯片的缓存容量决定了你一次能收到的字节数。

楼主,你可以一个字节一读,使用一个全局计数器来保证收到8个字节的数据。
可以使用select/poll来避免超时接收。

从通讯的角度来说,接受方必须自己解决如何识别一个祯的问题。

操作串口相当于操作物理层,OSI/ISO模型中的第一层,解决祯同步问题是第二层的任务,所以你需要自己搭一个第二层。

简单的说,你需要通过定义通讯协议,规定数据的内容自己分析什么时候收完了一次需要的数据。因为通讯过程中无法保证一次发送的数据肯定是一次接收的。
0 0
原创粉丝点击