aboutsummaryrefslogtreecommitdiffstats
path: root/fs/binfmt_misc.c
diff options
context:
space:
mode:
authorArnd Bergmann <arnd@arndb.de>2014-10-13 18:52:08 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2014-10-13 20:18:16 -0400
commitde8288b1f87146da06f6912028a1a9615a75c987 (patch)
tree1974c13d945203c4042e06133f7b7229e61d6598 /fs/binfmt_misc.c
parent43bd40e5b6eab989a2186b09d45b8ff8efd127b2 (diff)
binfmt_misc: work around gcc-4.9 warning
gcc-4.9 on ARM gives us a mysterious warning about the binfmt_misc parse_command function: fs/binfmt_misc.c: In function 'parse_command.part.3': fs/binfmt_misc.c:405:7: warning: array subscript is above array bounds [-Warray-bounds] I've managed to trace this back to the ARM implementation of memset, which is called from copy_from_user in case of a fault and which does #define memset(p,v,n) \ ({ \ void *__p = (p); size_t __n = n; \ if ((__n) != 0) { \ if (__builtin_constant_p((v)) && (v) == 0) \ __memzero((__p),(__n)); \ else \ memset((__p),(v),(__n)); \ } \ (__p); \ }) Apparently gcc gets confused by the check for "size != 0" and believes that the size might be zero when it gets to the line that does "if (s[count-1] == '\n')", so it would access data outside of the array. gcc is clearly wrong here, since this condition was already checked earlier in the function and the 'size' value can not change in the meantime. Fortunately, we can work around it and get rid of the warning by rearranging the function to check for zero size after doing the copy_from_user. It is still safe to pass a zero size into copy_from_user, so it does not cause any side effects. Signed-off-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'fs/binfmt_misc.c')
-rw-r--r--fs/binfmt_misc.c4
1 files changed, 2 insertions, 2 deletions
diff --git a/fs/binfmt_misc.c b/fs/binfmt_misc.c
index c96b855d2a5e..fd8beb9657a2 100644
--- a/fs/binfmt_misc.c
+++ b/fs/binfmt_misc.c
@@ -411,12 +411,12 @@ static int parse_command(const char __user *buffer, size_t count)
411{ 411{
412 char s[4]; 412 char s[4];
413 413
414 if (!count)
415 return 0;
416 if (count > 3) 414 if (count > 3)
417 return -EINVAL; 415 return -EINVAL;
418 if (copy_from_user(s, buffer, count)) 416 if (copy_from_user(s, buffer, count))
419 return -EFAULT; 417 return -EFAULT;
418 if (!count)
419 return 0;
420 if (s[count-1] == '\n') 420 if (s[count-1] == '\n')
421 count--; 421 count--;
422 if (count == 1 && s[0] == '0') 422 if (count == 1 && s[0] == '0')