Commit 02388bfc authored by Rusty Russell's avatar Rusty Russell

ccan/io: check for all idle.

It's probably a bug if we're waiting for nothing.
Signed-off-by: default avatarRusty Russell <rusty@rustcorp.com.au>
parent a2dffefa
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
#include <sys/types.h> #include <sys/types.h>
#include <sys/socket.h> #include <sys/socket.h>
static size_t num_fds = 0, max_fds = 0, num_next = 0, num_finished = 0; static size_t num_fds = 0, max_fds = 0, num_next = 0, num_finished = 0, num_waiting = 0;
static struct pollfd *pollfds = NULL; static struct pollfd *pollfds = NULL;
static struct fd **fds = NULL; static struct fd **fds = NULL;
...@@ -65,7 +65,10 @@ static void del_fd(struct fd *fd) ...@@ -65,7 +65,10 @@ static void del_fd(struct fd *fd)
bool add_listener(struct io_listener *l) bool add_listener(struct io_listener *l)
{ {
return add_fd(&l->fd, POLLIN); if (!add_fd(&l->fd, POLLIN))
return false;
num_waiting++;
return true;
} }
bool add_conn(struct io_conn *c) bool add_conn(struct io_conn *c)
...@@ -123,6 +126,9 @@ void backend_set_state(struct io_conn *conn, struct io_op *op) ...@@ -123,6 +126,9 @@ void backend_set_state(struct io_conn *conn, struct io_op *op)
enum io_state state = from_ioop(op); enum io_state state = from_ioop(op);
struct pollfd *pfd = &pollfds[conn->fd.backend_info]; struct pollfd *pfd = &pollfds[conn->fd.backend_info];
if (pfd->events)
num_waiting--;
pfd->events = pollmask(state); pfd->events = pollmask(state);
if (conn->duplex) { if (conn->duplex) {
int mask = pollmask(conn->duplex->state); int mask = pollmask(conn->duplex->state);
...@@ -130,6 +136,8 @@ void backend_set_state(struct io_conn *conn, struct io_op *op) ...@@ -130,6 +136,8 @@ void backend_set_state(struct io_conn *conn, struct io_op *op)
assert(!mask || pfd->events != mask); assert(!mask || pfd->events != mask);
pfd->events |= mask; pfd->events |= mask;
} }
if (pfd->events)
num_waiting++;
if (state == NEXT) if (state == NEXT)
num_next++; num_next++;
...@@ -206,6 +214,9 @@ void *io_loop(void) ...@@ -206,6 +214,9 @@ void *io_loop(void)
if (num_fds == 0) if (num_fds == 0)
break; break;
/* You can't tell them all to go to sleep! */
assert(num_waiting);
r = poll(pollfds, num_fds, -1); r = poll(pollfds, num_fds, -1);
if (r < 0) if (r < 0)
break; break;
......
#include <ccan/io/io.h>
/* Include the C files directly. */
#include <ccan/io/poll.c>
#include <ccan/io/io.c>
#include <ccan/tap/tap.h>
#include <sys/wait.h>
#include <stdio.h>
#include <signal.h>
static struct io_op *start(struct io_conn *conn, void *unused)
{
return io_idle(conn);
}
int main(void)
{
int status;
plan_tests(3);
if (fork() == 0) {
int fds[2];
ok1(pipe(fds) == 0);
io_new_conn(fds[0], start, NULL, NULL);
io_loop();
exit(1);
}
ok1(wait(&status) != -1);
ok1(WIFSIGNALED(status));
ok1(WTERMSIG(status) == SIGABRT);
/* This exits depending on whether all tests passed */
return exit_status();
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment