ioloop-epoll.c revision 6795f542ed816a3c977085d4f74df1d62a37b690
/*
* Linux epoll() based ioloop handler.
*
* Copyright (c) 2004 Andrey Panin <pazke@donpac.ru>
*
* This software is released under the MIT license.
*/
#include "lib.h"
#include "array.h"
#include "fd-close-on-exec.h"
#include "ioloop-internal.h"
#include "ioloop-iolist.h"
#ifdef IOLOOP_EPOLL
#include <unistd.h>
struct ioloop_handler_context {
int epfd;
unsigned int deleted_count;
};
{
struct ioloop_handler_context *ctx;
i_fatal("epoll_create(): %m");
else {
i_fatal("epoll_create(): %m (you may need to increase "
"/proc/sys/fs/epoll/max_user_instances)");
}
}
}
{
unsigned int i, count;
for (i = 0; i < count; i++)
i_error("close(epoll) failed: %m");
}
{
int events = 0, i;
for (i = 0; i < IOLOOP_IOLIST_IOS_PER_FD; i++) {
continue;
events |= IO_EPOLL_INPUT;
events |= IO_EPOLL_ERROR;
}
return events;
}
{
struct epoll_event event;
int op;
bool first;
i_fatal("io_loop_handle_add: epoll_ctl(%d, %d): %m",
}
if (first) {
/* allow epoll_wait() to return the maximum number of events
by keeping space allocated for each file descriptor */
if (ctx->deleted_count > 0)
ctx->deleted_count--;
else
}
}
{
struct epoll_event event;
int op;
bool last;
if (!closed) {
i_error("io_loop_handle_remove: epoll_ctl(%d, %d): %m",
}
}
if (last) {
/* since we're not freeing memory in any case, just increase
deleted counter so next handle_add() can just decrease it
insteading of appending to the events array */
ctx->deleted_count++;
}
}
{
struct epoll_event *events;
const struct epoll_event *event;
unsigned int events_count, t_id;
bool call;
/* get the time left for next timeout task */
i_fatal("epoll_wait(): %m");
/* execute timeout handlers */
return;
for (i = 0; i < ret; i++) {
/* io_loop_handle_add() may cause events array reallocation,
so we have use array_idx() */
for (j = 0; j < IOLOOP_IOLIST_IOS_PER_FD; j++) {
continue;
if (call) {
i_panic("Leaked a t_pop() call in "
"I/O handler %p",
}
}
}
}
}
#endif /* IOLOOP_EPOLL */