如何在读取函数调用中实现超时?

时间:2010-05-27 01:06:58

标签: c linux

我想使用串行com端口进行通信,我希望每次调用read函数调用时都实现超时。

int filedesc = open( "dev/ttyS0", O_RDWR );

read( filedesc, buff, len );

编辑:

我正在使用Linux操作系统。如何使用select函数调用实现?

6 个答案:

答案 0 :(得分:74)

选择()  需要5个参数,首先是最高文件描述符+ 1,然后是fd_set用于读取,一个用于写入,一个用于异常。最后一个参数是struct timeval,用于超时。它在出错时返回-1,在超时时返回0或在设置的集合中返回文件描述符的数量。

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/select.h>

int main(void)
{
  fd_set set;
  struct timeval timeout;
  int rv;
  char buff[100];
  int len = 100;
  int filedesc = open( "dev/ttyS0", O_RDWR );

  FD_ZERO(&set); /* clear the set */
  FD_SET(filedesc, &set); /* add our file descriptor to the set */

  timeout.tv_sec = 0;
  timeout.tv_usec = 10000;

  rv = select(filedesc + 1, &set, NULL, NULL, &timeout);
  if(rv == -1)
    perror("select"); /* an error accured */
  else if(rv == 0)
    printf("timeout"); /* a timeout occured */
  else
    read( filedesc, buff, len ); /* there was data to read */
}

答案 1 :(得分:39)

作为select()的替代方案,对于串行端口(终端)的特定情况,您可以使用tcsetattr()将文件描述符置于非规范模式,并具有读取超时。

要执行此操作,请取消设置ICANON标记,并设置VTIME控制字符:

struct termios termios;

tcgetattr(filedesc, &termios);
termios.c_lflag &= ~ICANON; /* Set non-canonical mode */
termios.c_cc[VTIME] = 100; /* Set timeout of 10.0 seconds */
tcsetattr(filedesc, TCSANOW, &termios);

注意VTIME以十分之一秒为单位测量,并且用于它的类型通常为unsigned char,这意味着最大超时为25.5秒。

答案 2 :(得分:8)

如果设置套接字确实以非阻塞模式运行,则每次读取调用将只读取当前可用的数据(如果有)。所以这实际上等于立即超时。

您可以使用如下函数在套接字上设置非阻塞模式:

int setnonblock(int sock) {
   int flags;
   flags = fcntl(sock, F_GETFL, 0);
   if (-1 == flags)
      return -1;
   return fcntl(sock, F_SETFL, flags | O_NONBLOCK);
}

(有关从非阻塞套接字读取的更多信息,请参阅read手册页)

答案 3 :(得分:4)

您没有说操作系统是什么,但如果您在Linux下运行,则可以使用select调用。如果在文件描述符上有要读取的内容,则返回,或者您可以将其设置为在没有任何内容可读的情况下超时。返回码表示哪个。

答案 4 :(得分:2)

下面的代码对每个字符使用Millisec超时。 我在我的项目之一中使用它从COM端口读取。

size_t TimeoutRead (int port, void*buf, size_t size, int mlsec_timeout)
{
    struct pollfd fd = { .fd = port, .events = POLLIN };

    size_t      bytesread = 0;

    while (poll (&fd, 1, mlsec_timeout) == 1)
    {
        int chunksize = read (port, buf + bytesread, size);
        if (chunksize == -1)
            return -1;

        bytesread += chunksize;
        size -= chunksize;

        if (size == 0)
            return bytesread;
    }

    // TODO: IsTimeout = true;
    return bytesread;
}

答案 5 :(得分:-5)

Linux考虑两种类型的设备。 “慢”(例如网络)和“快速”(例如磁盘)。非阻塞模式用于“慢”设备,而不是“快速”设备。阻止设备应该是“快速”,因此select / poll总是告诉它们已准备好进行I / O.对于某些闪存设备,这是一个谎言,我已经看到一些I / O调用,即使select / poll告诉设备准备好,也会使用几秒钟的几个字节。 这就是说,即使对于慢速设备,读取延迟本身也没有强有力的保证;如果您提供足够大的缓冲区并且数据可用(您已使用select或poll来确定),则read call本身可能会占用大量时间(网络上的数百ms)。