小编典典

使用poll()在命名管道上使用O_RDWR

linux

我经历了各种不同的名为管道客户端/服务器实现的Linux,但其中大多数在读取/写入时使用阻止默认值。

因为我已经在使用poll()来检查其他标志,所以也可以通过poll()检查传入的FIFO数据是一个好主意…

经过所有研究,我认为以O_RDWR模式打开管道是防止在没有任何编写者打开管道的情况下无限期发生EOF事件的唯一方法。

这样,管道的两端都关闭了,其他客户端也可以打开可写端。作为回应,我将使用单独的管道…

我的问题是,尽管我发现了一些使用O_RDWR标志的示例,但open()联机帮助页将该标志描述为在分配给FIFO时未定义。(http://linux.die.net/man/3/open

但是如何在没有O_RDWR的管道上使用poll()?您认为“ O_RDWR”是打开管道的合法方法吗???


阅读 714

收藏
2020-06-03

共1个答案

小编典典

首先,一些预备:

使用O_NONBLOCKpoll()是常见做法-并非相反。要顺利进行,您需要确保处理所有poll()read()正确的返回状态:

  • read()``0均值EOF的返回值-另一端已关闭其连接。(通常,但不是在所有操作系统上)这对应于poll()返回POLLHUP清除。您可能想POLLHUP在尝试之前进行检查read(),但这并不是绝对必要的,因为read()可以保证0在书写侧关闭后返回。
  • 如您所知read(),如果您在作家连接之前打电话,并且您有O_RDONLY | O_NONBLOCK,您将反复得到EOF(read()返回0)。但是,如果您习惯在调用之前poll()等待POLLIN事件read(),它将等待编写器连接,而不产生EOF。
  • read()返回值-1通常表示错误。但是,如果是errno == EAGAIN,这仅表示当前没有更多数据可用,并且您没有阻塞,因此可以返回到poll()其他设备需要处理的情况。如果errno == EINTR,则read()在读取任何数据之前已中断,您可以返回poll()或直接read()再次调用。

现在,对于Linux:

  • 如果您使用来打开阅读侧O_RDONLY,则:

    • open()将阻塞,直到有相应的作家开放。
    • poll()将给予POLLIN当数据准备好被读取,或出现EOF时r事件。
    • read()将一直阻塞,直到读取请求的字节数,关闭连接(返回0),被信号中断或发生致命的IO错误为止。这种阻碍性的破坏了使用的目的poll(),这就是为什么poll()几乎总是与一起使用的原因O_NONBLOCK。您可以使用an 在超时后alarm()唤醒read(),但这太复杂了。
    • 如果作家关闭,然后将读卡器将获得poll() POLLHUPr事件和read()将返回0之后下去。此时,读取器必须关闭其文件句柄并重新打开它。
  • 如果您使用来打开阅读侧O_RDONLY | O_NONBLOCK,则:

    • open()不会阻止。
    • poll()将给予POLLIN当数据准备好被读取,或出现EOF时r事件。poll()也会阻塞,直到没有可用的编写器为止。
    • 读取所有当前可用的数据之后,如果连接仍处于打开状态,read()则将返回-1并进行设置errno == EAGAIN,或者0如果连接器已关闭(EOF) 或尚未由写入器打开 ,则将返回-1 。如果为errno == EAGAIN,则意味着该返回到了poll(),因为该连接已打开但没有更多数据。当时errno == EINTRread()尚未读取任何字节并被信号中断,因此可以重新启动它。
    • 如果作家关闭,然后将读卡器将获得poll() POLLHUPr事件,并且read()将返回0之后下去。此时,阅读器必须关闭其文件句柄并重新打开它。
  • (特定于Linux :)如果您在阅读方面打开O_RDWR,则:

    • open()不会阻止。
    • poll()将给予POLLIN当数据准备好被读取r事件。但是,对于命名管道,EOF不会引起POLLINPOLLHUP阻止。
    • read()将一直阻塞,直到读取请求的字节数,被信号中断或发生其他严重的IO错误为止。对于命名管道,它不会返回errno == EAGAIN,甚至不会0在EOF上返回。它会一直坐在那里,直到读取了请求的确切字节数,或者直到接收到一个信号为止(在这种情况下,它将返回到目前为止已读取的字节数,或者errno == EINTR如果没有读取到字节,则返回-1并进行设置)。 。
    • 如果编写器关闭,则在另一个编写器打开命名管道的情况下,读取器也不会失去稍后读取命名管道的功能,但是读取器也不会收到任何通知。
  • (特定于Linux :)如果您在阅读方面打开O_RDWR | O_NONBLOCK,则:

    • open()不会阻止。
    • poll()将给予POLLIN当数据准备好被读取r事件。但是,EOF不会导致POLLIN或限制POLLHUP命名管道。
    • 读取所有当前可用数据后,read()将返回-1并设置errno == EAGAIN。现在该返回到poll()等待更多数据(可能来自其他流)的时间。
    • 如果编写器关闭,则在另一个编写器打开命名管道的情况下,读取器也不会失去稍后读取命名管道的功能。连接是持久的。

如您所知,O_RDWR与管道一起使用不是POSIX或其他标准。

然而,由于这个问题似乎来了的时候,在Linux上的最佳途径,使“弹性命名管道”的生存,即使一方关闭,并不会引起POLLHUPrevents中或回报0read(),就是用O_RDWR | O_NONBLOCK

我看到了在Linux上处理命名管道的三种主要方法:

  1. (便携式。)不使用poll(),并且使用单个管道:

    • open(pipe, O_RDONLY);
    • 主循环:
    • read()根据需要提供尽可能多的数据,可能会在read()调用时循环。
      • 如果read() == -1errno == EINTR,则read()重新遍历。
      • 如果为read() == 0,则连接已关闭,并且已接收所有数据。
  2. (可移植。)使用poll(),并期望管道(即使是命名管道)仅打开一次,并且一旦关闭,则读取器和写入器都必须重新打开它们,从而建立新的管道:

    • open(pipe, O_RDONLY | O_NONBLOCK);
    • 主循环:
    • poll()对于POLLIN事件,可能一次在多个管道上。(注意:这可以防止read()在连接编写器之前获得多个EOF。)
    • read()根据需要提供尽可能多的数据,可能会在read()调用时循环。
      • 如果read() == -1errno == EAGAIN,请返回poll()步骤。
      • 如果read() == -1errno == EINTR,则read()重新遍历。
      • 如果为read() == 0,则连接已关闭,您必须终止,或者关闭并重新打开管道。
  3. (不可移植,特定于Linux。)使用poll(),并期望命名管道永远不会终止,并且可以多次连接和断开连接:

    • open(pipe, O_RDWR | O_NONBLOCK);
    • 主循环:
    • poll()对于POLLIN事件,可能一次在多个管道上。
    • read()根据需要提供尽可能多的数据,可能会在read()调用时循环。
      • 如果read() == -1errno == EAGAIN,请返回poll()步骤。
      • 如果read() == -1errno == EINTR,则read()重新遍历。
      • 如果read() == 0,那是错误的-不应O_RDWR在命名管道上发生,而应该在O_RDONLY未命名管道上发生;它表示已关闭的管道,必须将其关闭并重新打开。如果您在同一poll()事件处理循环中混合使用命名管道和未命名管道,则可能仍需要处理这种情况。
2020-06-03