aboutsummaryrefslogtreecommitdiff
path: root/src/pevent.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/pevent.c')
-rw-r--r--src/pevent.c98
1 files changed, 98 insertions, 0 deletions
diff --git a/src/pevent.c b/src/pevent.c
new file mode 100644
index 0000000..a8845d3
--- /dev/null
+++ b/src/pevent.c
@@ -0,0 +1,98 @@
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+#include <signal.h>
+#include <sys/epoll.h>
+#include <assert.h>
+
+#include "pevent.h"
+#include "log.h"
+
+
+void event_init(event_ctx **ctx)
+{
+ assert(ctx);
+ if (!*ctx)
+ *ctx = (event_ctx *) malloc(sizeof(**ctx));
+ assert(*ctx);
+
+ memset(*ctx, 0, sizeof(**ctx));
+ (*ctx)->epoll_fd = -1;
+}
+
+int event_setup(event_ctx *ctx)
+{
+ assert(ctx);
+
+ if (ctx->epoll_fd < 0)
+ ctx->epoll_fd = epoll_create1(0);
+
+ return ctx->epoll_fd < 0;
+}
+
+int event_add_sock(event_ctx *ctx, psocket *sock)
+{
+ int s;
+ struct epoll_event ev = {0,{0}};
+
+ assert(ctx && sock);
+
+ ev.data.fd = sock->fd;
+ ev.events = EPOLLIN | EPOLLET;
+ s = epoll_ctl(ctx->epoll_fd, EPOLL_CTL_ADD, sock->fd, &ev);
+ if (s)
+ return 1;
+
+ return 0;
+}
+
+int event_add_fd(event_ctx *ctx, int fd)
+{
+ int s;
+ struct epoll_event ev = {0,{0}};
+
+ assert(ctx);
+
+ ev.data.fd = fd;
+ ev.events = EPOLLIN | EPOLLET;
+ s = epoll_ctl(ctx->epoll_fd, EPOLL_CTL_ADD, fd, &ev);
+ if (s)
+ return 1;
+
+ return 0;
+}
+
+int event_loop(event_ctx *ctx, on_event_cb on_event, void *user_data)
+{
+ int n, i;
+ sigset_t eset;
+
+ assert(ctx && on_event);
+ sigemptyset(&eset);
+ ctx->active = 1;
+
+ while (ctx->active) {
+ n = epoll_pwait(ctx->epoll_fd, ctx->events, POTD_MAXEVENTS, -1, &eset);
+ if (n < 0)
+ break;
+
+ for (i = 0; i < n; ++i) {
+ if ((ctx->events[i].events & EPOLLERR) ||
+ (ctx->events[i].events & EPOLLHUP) ||
+ (!(ctx->events[i].events & EPOLLIN)))
+ {
+ E_STRERR("Event epoll for descriptor %d",
+ ctx->events[i].data.fd);
+ close(ctx->events[i].data.fd);
+ continue;
+ } else {
+ if (on_event(ctx->events[i].data.fd, user_data))
+ continue;
+ W2("Event callback failed: [fd: %d , npoll: %d]",
+ ctx->events[i].data.fd, n);
+ }
+ }
+ }
+
+ return ctx->active == 0;
+}