lighttpd 1.4.x
https://www.lighttpd.net/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
227 lines
4.4 KiB
227 lines
4.4 KiB
#include "fdevent.h" |
|
#include "buffer.h" |
|
#include "log.h" |
|
|
|
#include <sys/types.h> |
|
|
|
#include <unistd.h> |
|
#include <stdlib.h> |
|
#include <stdio.h> |
|
#include <string.h> |
|
#include <errno.h> |
|
#include <signal.h> |
|
#include <fcntl.h> |
|
|
|
#ifdef USE_FREEBSD_KQUEUE |
|
# include <sys/event.h> |
|
# include <sys/time.h> |
|
|
|
static void fdevent_freebsd_kqueue_free(fdevents *ev) { |
|
close(ev->kq_fd); |
|
free(ev->kq_results); |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_event_del(fdevents *ev, int fde_ndx, int fd) { |
|
int ret, n = 0; |
|
struct kevent kev[2]; |
|
struct timespec ts; |
|
int oevents; |
|
|
|
if (fde_ndx < 0) return -1; |
|
|
|
oevents = ev->fdarray[fd]->events; |
|
|
|
if (oevents & FDEVENT_IN) { |
|
EV_SET(&kev[n], fd, EVFILT_READ, EV_DELETE, 0, 0, NULL); |
|
n++; |
|
} |
|
if (oevents & FDEVENT_OUT) { |
|
EV_SET(&kev[n], fd, EVFILT_WRITE, EV_DELETE, 0, 0, NULL); |
|
n++; |
|
} |
|
|
|
if (0 == n) return -1; |
|
|
|
ts.tv_sec = 0; |
|
ts.tv_nsec = 0; |
|
|
|
ret = kevent(ev->kq_fd, |
|
kev, n, |
|
NULL, 0, |
|
&ts); |
|
|
|
if (ret == -1) { |
|
log_error_write(ev->srv, __FILE__, __LINE__, "SS", |
|
"kqueue event delete failed: ", strerror(errno)); |
|
|
|
return -1; |
|
} |
|
|
|
return -1; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_event_set(fdevents *ev, int fde_ndx, int fd, int events) { |
|
int ret, n = 0; |
|
struct kevent kev[2]; |
|
struct timespec ts; |
|
int oevents = ev->fdarray[fd]->events; |
|
int addevents = events & ~oevents; |
|
int delevents = ~events & oevents; |
|
|
|
UNUSED(fde_ndx); |
|
|
|
if (events == oevents) return fd; |
|
|
|
if (addevents & FDEVENT_IN) { |
|
EV_SET(&kev[n], fd, EVFILT_READ, EV_ADD|EV_CLEAR, 0, 0, NULL); |
|
n++; |
|
} else if (delevents & FDEVENT_IN) { |
|
EV_SET(&kev[n], fd, EVFILT_READ, EV_DELETE, 0, 0, NULL); |
|
n++; |
|
} |
|
if (addevents & FDEVENT_OUT) { |
|
EV_SET(&kev[n], fd, EVFILT_WRITE, EV_ADD|EV_CLEAR, 0, 0, NULL); |
|
n++; |
|
} else if (delevents & FDEVENT_OUT) { |
|
EV_SET(&kev[n], fd, EVFILT_WRITE, EV_DELETE, 0, 0, NULL); |
|
n++; |
|
} |
|
|
|
if (0 == n) return fd; |
|
|
|
ts.tv_sec = 0; |
|
ts.tv_nsec = 0; |
|
|
|
ret = kevent(ev->kq_fd, |
|
kev, n, |
|
NULL, 0, |
|
&ts); |
|
|
|
if (ret == -1) { |
|
log_error_write(ev->srv, __FILE__, __LINE__, "SS", |
|
"kqueue event set failed: ", strerror(errno)); |
|
|
|
return -1; |
|
} |
|
|
|
return fd; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_poll(fdevents *ev, int timeout_ms) { |
|
int ret; |
|
struct timespec ts; |
|
|
|
ts.tv_sec = timeout_ms / 1000; |
|
ts.tv_nsec = (timeout_ms % 1000) * 1000000; |
|
|
|
ret = kevent(ev->kq_fd, |
|
NULL, 0, |
|
ev->kq_results, ev->maxfds, |
|
&ts); |
|
|
|
if (ret == -1) { |
|
switch(errno) { |
|
case EINTR: |
|
/* we got interrupted, perhaps just a SIGCHLD of a CGI script */ |
|
return 0; |
|
default: |
|
log_error_write(ev->srv, __FILE__, __LINE__, "SS", |
|
"kqueue failed polling: ", strerror(errno)); |
|
break; |
|
} |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_event_get_revent(fdevents *ev, size_t ndx) { |
|
int events = 0, e; |
|
|
|
e = ev->kq_results[ndx].filter; |
|
|
|
if (e == EVFILT_READ) { |
|
events |= FDEVENT_IN; |
|
} else if (e == EVFILT_WRITE) { |
|
events |= FDEVENT_OUT; |
|
} |
|
|
|
e = ev->kq_results[ndx].flags; |
|
|
|
if (e & EV_EOF) { |
|
events |= FDEVENT_HUP; |
|
} |
|
|
|
if (e & EV_ERROR) { |
|
events |= FDEVENT_ERR; |
|
} |
|
|
|
return events; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_event_get_fd(fdevents *ev, size_t ndx) { |
|
return ev->kq_results[ndx].ident; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_event_next_fdndx(fdevents *ev, int ndx) { |
|
UNUSED(ev); |
|
|
|
return (ndx < 0) ? 0 : ndx + 1; |
|
} |
|
|
|
static int fdevent_freebsd_kqueue_reset(fdevents *ev) { |
|
if (-1 == (ev->kq_fd = kqueue())) { |
|
log_error_write(ev->srv, __FILE__, __LINE__, "SSS", |
|
"kqueue failed (", strerror(errno), "), try to set server.event-handler = \"poll\" or \"select\""); |
|
|
|
return -1; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
|
|
int fdevent_freebsd_kqueue_init(fdevents *ev) { |
|
ev->type = FDEVENT_HANDLER_FREEBSD_KQUEUE; |
|
#define SET(x) \ |
|
ev->x = fdevent_freebsd_kqueue_##x; |
|
|
|
SET(free); |
|
SET(poll); |
|
SET(reset); |
|
|
|
SET(event_del); |
|
SET(event_set); |
|
|
|
SET(event_next_fdndx); |
|
SET(event_get_fd); |
|
SET(event_get_revent); |
|
|
|
ev->kq_fd = -1; |
|
|
|
ev->kq_results = calloc(ev->maxfds, sizeof(*ev->kq_results)); |
|
force_assert(NULL != ev->kq_results); |
|
|
|
/* check that kqueue works */ |
|
|
|
if (-1 == (ev->kq_fd = kqueue())) { |
|
log_error_write(ev->srv, __FILE__, __LINE__, "SSS", |
|
"kqueue failed (", strerror(errno), "), try to set server.event-handler = \"poll\" or \"select\""); |
|
|
|
return -1; |
|
} |
|
|
|
close(ev->kq_fd); |
|
ev->kq_fd = -1; |
|
|
|
return 0; |
|
} |
|
#else |
|
int fdevent_freebsd_kqueue_init(fdevents *ev) { |
|
UNUSED(ev); |
|
|
|
log_error_write(ev->srv, __FILE__, __LINE__, "S", |
|
"kqueue not available, try to set server.event-handler = \"poll\" or \"select\""); |
|
|
|
return -1; |
|
} |
|
#endif
|
|
|