3

文档不清楚事件是否合并,我的测试表明它们在某些情况下但并非总是如此。

考虑man 7 epoll

由于即使使用边缘触发的 epoll,在接收到多个数据块时也可以生成多个事件, 调用者可以选择指定 EPOLLONESHOT 标志...

和问答部分:

Q7 如果 epoll_wait(2) 调用之间发生了多个事件,是合并还是单独报告?

A7 它们将被合并。

我假设手册中的第一条语句意味着在从套接字读取、数据包到达、您读取它、然后另一个数据包到达等情况下,您可以接收多个 EPOLLIN 事件。Q&A 部分的答案是讨论不同的事件,如 EPOLLIN 和 EPOLLOUT。如果我错了,请纠正我。

我正在玩一些代码,以便更好地理解 epoll 是如何工作的,并且对于基于是否设置了另一个事件,它似乎对同一种事件的行为有所不同。更准确地说,如果我只等待 EPOLLIN,则多个输入会生成一个事件,但如果我同时等待 EPOLLIN 和 EPOLLOUT,则多个输入会生成多个事件。

这是我用来测试的代码:

#include <stdio.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <stdlib.h>
#include <unistd.h>

int main(int argc, char* argv[])
{
  struct epoll_event ev = {EPOLLIN|EPOLLOUT|EPOLLET};
  int epoll = epoll_create1(0);
  epoll_ctl(epoll, EPOLL_CTL_ADD, 0, &ev);

  //async stdin
  int flags = fcntl(0, F_GETFL);
  flags |= O_NONBLOCK;
  fcntl(0, F_SETFL, flags);

  while(1){
    struct epoll_event events[64];
    int n = epoll_wait(epoll, events, 64, -1);

    printf("Event count: %d\n", n);

    if(events[0].events == EPOLLIN)
      printf("EPOLLIN only\n\n");

    else
    if(events[0].events == (EPOLLIN|EPOLLOUT))
      printf("EPOLLIN and EPOLLOUT\n\n");

    else
      printf("EPOLLOUT only\n\n");

    char buffer[256];
    read(0, buffer, 256);

    sleep(1);
  }
  return 0;
}

按回车后的输出显示 EPOLLIN 和 EPOLLOUT 都收到了,此消息出现的次数与按回车的次数一样多,然后显示只有 EPOLLOUT 正在生成。

但是如果你编译没有EPOLLOUT标志的程序并多次按下return,单个事件将只报告一次。

如果我删除read呼叫,则在设置 EPOLLOUT 时继续报告 EPOLLIN,但在仅设置 EPOLLIN 时不会报告。

行为取决于它正在等待的事件还是我的测试代码有问题?如果它是依赖的,我可以放心它将来不会改变吗?

4

1 回答 1

2

我相信您正在观察未定义行为的影响,因为您正在滥用 API。

具体来说,您正在传递STDIN_FILENO(即,0epoll_ctl并要求等待EPOLLOUT只读文件描述符。可能发生的情况是操作系统试图告诉您文件描述符的写入方向有问题。

此外,当使用边沿触发模式时,您应该继续您的 I/O,直到您看到EAGAIN. 当epoll_wait操作不再阻塞时调用返回。

我修改了您的程序以改用套接字,并从套接字读取 until EAGAIN,它的行为与我预期的一样。

在我的版本中,我创建了一个套接字对,以及一个读取STDIN_FILENO和写入其中一个套接字的线程。然后main主体循环epoll_wait位于另一个套接字上。

当我启动程序时,它在第一次调用时返回epoll_wait以报告可写,但在下一次迭代时阻塞:

Event count: 1
EPOLLOUT only

当我输入输入时,它报告可读和可写,然后epoll_wait在下一次迭代中阻塞,如预期的那样:

asdf
Event count: 1
EPOLLIN and EPOLLOUT

我使用的代码如下。首先,线程:

static void * iothread (void *svp) {
    int *sv = svp;
    char buf[256];
    ssize_t r;
again:
    while ((r = read(0, buf, sizeof(buf))) > 0) {
        ssize_t n = r;
        const char *p = buf;
        while (n > 0) {
            r = write(sv[1], p, n);
            if (r < 0) {
                if (errno == EINTR) continue;
                break;
            }
            n -= r;
            p += r;
        }
        if (n > 0) break;
    }
    if (r < 0 && errno == EINTR) {
        goto again;
    }
    close(sv[1]);
    return NULL;
}

然后,main身体:

int main(int argc, char* argv[]) {
  int sv[2];
  struct epoll_event ev = {EPOLLIN | EPOLLOUT | EPOLLET};
  int epoll = epoll_create1(0);
  pthread_t t;

  socketpair(AF_LOCAL, SOCK_STREAM, 0, sv);
  pthread_create(&t, NULL, iothread, sv);
  epoll_ctl(epoll, EPOLL_CTL_ADD, sv[0], &ev);
  while(1){
    struct epoll_event events[64];
    int n = epoll_wait(epoll, events, 64, -1);

    printf("Event count: %d\n", n);

    if(events[0].events == EPOLLIN)
      printf("EPOLLIN only\n\n");
    else
    if(events[0].events == (EPOLLIN|EPOLLOUT))
      printf("EPOLLIN and EPOLLOUT\n\n");
    else
      printf("EPOLLOUT only\n\n");

    char buffer[256];
    ssize_t r;
again:
    r = recv(sv[0], buffer, 256, MSG_DONTWAIT);
    if (r > 0) goto again;
    if (r < 0 && errno == EAGAIN) {
        sleep(1);
        continue;
    }
    break;
  }
  return 0;
}
于 2019-09-05T17:26:17.917 回答