aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael Kerrisk (man-pages) <mtk.manpages@gmail.com>2016-10-11 16:53:37 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2016-10-11 18:06:32 -0400
commita005ca0e6813e1d796a7422a7e31d8b8d6555df1 (patch)
tree72119a865d6f3df796555eec04558a1ef99c8140
parent09b4d1990094dd22c27fb0163534db419458569c (diff)
pipe: fix limit checking in alloc_pipe_info()
The limit checking in alloc_pipe_info() (used by pipe(2) and when opening a FIFO) has the following problems: (1) When checking capacity required for the new pipe, the checks against the limit in /proc/sys/fs/pipe-user-pages-{soft,hard} are made against existing consumption, and exclude the memory required for the new pipe capacity. As a consequence: (1) the memory allocation throttling provided by the soft limit does not kick in quite as early as it should, and (2) the user can overrun the hard limit. (2) As currently implemented, accounting and checking against the limits is done as follows: (a) Test whether the user has exceeded the limit. (b) Make new pipe buffer allocation. (c) Account new allocation against the limits. This is racey. Multiple processes may pass point (a) simultaneously, and then allocate pipe buffers that are accounted for only in step (c). The race means that the user's pipe buffer allocation could be pushed over the limit (by an arbitrary amount, depending on how unlucky we were in the race). [Thanks to Vegard Nossum for spotting this point, which I had missed.] This patch addresses the above problems as follows: * Alter the checks against limits to include the memory required for the new pipe. * Re-order the accounting step so that it precedes the buffer allocation. If the accounting step determines that a limit has been reached, revert the accounting and cause the operation to fail. Link: http://lkml.kernel.org/r/8ff3e9f9-23f6-510c-644f-8e70cd1c0bd9@gmail.com Signed-off-by: Michael Kerrisk <mtk.manpages@gmail.com> Reviewed-by: Vegard Nossum <vegard.nossum@oracle.com> Cc: Willy Tarreau <w@1wt.eu> Cc: <socketpair@gmail.com> Cc: Tetsuo Handa <penguin-kernel@I-love.SAKURA.ne.jp> Cc: Jens Axboe <axboe@fb.com> Cc: Al Viro <viro@zeniv.linux.org.uk> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--fs/pipe.c20
1 files changed, 13 insertions, 7 deletions
diff --git a/fs/pipe.c b/fs/pipe.c
index eaa3c8d36291..83a292464de7 100644
--- a/fs/pipe.c
+++ b/fs/pipe.c
@@ -629,24 +629,30 @@ struct pipe_inode_info *alloc_pipe_info(void)
629 if (pipe == NULL) 629 if (pipe == NULL)
630 goto out_free_uid; 630 goto out_free_uid;
631 631
632 if (!too_many_pipe_buffers_hard(user)) { 632 account_pipe_buffers(user, 0, pipe_bufs);
633 if (too_many_pipe_buffers_soft(user)) 633
634 pipe_bufs = 1; 634 if (too_many_pipe_buffers_soft(user)) {
635 pipe->bufs = kcalloc(pipe_bufs, 635 account_pipe_buffers(user, pipe_bufs, 1);
636 sizeof(struct pipe_buffer), 636 pipe_bufs = 1;
637 GFP_KERNEL_ACCOUNT);
638 } 637 }
639 638
639 if (too_many_pipe_buffers_hard(user))
640 goto out_revert_acct;
641
642 pipe->bufs = kcalloc(pipe_bufs, sizeof(struct pipe_buffer),
643 GFP_KERNEL_ACCOUNT);
644
640 if (pipe->bufs) { 645 if (pipe->bufs) {
641 init_waitqueue_head(&pipe->wait); 646 init_waitqueue_head(&pipe->wait);
642 pipe->r_counter = pipe->w_counter = 1; 647 pipe->r_counter = pipe->w_counter = 1;
643 pipe->buffers = pipe_bufs; 648 pipe->buffers = pipe_bufs;
644 pipe->user = user; 649 pipe->user = user;
645 account_pipe_buffers(user, 0, pipe_bufs);
646 mutex_init(&pipe->mutex); 650 mutex_init(&pipe->mutex);
647 return pipe; 651 return pipe;
648 } 652 }
649 653
654out_revert_acct:
655 account_pipe_buffers(user, pipe_bufs, 0);
650 kfree(pipe); 656 kfree(pipe);
651out_free_uid: 657out_free_uid:
652 free_uid(user); 658 free_uid(user);