[core] consolidate duplicated read-to-close code
[lighttpd.git] / src / fdevent_freebsd_kqueue.c
blobb82ccfdfd8ccf10e81ebb08d48afa8bcb20381db
1 #include "first.h"
3 #include "fdevent.h"
4 #include "buffer.h"
5 #include "log.h"
7 #include <sys/types.h>
9 #include <unistd.h>
10 #include <stdlib.h>
11 #include <stdio.h>
12 #include <string.h>
13 #include <errno.h>
14 #include <signal.h>
15 #include <fcntl.h>
17 #ifdef USE_FREEBSD_KQUEUE
18 # include <sys/event.h>
19 # include <sys/time.h>
21 static void fdevent_freebsd_kqueue_free(fdevents *ev) {
22 close(ev->kq_fd);
23 free(ev->kq_results);
26 static int fdevent_freebsd_kqueue_event_del(fdevents *ev, int fde_ndx, int fd) {
27 int ret, n = 0;
28 struct kevent kev[2];
29 struct timespec ts;
30 int oevents;
32 if (fde_ndx < 0) return -1;
34 oevents = ev->fdarray[fd]->events;
36 if (oevents & FDEVENT_IN) {
37 EV_SET(&kev[n], fd, EVFILT_READ, EV_DELETE, 0, 0, NULL);
38 n++;
40 if (oevents & FDEVENT_OUT) {
41 EV_SET(&kev[n], fd, EVFILT_WRITE, EV_DELETE, 0, 0, NULL);
42 n++;
45 if (0 == n) return -1;
47 ts.tv_sec = 0;
48 ts.tv_nsec = 0;
50 ret = kevent(ev->kq_fd,
51 kev, n,
52 NULL, 0,
53 &ts);
55 if (ret == -1) {
56 log_error_write(ev->srv, __FILE__, __LINE__, "SS",
57 "kqueue event delete failed: ", strerror(errno));
59 return -1;
62 return -1;
65 static int fdevent_freebsd_kqueue_event_set(fdevents *ev, int fde_ndx, int fd, int events) {
66 int ret, n = 0;
67 struct kevent kev[2];
68 struct timespec ts;
69 int oevents = ev->fdarray[fd]->events;
70 int addevents = events & ~oevents;
71 int delevents = ~events & oevents;
73 UNUSED(fde_ndx);
75 if (events == oevents) return fd;
77 if (addevents & FDEVENT_IN) {
78 EV_SET(&kev[n], fd, EVFILT_READ, EV_ADD|EV_CLEAR, 0, 0, NULL);
79 n++;
80 } else if (delevents & FDEVENT_IN) {
81 EV_SET(&kev[n], fd, EVFILT_READ, EV_DELETE, 0, 0, NULL);
82 n++;
84 if (addevents & FDEVENT_OUT) {
85 EV_SET(&kev[n], fd, EVFILT_WRITE, EV_ADD|EV_CLEAR, 0, 0, NULL);
86 n++;
87 } else if (delevents & FDEVENT_OUT) {
88 EV_SET(&kev[n], fd, EVFILT_WRITE, EV_DELETE, 0, 0, NULL);
89 n++;
92 if (0 == n) return fd;
94 ts.tv_sec = 0;
95 ts.tv_nsec = 0;
97 ret = kevent(ev->kq_fd,
98 kev, n,
99 NULL, 0,
100 &ts);
102 if (ret == -1) {
103 log_error_write(ev->srv, __FILE__, __LINE__, "SS",
104 "kqueue event set failed: ", strerror(errno));
106 return -1;
109 return fd;
112 static int fdevent_freebsd_kqueue_poll(fdevents *ev, int timeout_ms) {
113 int ret;
114 struct timespec ts;
116 ts.tv_sec = timeout_ms / 1000;
117 ts.tv_nsec = (timeout_ms % 1000) * 1000000;
119 ret = kevent(ev->kq_fd,
120 NULL, 0,
121 ev->kq_results, ev->maxfds,
122 &ts);
124 if (ret == -1) {
125 switch(errno) {
126 case EINTR:
127 /* we got interrupted, perhaps just a SIGCHLD of a CGI script */
128 return 0;
129 default:
130 log_error_write(ev->srv, __FILE__, __LINE__, "SS",
131 "kqueue failed polling: ", strerror(errno));
132 break;
136 return ret;
139 static int fdevent_freebsd_kqueue_event_get_revent(fdevents *ev, size_t ndx) {
140 int events = 0, e;
142 e = ev->kq_results[ndx].filter;
144 if (e == EVFILT_READ) {
145 events |= FDEVENT_IN;
146 } else if (e == EVFILT_WRITE) {
147 events |= FDEVENT_OUT;
150 e = ev->kq_results[ndx].flags;
152 if (e & EV_EOF) {
153 events |= FDEVENT_HUP;
156 if (e & EV_ERROR) {
157 events |= FDEVENT_ERR;
160 return events;
163 static int fdevent_freebsd_kqueue_event_get_fd(fdevents *ev, size_t ndx) {
164 return ev->kq_results[ndx].ident;
167 static int fdevent_freebsd_kqueue_event_next_fdndx(fdevents *ev, int ndx) {
168 UNUSED(ev);
170 return (ndx < 0) ? 0 : ndx + 1;
173 static int fdevent_freebsd_kqueue_reset(fdevents *ev) {
174 if (-1 == (ev->kq_fd = kqueue())) {
175 log_error_write(ev->srv, __FILE__, __LINE__, "SSS",
176 "kqueue failed (", strerror(errno), "), try to set server.event-handler = \"poll\" or \"select\"");
178 return -1;
181 return 0;
185 int fdevent_freebsd_kqueue_init(fdevents *ev) {
186 ev->type = FDEVENT_HANDLER_FREEBSD_KQUEUE;
187 #define SET(x) \
188 ev->x = fdevent_freebsd_kqueue_##x;
190 SET(free);
191 SET(poll);
192 SET(reset);
194 SET(event_del);
195 SET(event_set);
197 SET(event_next_fdndx);
198 SET(event_get_fd);
199 SET(event_get_revent);
201 ev->kq_fd = -1;
203 ev->kq_results = calloc(ev->maxfds, sizeof(*ev->kq_results));
204 force_assert(NULL != ev->kq_results);
206 /* check that kqueue works */
208 if (-1 == (ev->kq_fd = kqueue())) {
209 log_error_write(ev->srv, __FILE__, __LINE__, "SSS",
210 "kqueue failed (", strerror(errno), "), try to set server.event-handler = \"poll\" or \"select\"");
212 return -1;
215 close(ev->kq_fd);
216 ev->kq_fd = -1;
218 return 0;
220 #else
221 int fdevent_freebsd_kqueue_init(fdevents *ev) {
222 UNUSED(ev);
224 log_error_write(ev->srv, __FILE__, __LINE__, "S",
225 "kqueue not available, try to set server.event-handler = \"poll\" or \"select\"");
227 return -1;
229 #endif