diff options
author | Eric Biggers <ebiggers@google.com> | 2018-02-06 18:42:05 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2018-02-06 21:32:48 -0500 |
commit | c4fed5a91fadc8a277b1eda474317b501651dd3e (patch) | |
tree | 6194bd7d003a2527f675dee2ab3fbb03332a8e19 | |
parent | 96e99be40e4cff870a83233731121ec0f7f95075 (diff) |
pipe: simplify round_pipe_size()
round_pipe_size() calculates the number of pages the requested size
corresponds to, then rounds the page count up to the next power of 2.
However, it also rounds everything < PAGE_SIZE up to PAGE_SIZE.
Therefore, there's no need to actually translate the size into a page
count; we just need to round the size up to the next power of 2.
We do need to verify the size isn't greater than (1 << 31), since on
32-bit systems roundup_pow_of_two() would be undefined in that case. But
that can just be combined with the UINT_MAX check which we need anyway
now.
Finally, update pipe_set_size() to not redundantly check the return value
of round_pipe_size() for the "invalid size" case twice.
Link: http://lkml.kernel.org/r/20180111052902.14409-7-ebiggers3@gmail.com
Signed-off-by: Eric Biggers <ebiggers@google.com>
Acked-by: Kees Cook <keescook@chromium.org>
Acked-by: Joe Lawrence <joe.lawrence@redhat.com>
Cc: Alexander Viro <viro@zeniv.linux.org.uk>
Cc: "Luis R . Rodriguez" <mcgrof@kernel.org>
Cc: Michael Kerrisk <mtk.manpages@gmail.com>
Cc: Mikulas Patocka <mpatocka@redhat.com>
Cc: Willy Tarreau <w@1wt.eu>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r-- | fs/pipe.c | 14 |
1 files changed, 3 insertions, 11 deletions
@@ -1022,20 +1022,14 @@ const struct file_operations pipefifo_fops = { | |||
1022 | */ | 1022 | */ |
1023 | unsigned int round_pipe_size(unsigned long size) | 1023 | unsigned int round_pipe_size(unsigned long size) |
1024 | { | 1024 | { |
1025 | unsigned long nr_pages; | 1025 | if (size > (1U << 31)) |
1026 | |||
1027 | if (size > UINT_MAX) | ||
1028 | return 0; | 1026 | return 0; |
1029 | 1027 | ||
1030 | /* Minimum pipe size, as required by POSIX */ | 1028 | /* Minimum pipe size, as required by POSIX */ |
1031 | if (size < PAGE_SIZE) | 1029 | if (size < PAGE_SIZE) |
1032 | size = PAGE_SIZE; | 1030 | return PAGE_SIZE; |
1033 | |||
1034 | nr_pages = (size + PAGE_SIZE - 1) >> PAGE_SHIFT; | ||
1035 | if (nr_pages == 0) | ||
1036 | return 0; | ||
1037 | 1031 | ||
1038 | return roundup_pow_of_two(nr_pages) << PAGE_SHIFT; | 1032 | return roundup_pow_of_two(size); |
1039 | } | 1033 | } |
1040 | 1034 | ||
1041 | /* | 1035 | /* |
@@ -1050,8 +1044,6 @@ static long pipe_set_size(struct pipe_inode_info *pipe, unsigned long arg) | |||
1050 | long ret = 0; | 1044 | long ret = 0; |
1051 | 1045 | ||
1052 | size = round_pipe_size(arg); | 1046 | size = round_pipe_size(arg); |
1053 | if (size == 0) | ||
1054 | return -EINVAL; | ||
1055 | nr_pages = size >> PAGE_SHIFT; | 1047 | nr_pages = size >> PAGE_SHIFT; |
1056 | 1048 | ||
1057 | if (!nr_pages) | 1049 | if (!nr_pages) |