MINOR: fd: pass the iocb and owner to fd_insert()

fd_insert() is currently called just after setting the owner and iocb,
but proceeding like this prevents the operation from being atomic and
requires a lock to protect the maxfd computation in another thread from
meeting an incompletely initialized FD and computing a wrong maxfd.
Fortunately for now all fdtab[].owner are set before calling fd_insert(),
and the first lock in fd_insert() enforces a memory barrier so the code
is safe.

This patch moves the initialization of the owner and iocb to fd_insert()
so that the function will be able to properly arrange its operations and
remain safe even when modified to become lockless. There's no other change
beyond the internal API.
This commit is contained in:
Willy Tarreau 2018-01-25 07:22:13 +01:00
parent fc6eea4de2
commit a9786b6f04
8 changed files with 16 additions and 32 deletions

View File

@ -111,9 +111,7 @@ static inline void conn_ctrl_init(struct connection *conn)
if (!conn_ctrl_ready(conn)) { if (!conn_ctrl_ready(conn)) {
int fd = conn->handle.fd; int fd = conn->handle.fd;
fdtab[fd].owner = conn; fd_insert(fd, conn, conn_fd_handler, tid_bit);
fdtab[fd].iocb = conn_fd_handler;
fd_insert(fd, tid_bit);
/* mark the fd as ready so as not to needlessly poll at the beginning */ /* mark the fd as ready so as not to needlessly poll at the beginning */
fd_may_recv(fd); fd_may_recv(fd);
fd_may_send(fd); fd_may_send(fd);

View File

@ -393,9 +393,11 @@ static inline void fd_update_events(int fd, int evts)
} }
/* Prepares <fd> for being polled */ /* Prepares <fd> for being polled */
static inline void fd_insert(int fd, unsigned long thread_mask) static inline void fd_insert(int fd, void *owner, void (*iocb)(int fd), unsigned long thread_mask)
{ {
HA_SPIN_LOCK(FD_LOCK, &fdtab[fd].lock); HA_SPIN_LOCK(FD_LOCK, &fdtab[fd].lock);
fdtab[fd].owner = owner;
fdtab[fd].iocb = iocb;
fdtab[fd].ev = 0; fdtab[fd].ev = 0;
fdtab[fd].update_mask &= ~tid_bit; fdtab[fd].update_mask &= ~tid_bit;
fdtab[fd].linger_risk = 0; fdtab[fd].linger_risk = 0;

View File

@ -193,9 +193,7 @@ static int dns_connect_namesaver(struct dns_nameserver *ns)
/* Add the fd in the fd list and update its parameters */ /* Add the fd in the fd list and update its parameters */
dgram->t.sock.fd = fd; dgram->t.sock.fd = fd;
fdtab[fd].owner = dgram; fd_insert(fd, dgram, dgram_fd_handler, MAX_THREADS_MASK);
fdtab[fd].iocb = dgram_fd_handler;
fd_insert(fd, (unsigned long)-1);
fd_want_recv(fd); fd_want_recv(fd);
return 0; return 0;
} }

View File

@ -2347,9 +2347,7 @@ void mworker_pipe_register()
return; return;
fcntl(mworker_pipe[0], F_SETFL, O_NONBLOCK); fcntl(mworker_pipe[0], F_SETFL, O_NONBLOCK);
fdtab[mworker_pipe[0]].owner = mworker_pipe; fd_insert(mworker_pipe[0], mworker_pipe, mworker_pipe_handler, MAX_THREADS_MASK);
fdtab[mworker_pipe[0]].iocb = mworker_pipe_handler;
fd_insert(mworker_pipe[0], MAX_THREADS_MASK);
fd_want_recv(mworker_pipe[0]); fd_want_recv(mworker_pipe[0]);
} }

View File

@ -48,10 +48,7 @@ int thread_sync_init(unsigned long mask)
rfd = threads_sync_pipe[0]; rfd = threads_sync_pipe[0];
fcntl(rfd, F_SETFL, O_NONBLOCK); fcntl(rfd, F_SETFL, O_NONBLOCK);
fd_insert(rfd, thread_sync_io_handler, thread_sync_io_handler, MAX_THREADS_MASK);
fdtab[rfd].owner = thread_sync_io_handler;
fdtab[rfd].iocb = thread_sync_io_handler;
fd_insert(rfd, MAX_THREADS_MASK);
all_threads_mask = mask; all_threads_mask = mask;
return 0; return 0;

View File

@ -1105,12 +1105,9 @@ int tcp_bind_listener(struct listener *listener, char *errmsg, int errlen)
listener->fd = fd; listener->fd = fd;
listener->state = LI_LISTEN; listener->state = LI_LISTEN;
fdtab[fd].owner = listener; /* reference the listener instead of a task */ fd_insert(fd, listener, listener->proto->accept,
fdtab[fd].iocb = listener->proto->accept; listener->bind_conf->bind_thread[relative_pid-1] ?
if (listener->bind_conf->bind_thread[relative_pid-1]) listener->bind_conf->bind_thread[relative_pid-1] : MAX_THREADS_MASK);
fd_insert(fd, listener->bind_conf->bind_thread[relative_pid-1]);
else
fd_insert(fd, MAX_THREADS_MASK);
tcp_return: tcp_return:
if (msg && errlen) { if (msg && errlen) {

View File

@ -331,13 +331,10 @@ static int uxst_bind_listener(struct listener *listener, char *errmsg, int errle
listener->fd = fd; listener->fd = fd;
listener->state = LI_LISTEN; listener->state = LI_LISTEN;
/* the function for the accept() event */ fd_insert(fd, listener, listener->proto->accept,
fdtab[fd].iocb = listener->proto->accept; listener->bind_conf->bind_thread[relative_pid-1] ?
fdtab[fd].owner = listener; /* reference the listener instead of a task */ listener->bind_conf->bind_thread[relative_pid-1] : MAX_THREADS_MASK);
if (listener->bind_conf->bind_thread[relative_pid-1])
fd_insert(fd, listener->bind_conf->bind_thread[relative_pid-1]);
else
fd_insert(fd, MAX_THREADS_MASK);
return err; return err;
err_rename: err_rename:

View File

@ -488,7 +488,7 @@ static void ssl_async_fd_free(int fd)
*/ */
static void inline ssl_async_process_fds(struct connection *conn, SSL *ssl) static void inline ssl_async_process_fds(struct connection *conn, SSL *ssl)
{ {
OSSL_ASYNC_FD add_fd[32], afd; OSSL_ASYNC_FD add_fd[32];
OSSL_ASYNC_FD del_fd[32]; OSSL_ASYNC_FD del_fd[32];
size_t num_add_fds = 0; size_t num_add_fds = 0;
size_t num_del_fds = 0; size_t num_del_fds = 0;
@ -509,10 +509,7 @@ static void inline ssl_async_process_fds(struct connection *conn, SSL *ssl)
/* We add new fds to the fdtab */ /* We add new fds to the fdtab */
for (i=0 ; i < num_add_fds ; i++) { for (i=0 ; i < num_add_fds ; i++) {
afd = add_fd[i]; fd_insert(add_fd[i], conn, ssl_async_fd_handler, tid_bit);
fdtab[afd].owner = conn;
fdtab[afd].iocb = ssl_async_fd_handler;
fd_insert(afd, tid_bit);
} }
num_add_fds = 0; num_add_fds = 0;