diff options
author | Kurt Miller <kurt@cvs.openbsd.org> | 2006-09-26 14:18:29 +0000 |
---|---|---|
committer | Kurt Miller <kurt@cvs.openbsd.org> | 2006-09-26 14:18:29 +0000 |
commit | 151ccbfafca99c73389fbc25052c4853f6a36356 (patch) | |
tree | 175dc9fae5c2053fb3d5e430501cea32b0a43626 /lib/libpthread/uthread/uthread_dup2.c | |
parent | eb1c7bb141724021a7aacdf3f2d9931a6150fcaf (diff) |
Part 2 of file descriptor race and deadlock corrections.
Adjust design of file descriptor table to eliminate races
with both opening and closing of file descriptor entries
and eliminates one class of deadlocks. One nice side effect
of this change in design should be better performance for
applications that open and close many file descriptors due
to reduced fd_table_lock contention and fd entry reuse.
- Add entry states to manage use of entry and eliminate
some closing races. fd entries are not deallocated upon
close() now.
- Call _thread_fd_table_init with one of five discreet
modes to properly initialize an entry and manage the
state transition to open.
- When closing an entry hold the entry spinlock locked
across the state transition and the _thread_sys_close
call to close another race.
- Introduce a new lock type FD_RDWR_CLOSE that transitions
either a closed entry or an open entry into closing state
and then waits for a RDWR lock so that the lock queue can
unwind normally. All subsequent fd lock attempts for that
entry are rejected with EBADF until the fd is fully closed,
or reopened by dup2(). Once a thread holds the FD_RDWR_LOCK
it is safe to close() it or dup2() on it.
- When a thread creates a new fd there is a window of time
when another thread could attempt to use the fd before the
creating thread has initialized the entry for it. This can
result in improper status_flags for the entry, so record
the entries init mode, detect when this has happened and
correct the status_flags when needed.
reviewed by marc@ & brad@, tested by several, okay brad@
Diffstat (limited to 'lib/libpthread/uthread/uthread_dup2.c')
-rw-r--r-- | lib/libpthread/uthread/uthread_dup2.c | 21 |
1 files changed, 9 insertions, 12 deletions
diff --git a/lib/libpthread/uthread/uthread_dup2.c b/lib/libpthread/uthread/uthread_dup2.c index d8ef28d729d..ea1b88c7c5e 100644 --- a/lib/libpthread/uthread/uthread_dup2.c +++ b/lib/libpthread/uthread/uthread_dup2.c @@ -1,4 +1,4 @@ -/* $OpenBSD: uthread_dup2.c,v 1.8 2006/09/22 19:04:33 kurt Exp $ */ +/* $OpenBSD: uthread_dup2.c,v 1.9 2006/09/26 14:18:28 kurt Exp $ */ /* PUBLIC DOMAIN <marc@snafu.org> */ #include <errno.h> @@ -12,25 +12,22 @@ dup2(int fd, int newfd) { int ret; int saved_errno; - int newfd_opened; if (newfd >= 0 && newfd < _thread_dtablesize && newfd != _thread_kern_pipe[0] && newfd != _thread_kern_pipe[1]) { ret = _FD_LOCK(fd, FD_RDWR, NULL); if (ret == 0) { - newfd_opened = _thread_fd_table[newfd] != NULL; - if (!newfd_opened || - (ret = _FD_LOCK(newfd, FD_RDWR, NULL)) == 0) { + if ((ret = _FD_LOCK(newfd, FD_RDWR_CLOSE, NULL)) == 0) { /* * If newfd was open then drop reference * and reset flags if needed. */ - if (newfd_opened) - _thread_fs_flags_replace(newfd, NULL); + _thread_fs_flags_replace(newfd, NULL); ret = _thread_sys_dup2(fd, newfd); if (ret != -1) - ret = _thread_fd_table_init(newfd, + ret = _thread_fd_table_init(newfd, FD_INIT_DUP2, _thread_fd_table[fd]->status_flags); + /* * If the dup2 or the _thread_fd_table_init * failed we've already removed the status @@ -40,15 +37,15 @@ dup2(int fd, int newfd) */ if (ret == -1) { saved_errno = errno; + _SPINLOCK(&_thread_fd_table[newfd]->lock); - if (newfd_opened) - _thread_fd_table_remove(newfd); + _thread_fd_entry_close(newfd); _thread_sys_close(newfd); + _SPINUNLOCK(&_thread_fd_table[newfd]->lock); errno = saved_errno ; - } else if (newfd_opened) { - _FD_UNLOCK(newfd, FD_RDWR); } + _FD_UNLOCK(newfd, FD_RDWR_CLOSE); } _FD_UNLOCK(fd, FD_RDWR); } |