diff options
-rw-r--r-- | kernel/compat.c | 25 |
1 files changed, 11 insertions, 14 deletions
diff --git a/kernel/compat.c b/kernel/compat.c index 7f40e9275fd9..5adab05a3172 100644 --- a/kernel/compat.c +++ b/kernel/compat.c | |||
@@ -495,29 +495,26 @@ asmlinkage long compat_sys_sched_getaffinity(compat_pid_t pid, unsigned int len, | |||
495 | { | 495 | { |
496 | int ret; | 496 | int ret; |
497 | cpumask_var_t mask; | 497 | cpumask_var_t mask; |
498 | unsigned long *k; | ||
499 | unsigned int min_length = cpumask_size(); | ||
500 | |||
501 | if (nr_cpu_ids <= BITS_PER_COMPAT_LONG) | ||
502 | min_length = sizeof(compat_ulong_t); | ||
503 | 498 | ||
504 | if (len < min_length) | 499 | if ((len * BITS_PER_BYTE) < nr_cpu_ids) |
500 | return -EINVAL; | ||
501 | if (len & (sizeof(compat_ulong_t)-1)) | ||
505 | return -EINVAL; | 502 | return -EINVAL; |
506 | 503 | ||
507 | if (!alloc_cpumask_var(&mask, GFP_KERNEL)) | 504 | if (!alloc_cpumask_var(&mask, GFP_KERNEL)) |
508 | return -ENOMEM; | 505 | return -ENOMEM; |
509 | 506 | ||
510 | ret = sched_getaffinity(pid, mask); | 507 | ret = sched_getaffinity(pid, mask); |
511 | if (ret < 0) | 508 | if (ret == 0) { |
512 | goto out; | 509 | size_t retlen = min_t(size_t, len, cpumask_size()); |
513 | 510 | ||
514 | k = cpumask_bits(mask); | 511 | if (compat_put_bitmap(user_mask_ptr, cpumask_bits(mask), retlen * 8)) |
515 | ret = compat_put_bitmap(user_mask_ptr, k, min_length * 8); | 512 | ret = -EFAULT; |
516 | if (ret == 0) | 513 | else |
517 | ret = min_length; | 514 | ret = retlen; |
518 | 515 | } | |
519 | out: | ||
520 | free_cpumask_var(mask); | 516 | free_cpumask_var(mask); |
517 | |||
521 | return ret; | 518 | return ret; |
522 | } | 519 | } |
523 | 520 | ||