summaryrefslogtreecommitdiff
path: root/src/fdevent_libev.c
blob: 993c17da183aea53fd048a8cee2f11167abc92a3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#include "first.h"

#include <string.h>
#include <stdlib.h>

#include "fdevent_impl.h"
#include "fdevent.h"
#include "buffer.h"

#ifdef FDEVENT_USE_LIBEV

# include <ev.h>

static void io_watcher_cb(struct ev_loop *loop, ev_io *w, int revents) {
	fdevents *ev = w->data;
	fdnode *fdn = ev->fdarray[w->fd];
	int rv = 0;
	UNUSED(loop);

	if (revents & EV_READ) rv |= FDEVENT_IN;
	if (revents & EV_WRITE) rv |= FDEVENT_OUT;
	if (revents & EV_ERROR) rv |= FDEVENT_ERR;

	if (0 == ((uintptr_t)fdn & 0x3)) {
		(*fdn->handler)(fdn->ctx, rv);
	}
}

__attribute_cold__
static void fdevent_libev_free(fdevents *ev) {
	UNUSED(ev);
}

static int fdevent_libev_event_del(fdevents *ev, fdnode *fdn) {
	ev_io *watcher = fdn->handler_ctx;
	if (!watcher) return 0;
	fdn->handler_ctx = NULL;

	ev_io_stop(ev->libev_loop, watcher);
	free(watcher);

	return 0;
}

static int fdevent_libev_event_set(fdevents *ev, fdnode *fdn, int events) {
	ev_io *watcher = fdn->handler_ctx;
	int ev_events = 0;

	if (events & FDEVENT_IN)  ev_events |= EV_READ;
	if (events & FDEVENT_OUT) ev_events |= EV_WRITE;

	if (!watcher) {
		fdn->handler_ctx = watcher = calloc(1, sizeof(ev_io));
		force_assert(watcher);
		fdn->fde_ndx = fdn->fd;

		ev_io_init(watcher, io_watcher_cb, fdn->fd, ev_events);
		watcher->data = ev;
		ev_io_start(ev->libev_loop, watcher);
	} else {
		if ((watcher->events & (EV_READ | EV_WRITE)) != ev_events) {
			ev_io_stop(ev->libev_loop, watcher);
			ev_io_set(watcher, watcher->fd, ev_events);
			ev_io_start(ev->libev_loop, watcher);
		}
	}

	return 0;
}

static void timeout_watcher_cb(struct ev_loop *loop, ev_timer *w, int revents) {
	UNUSED(loop);
	UNUSED(w);
	UNUSED(revents);
}

static ev_timer timeout_watcher;

static int fdevent_libev_poll(fdevents *ev, int timeout_ms) {
	timeout_watcher.repeat = (timeout_ms > 0) ? timeout_ms/1000.0 : 0.001;

	ev_timer_again(ev->libev_loop, &timeout_watcher);
	ev_run(ev->libev_loop, EVRUN_ONCE);

	return 0;
}

__attribute_cold__
static int fdevent_libev_reset(fdevents *ev) {
	UNUSED(ev);
	ev_default_fork();
	return 0;
}

__attribute_cold__
int fdevent_libev_init(fdevents *ev) {
	struct ev_timer * const timer = &timeout_watcher;
	memset(timer, 0, sizeof(*timer));

	ev->type      = FDEVENT_HANDLER_LIBEV;
	ev->event_set = fdevent_libev_event_set;
	ev->event_del = fdevent_libev_event_del;
	ev->poll      = fdevent_libev_poll;
	ev->reset     = fdevent_libev_reset;
	ev->free      = fdevent_libev_free;

	if (NULL == (ev->libev_loop = ev_default_loop(0))) return -1;

	ev_timer_init(timer, timeout_watcher_cb, 0.0, 1.0);

	return 0;
}

#endif