mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-01-09 23:27:17 +01:00
736a80db08
In particular, we're going to capture SIGUSR1. Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
324 lines
6.7 KiB
C
324 lines
6.7 KiB
C
/* Licensed under LGPLv2.1+ - see LICENSE file for details */
|
|
#include "io.h"
|
|
#include "backend.h"
|
|
#include <assert.h>
|
|
#include <poll.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <ccan/time/time.h>
|
|
#include <ccan/timer/timer.h>
|
|
|
|
static size_t num_fds = 0, max_fds = 0, num_waiting = 0;
|
|
static struct pollfd *pollfds = NULL;
|
|
static struct fd **fds = NULL;
|
|
static LIST_HEAD(closing);
|
|
static LIST_HEAD(always);
|
|
static struct timemono (*nowfn)(void) = time_mono;
|
|
static int (*pollfn)(struct pollfd *fds, nfds_t nfds, int timeout) = poll;
|
|
|
|
struct timemono (*io_time_override(struct timemono (*now)(void)))(void)
|
|
{
|
|
struct timemono (*old)(void) = nowfn;
|
|
nowfn = now;
|
|
return old;
|
|
}
|
|
|
|
int (*io_poll_override(int (*poll)(struct pollfd *fds, nfds_t nfds, int timeout)))(struct pollfd *, nfds_t, int)
|
|
{
|
|
int (*old)(struct pollfd *fds, nfds_t nfds, int timeout) = pollfn;
|
|
pollfn = poll;
|
|
return old;
|
|
}
|
|
|
|
static bool add_fd(struct fd *fd, short events)
|
|
{
|
|
if (!max_fds) {
|
|
assert(num_fds == 0);
|
|
pollfds = tal_arr(NULL, struct pollfd, 8);
|
|
if (!pollfds)
|
|
return false;
|
|
fds = tal_arr(pollfds, struct fd *, 8);
|
|
if (!fds)
|
|
return false;
|
|
max_fds = 8;
|
|
}
|
|
|
|
if (num_fds + 1 > max_fds) {
|
|
size_t num = max_fds * 2;
|
|
|
|
if (!tal_resize(&pollfds, num))
|
|
return false;
|
|
if (!tal_resize(&fds, num))
|
|
return false;
|
|
max_fds = num;
|
|
}
|
|
|
|
pollfds[num_fds].events = events;
|
|
/* In case it's idle. */
|
|
if (!events)
|
|
pollfds[num_fds].fd = -fd->fd;
|
|
else
|
|
pollfds[num_fds].fd = fd->fd;
|
|
pollfds[num_fds].revents = 0; /* In case we're iterating now */
|
|
fds[num_fds] = fd;
|
|
fd->backend_info = num_fds;
|
|
num_fds++;
|
|
if (events)
|
|
num_waiting++;
|
|
|
|
return true;
|
|
}
|
|
|
|
static void del_fd(struct fd *fd)
|
|
{
|
|
size_t n = fd->backend_info;
|
|
|
|
assert(n != -1);
|
|
assert(n < num_fds);
|
|
if (pollfds[n].events)
|
|
num_waiting--;
|
|
if (n != num_fds - 1) {
|
|
/* Move last one over us. */
|
|
pollfds[n] = pollfds[num_fds-1];
|
|
fds[n] = fds[num_fds-1];
|
|
assert(fds[n]->backend_info == num_fds-1);
|
|
fds[n]->backend_info = n;
|
|
} else if (num_fds == 1) {
|
|
/* Free everything when no more fds. */
|
|
pollfds = tal_free(pollfds);
|
|
fds = NULL;
|
|
max_fds = 0;
|
|
}
|
|
num_fds--;
|
|
fd->backend_info = -1;
|
|
}
|
|
|
|
static void destroy_listener(struct io_listener *l)
|
|
{
|
|
close(l->fd.fd);
|
|
del_fd(&l->fd);
|
|
}
|
|
|
|
bool add_listener(struct io_listener *l)
|
|
{
|
|
if (!add_fd(&l->fd, POLLIN))
|
|
return false;
|
|
tal_add_destructor(l, destroy_listener);
|
|
return true;
|
|
}
|
|
|
|
void remove_from_always(struct io_conn *conn)
|
|
{
|
|
list_del_init(&conn->always);
|
|
}
|
|
|
|
void backend_new_always(struct io_conn *conn)
|
|
{
|
|
/* In case it's already in always list. */
|
|
list_del(&conn->always);
|
|
list_add_tail(&always, &conn->always);
|
|
}
|
|
|
|
void backend_new_plan(struct io_conn *conn)
|
|
{
|
|
struct pollfd *pfd = &pollfds[conn->fd.backend_info];
|
|
|
|
if (pfd->events)
|
|
num_waiting--;
|
|
|
|
pfd->events = 0;
|
|
if (conn->plan[IO_IN].status == IO_POLLING_NOTSTARTED
|
|
|| conn->plan[IO_IN].status == IO_POLLING_STARTED)
|
|
pfd->events |= POLLIN;
|
|
if (conn->plan[IO_OUT].status == IO_POLLING_NOTSTARTED
|
|
|| conn->plan[IO_OUT].status == IO_POLLING_STARTED)
|
|
pfd->events |= POLLOUT;
|
|
|
|
if (pfd->events) {
|
|
num_waiting++;
|
|
pfd->fd = conn->fd.fd;
|
|
} else {
|
|
pfd->fd = -conn->fd.fd;
|
|
}
|
|
}
|
|
|
|
void backend_wake(const void *wait)
|
|
{
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < num_fds; i++) {
|
|
struct io_conn *c;
|
|
|
|
/* Ignore listeners */
|
|
if (fds[i]->listener)
|
|
continue;
|
|
|
|
c = (void *)fds[i];
|
|
if (c->plan[IO_IN].status == IO_WAITING
|
|
&& c->plan[IO_IN].arg.u1.const_vp == wait)
|
|
io_do_wakeup(c, IO_IN);
|
|
|
|
if (c->plan[IO_OUT].status == IO_WAITING
|
|
&& c->plan[IO_OUT].arg.u1.const_vp == wait)
|
|
io_do_wakeup(c, IO_OUT);
|
|
}
|
|
}
|
|
|
|
static void destroy_conn(struct io_conn *conn, bool close_fd)
|
|
{
|
|
int saved_errno = errno;
|
|
|
|
if (close_fd)
|
|
close(conn->fd.fd);
|
|
del_fd(&conn->fd);
|
|
/* In case it's on always list, remove it. */
|
|
list_del_init(&conn->always);
|
|
|
|
/* errno saved/restored by tal_free itself. */
|
|
if (conn->finish) {
|
|
errno = saved_errno;
|
|
conn->finish(conn, conn->finish_arg);
|
|
}
|
|
}
|
|
|
|
static void destroy_conn_close_fd(struct io_conn *conn)
|
|
{
|
|
destroy_conn(conn, true);
|
|
}
|
|
|
|
bool add_conn(struct io_conn *c)
|
|
{
|
|
if (!add_fd(&c->fd, 0))
|
|
return false;
|
|
tal_add_destructor(c, destroy_conn_close_fd);
|
|
return true;
|
|
}
|
|
|
|
void cleanup_conn_without_close(struct io_conn *conn)
|
|
{
|
|
tal_del_destructor(conn, destroy_conn_close_fd);
|
|
destroy_conn(conn, false);
|
|
}
|
|
|
|
static void accept_conn(struct io_listener *l)
|
|
{
|
|
int fd = accept(l->fd.fd, NULL, NULL);
|
|
|
|
/* FIXME: What to do here? */
|
|
if (fd < 0)
|
|
return;
|
|
|
|
io_new_conn(l->ctx, fd, l->init, l->arg);
|
|
}
|
|
|
|
static bool handle_always(void)
|
|
{
|
|
bool ret = false;
|
|
struct io_conn *conn;
|
|
|
|
while ((conn = list_pop(&always, struct io_conn, always)) != NULL) {
|
|
assert(conn->plan[IO_IN].status == IO_ALWAYS
|
|
|| conn->plan[IO_OUT].status == IO_ALWAYS);
|
|
|
|
/* Re-initialize, for next time. */
|
|
list_node_init(&conn->always);
|
|
io_do_always(conn);
|
|
ret = true;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/* This is the main loop. */
|
|
void *io_loop(struct timers *timers, struct timer **expired)
|
|
{
|
|
void *ret;
|
|
|
|
/* if timers is NULL, expired must be. If not, not. */
|
|
assert(!timers == !expired);
|
|
|
|
/* Make sure this is NULL if we exit for some other reason. */
|
|
if (expired)
|
|
*expired = NULL;
|
|
|
|
while (!io_loop_return) {
|
|
int i, r, ms_timeout = -1;
|
|
|
|
if (handle_always()) {
|
|
/* Could have started/finished more. */
|
|
continue;
|
|
}
|
|
|
|
/* Everything closed? */
|
|
if (num_fds == 0)
|
|
break;
|
|
|
|
/* You can't tell them all to go to sleep! */
|
|
assert(num_waiting);
|
|
|
|
if (timers) {
|
|
struct timemono now, first;
|
|
|
|
now = nowfn();
|
|
|
|
/* Call functions for expired timers. */
|
|
*expired = timers_expire(timers, now);
|
|
if (*expired)
|
|
break;
|
|
|
|
/* Now figure out how long to wait for the next one. */
|
|
if (timer_earliest(timers, &first)) {
|
|
uint64_t next;
|
|
next = time_to_msec(timemono_between(first, now));
|
|
if (next < INT_MAX)
|
|
ms_timeout = next;
|
|
else
|
|
ms_timeout = INT_MAX;
|
|
}
|
|
}
|
|
|
|
r = pollfn(pollfds, num_fds, ms_timeout);
|
|
if (r < 0) {
|
|
/* Signals shouldn't break us, unless they set
|
|
* io_loop_return. */
|
|
if (errno == EINTR)
|
|
continue;
|
|
break;
|
|
}
|
|
|
|
for (i = 0; i < num_fds && !io_loop_return; i++) {
|
|
struct io_conn *c = (void *)fds[i];
|
|
int events = pollfds[i].revents;
|
|
|
|
if (r == 0)
|
|
break;
|
|
|
|
if (fds[i]->listener) {
|
|
struct io_listener *l = (void *)fds[i];
|
|
if (events & POLLIN) {
|
|
accept_conn(l);
|
|
r--;
|
|
} else if (events & (POLLHUP|POLLNVAL|POLLERR)) {
|
|
r--;
|
|
errno = EBADF;
|
|
io_close_listener(l);
|
|
}
|
|
} else if (events & (POLLIN|POLLOUT)) {
|
|
r--;
|
|
io_ready(c, events);
|
|
} else if (events & (POLLHUP|POLLNVAL|POLLERR)) {
|
|
r--;
|
|
errno = EBADF;
|
|
io_close(c);
|
|
}
|
|
}
|
|
}
|
|
|
|
ret = io_loop_return;
|
|
io_loop_return = NULL;
|
|
|
|
return ret;
|
|
}
|