gpt4 book ai didi

linux - 无法让 eventfd 与 epoll 一起工作

转载 作者:太空狗 更新时间:2023-10-29 11:06:02 27 4
gpt4 key购买 nike

我正在编写一个基于 epoll 的简单服务器类。为了唤醒epoll_wait(),我决定使用一个eventfd。据说它更适合简单的事件通信,我同意这一点。所以我创建了我的事件并在上面放了一个 watch :

_epollfd = epoll_create1(0);
if (_epollfd == -1) throw ServerError("epoll_create");
_eventfd = eventfd(0, EFD_NONBLOCK);
epoll_event evnt = {0};
evnt.data.fd = _eventfd;
evnt.events = _events;
if (epoll_ctl(_epollfd, EPOLL_CTL_ADD, _eventfd, &evnt) == -1)
throw ServerError("epoll_ctl(add)");

稍后在消息等待循环中,在一个单独的线程上:

    int count = epoll_wait(_epollfd, evnts, EVENTS, -1);
if (count == -1)
{
if (errno != EINTR)
{
perror("epoll_wait");
return;
}
}

for (int i = 0; i < count; ++i)
{
epoll_event & e = evnts[i];
if (e.data.fd == _serverSock)
connectionAccepted();
else if (e.data.fd == _eventfd)
{
eventfd_t val;
eventfd_read(_eventfd, &val);
return;
}
}

当然,停止服务器的代码是:

eventfd_write(_eventfd, 1);

由于我无法解释的原因,我无法通过写入事件来唤醒 epoll_wait()。最终,这在几个调试 session 中奏效了。

这是我的解决方法:知道每次 fd 可用于写入时 EPOLLOUT 都会触发一个事件,我将停止代码更改为

epoll_event evnt = {0};
evnt.data.fd = _eventfd;
evnt.events = EPOLLOUT;
if (epoll_ctl(_epollfd, EPOLL_CTL_MOD, _eventfd, &evnt) == -1)
throw ServerError("epoll_ctl(mod)");

现在可以了,但不应该这样。

我不认为这应该有任何困难。我做错了什么?

谢谢

最佳答案

适合我。作为引用,这里是完整的 C 代码:它打印“eventfd_write”、“1”和“DING: 1”。在 Linux 2.6.35-30-generic#56-Ubuntu SMP 上测试。

#include <stdio.h>
#include <errno.h>
#include <sys/epoll.h>
#include <sys/eventfd.h>
#include <pthread.h>
#include <stdlib.h>

int _epollfd, _eventfd;

int init()
{
_epollfd = epoll_create1(0);
if (_epollfd == -1) abort();
_eventfd = eventfd(0, EFD_NONBLOCK);
struct epoll_event evnt = {0};
evnt.data.fd = _eventfd;
evnt.events = EPOLLIN | EPOLLET;
if (epoll_ctl(_epollfd, EPOLL_CTL_ADD, _eventfd, &evnt) == -1)
abort();
}

void *subprocess(void *arg)
{
static const int EVENTS = 20;
struct epoll_event evnts[EVENTS];
while (1) {
int count = epoll_wait(_epollfd, evnts, EVENTS, -1);
printf("%d\n", count);
if (count == -1)
{
if (errno != EINTR)
{
perror("epoll_wait");
return NULL;
}
}

int i;
for (i = 0; i < count; ++i)
{
struct epoll_event *e = evnts + i;
if (e->data.fd == _eventfd)
{
eventfd_t val;
eventfd_read(_eventfd, &val);
printf("DING: %lld\n", (long long)val);
return NULL;
}
}
}
}

int main()
{
pthread_t th;
init();
if (pthread_create(&th, NULL, subprocess, NULL) != 0)
abort();
sleep(2);
printf("eventfd_write\n");
eventfd_write(_eventfd, 1);
sleep(2);
}

关于linux - 无法让 eventfd 与 epoll 一起工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5355791/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com