diff options
-rw-r--r-- | sys/kern/sys_pipe.c | 56 |
1 files changed, 36 insertions, 20 deletions
diff --git a/sys/kern/sys_pipe.c b/sys/kern/sys_pipe.c index 2a5f5613a5d..034d7937932 100644 --- a/sys/kern/sys_pipe.c +++ b/sys/kern/sys_pipe.c @@ -1,4 +1,4 @@ -/* $OpenBSD: sys_pipe.c,v 1.98 2019/11/11 16:45:46 anton Exp $ */ +/* $OpenBSD: sys_pipe.c,v 1.99 2019/11/19 19:19:28 anton Exp $ */ /* * Copyright (c) 1996 John S. Dyson @@ -106,6 +106,8 @@ void pipe_destroy(struct pipe *); int pipe_buffer_realloc(struct pipe *, u_int); void pipe_buffer_free(struct pipe *); +int pipe_sleep(struct pipe *, const char *); + /* * The pipe system call for the DTYPE_PIPE type of pipes */ @@ -287,6 +289,30 @@ pipeunlock(struct pipe *cpipe) rw_exit(&cpipe->pipe_lock); } +/* + * Unlock the given pipe and go to sleep. Returns 0 on success and the + * pipe is relocked. Otherwise if a signal was caught, non-zero is returned and + * the pipe is not locked. + * + * Any caller must obtain a reference to the pipe by incrementing `pipe_busy' + * before calling this function in order ensure that the same pipe is not + * destroyed while sleeping. + */ +int +pipe_sleep(struct pipe *cpipe, const char *wmesg) +{ + int error; + + rw_assert_wrlock(&cpipe->pipe_lock); + + error = rwsleep_nsec(cpipe, &cpipe->pipe_lock, + PRIBIO | PCATCH | PNORELOCK, wmesg, INFSLP); + if (error) + return (error); + + return (pipelock(cpipe)); +} + void pipeselwakeup(struct pipe *cpipe) { @@ -368,23 +394,18 @@ pipe_read(struct file *fp, struct uio *uio, int fflags) break; /* - * Unlock the pipe buffer for our remaining processing. - * We will either break out with an error or we will - * sleep and relock to loop. - */ - pipeunlock(rpipe); - - /* - * Handle non-blocking mode operation or - * wait for more data. + * Handle non-blocking mode operation. */ if (fp->f_flag & FNONBLOCK) { error = EAGAIN; - } else { - rpipe->pipe_state |= PIPE_WANTR; - if ((error = tsleep(rpipe, PRIBIO|PCATCH, "piperd", 0)) == 0) - error = pipelock(rpipe); + break; } + + /* + * Wait for more data. + */ + rpipe->pipe_state |= PIPE_WANTR; + error = pipe_sleep(rpipe, "piperd"); if (error) goto unlocked_error; } @@ -573,13 +594,8 @@ pipe_write(struct file *fp, struct uio *uio, int fflags) */ pipeselwakeup(wpipe); - pipeunlock(wpipe); wpipe->pipe_state |= PIPE_WANTW; - error = tsleep(wpipe, (PRIBIO + 1)|PCATCH, - "pipewr", 0); - if (error) - goto unlocked_error; - error = pipelock(wpipe); + error = pipe_sleep(wpipe, "pipewr"); if (error) goto unlocked_error; |