【问题标题】:epoll_wait always sets EPOLLOUT bit?epoll_wait 总是设置 EPOLLOUT 位?
【发布时间】:2012-11-14 03:59:44
【问题描述】:

在侦听套接字上,我设置了 EPOLLIN 位,但在客户端连接上,我将 EPOLLIN | EPOLLOUT 位设置为 struct epoll_event,如下所示:

struct epoll_event ev;

ev.data.fd = fd;
ev.events = EPOLLIN | EPOLLOUT;
if (epoll_ctl(evs->epoll_fd, EPOLL_CTL_ADD, fd, &ev) < 0)
     ...

这就是我测试位的方式:

if ((events & EPOLLIN) == EPOLLIN)
     ...
if ((events & EPOLLOUT) == EPOLLOUT)
     ...

我也试过这样:

if (events & EPOLLIN)
     ...
if (events & EPOLLOUT)
     ...

这两种方式都是正确的!

然而,每当我在我的 epoll fd 上调用 epoll_wait 时,返回的活动文件描述符总是设置两个位,即使 send() 没有返回 EAGAIN 但是当我尝试 recv() 时它返回 EAGAIN。

我不知道当 recv() 返回 EAGAIN 时我应该做什么,我应该删除 EPOLLOUT 标志还是什么?

@Nikolai N Fetissov 要求的更多代码:

static int get_active_fd(events *evs, int index, sstate_t *flags)
{
    uint32_t events = evs->events[index].events;
    int fd = evs->events[index].data.fd;;

    if ((events & EPOLLERR) == EPOLLERR || (events & EPOLLHUP) == EPOLLHUP) {
        close(fd);
        return -1;
    }

    if (events & EPOLLIN)
        *flags |= DATA_IN;

    return fd;
}

void sockset_add(events *evs, int fd)
{
    struct epoll_event ev;
    ...
    ev.data.fd = fd;
    ev.events = EPOLLIN;
    if (epoll_ctl(evs->epoll_fd, EPOLL_CTL_ADD, fd, &ev) < 0)
        eprintf("sockset_add(): epoll_ctl(%d) returned an error %d(%s)\n",
                fd, errno, strerror(errno));
}

然后在我调用 epoll_wait() 的地方:

if (flags & DATA_IN) {
       /* try to read which is impossible because this is never set.  */

【问题讨论】:

    标签: c linux sockets epoll


    【解决方案1】:

    我认为你把它弄反了。不要包含EPOLLOUT,除非您从写入尝试中获得EAGAIN,并在您成功将字节写入套接字后将其删除。

    这基本上意味着只要套接字在内核发送缓冲区中有空间,套接字就总是可写。

    编辑0:

    您有两个数据流 - inputoutput

    您正在通过在标志中包含EPOLLIN 来等待输入。如果从epoll_wait(2) 返回时未设置该标志,则说明某些事件发生在其他套接字上,或者此套接字有其他事件。将标志留在事件中,除非出现错误(意味着您仍然对套接字上的输入感兴趣)。

    您不必等待输出(因为这是您的操作),您只需写入套接字,但如果套接字发送缓冲区溢出,您将从 send(2)write(2) 获得 EAGAIN .在这种情况下,您通过包含EPOLLOUT 开始等待输出可能(内核耗尽套接字发送缓冲区,从而为您发送更多空间)。一旦你得到它,将你的挂起输出字节写入套接字,如果你成功了,从事件中删除EPOLLOUT

    现在EPOLLET 表示边缘触发等待,这意味着每次状态更改(例如从“无输入”到“有输入”)只会发出一次所需的事件信号。在这种模式下,您应该循环读取输入字节,直到获得EAGAIN

    希望这会有所帮助。

    【讨论】:

    • 我删除了位,但仍然测试位结果为真
    • 我的意思是,如果我删除 EPOLLOUT 并测试 EPOLLIN 那么它会失败,这完全令人困惑
    • 这就是我的意思 - 你很困惑,所以展示你如何设置轮询描述符以及如何测试轮询结果。混淆就在那里。
    • 好吧,让我猜猜,当不需要写入数据时(也就是当 send() 不返回 EAGAIN 时),我是否需要 EPOLLET 来替换 EPOLLOUT?编辑:现在添加了更多代码。
    • @user9000 如果要读取数据,设置 EPOLLIN ,当返回true时读取数据,如果读取返回EAGAIN,请等待下次。如果您有数据要发送,请发送。如果发送返回 EAGAIN 和/或 EWOULDBLOCK,则打开 EPOLLOUT 并在 EPOLLOUT 变为真时恢复发送。如果您没有任何东西要发送,请关闭 EPOLLOUT。很难从您的代码中看出您正在检查哪些结构中的哪些位,但您可能应该显示执行 epoll_wait() 的代码。 EPOLLET 很难正确,我建议你在完全 100% 完全理解 epoll 之前不要使用它。
    猜你喜欢
    • 2020-04-04
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-01-11
    • 1970-01-01
    • 2012-05-27
    相关资源
    最近更新 更多