diff options
author | Chris Metcalf <cmetcalf@ezchip.com> | 2015-06-25 15:02:08 -0700 |
---|---|---|
committer | Sasha Levin <sasha.levin@oracle.com> | 2015-08-04 14:36:14 -0400 |
commit | ea82d2674523e6b51055e251c94b93114c99045a (patch) | |
tree | e5abb59bdece38aeb581a3c57de9a5f47e13dbb2 | |
parent | f078fd1dae84e8695fdd717554addd0e13d2dc35 (diff) | |
download | lwn-ea82d2674523e6b51055e251c94b93114c99045a.tar.gz lwn-ea82d2674523e6b51055e251c94b93114c99045a.zip |
__bitmap_parselist: fix bug in empty string handling
[ Upstream commit 2528a8b8f457d7432552d0e2b6f0f4046bb702f4 ]
bitmap_parselist("", &mask, nmaskbits) will erroneously set bit zero in
the mask. The same bug is visible in cpumask_parselist() since it is
layered on top of the bitmask code, e.g. if you boot with "isolcpus=",
you will actually end up with cpu zero isolated.
The bug was introduced in commit 4b060420a596 ("bitmap, irq: add
smp_affinity_list interface to /proc/irq") when bitmap_parselist() was
generalized to support userspace as well as kernelspace.
Fixes: 4b060420a596 ("bitmap, irq: add smp_affinity_list interface to /proc/irq")
Signed-off-by: Chris Metcalf <cmetcalf@ezchip.com>
Cc: Rasmus Villemoes <linux@rasmusvillemoes.dk>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Sasha Levin <sasha.levin@oracle.com>
-rw-r--r-- | lib/bitmap.c | 17 |
1 files changed, 9 insertions, 8 deletions
diff --git a/lib/bitmap.c b/lib/bitmap.c index b499ab6ada29..2ed91904e806 100644 --- a/lib/bitmap.c +++ b/lib/bitmap.c @@ -610,12 +610,12 @@ static int __bitmap_parselist(const char *buf, unsigned int buflen, unsigned a, b; int c, old_c, totaldigits; const char __user __force *ubuf = (const char __user __force *)buf; - int exp_digit, in_range; + int at_start, in_range; totaldigits = c = 0; bitmap_zero(maskp, nmaskbits); do { - exp_digit = 1; + at_start = 1; in_range = 0; a = b = 0; @@ -644,11 +644,10 @@ static int __bitmap_parselist(const char *buf, unsigned int buflen, break; if (c == '-') { - if (exp_digit || in_range) + if (at_start || in_range) return -EINVAL; b = 0; in_range = 1; - exp_digit = 1; continue; } @@ -658,16 +657,18 @@ static int __bitmap_parselist(const char *buf, unsigned int buflen, b = b * 10 + (c - '0'); if (!in_range) a = b; - exp_digit = 0; + at_start = 0; totaldigits++; } if (!(a <= b)) return -EINVAL; if (b >= nmaskbits) return -ERANGE; - while (a <= b) { - set_bit(a, maskp); - a++; + if (!at_start) { + while (a <= b) { + set_bit(a, maskp); + a++; + } } } while (buflen && c == ','); return 0; |