diff options
author | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2015-09-08 09:25:39 -0400 |
---|---|---|
committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2015-09-17 07:43:42 -0400 |
commit | 8d4bd0ed0439dfc780aab801a085961925ed6838 (patch) | |
tree | 7faf6dc07fc2752cdbc1d5edc9d8faf202d60459 | |
parent | 9380cf5a884e237fc0e2571d5adf0b43bb4412c8 (diff) |
s390/compat: correct uc_sigmask of the compat signal frame
The uc_sigmask in the ucontext structure is an array of words to keep
the 64 signal bits (or 1024 if you ask glibc but the kernel sigset_t
only has 64 bits).
For 64 bit the sigset_t contains a single 8 byte word, but for 31 bit
there are two 4 byte words. The compat signal handler code uses a
simple copy of the 64 bit sigset_t to the 31 bit compat_sigset_t.
As s390 is a big-endian architecture this is incorrect, the two words
in the 31 bit sigset_t array need to be swapped.
Cc: <stable@vger.kernel.org>
Reported-by: Stefan Liebler <stli@linux.vnet.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
-rw-r--r-- | arch/s390/kernel/compat_signal.c | 27 |
1 files changed, 23 insertions, 4 deletions
diff --git a/arch/s390/kernel/compat_signal.c b/arch/s390/kernel/compat_signal.c index eb4664238613..e0f9d270b30f 100644 --- a/arch/s390/kernel/compat_signal.c +++ b/arch/s390/kernel/compat_signal.c | |||
@@ -48,6 +48,19 @@ typedef struct | |||
48 | struct ucontext32 uc; | 48 | struct ucontext32 uc; |
49 | } rt_sigframe32; | 49 | } rt_sigframe32; |
50 | 50 | ||
51 | static inline void sigset_to_sigset32(unsigned long *set64, | ||
52 | compat_sigset_word *set32) | ||
53 | { | ||
54 | set32[0] = (compat_sigset_word) set64[0]; | ||
55 | set32[1] = (compat_sigset_word)(set64[0] >> 32); | ||
56 | } | ||
57 | |||
58 | static inline void sigset32_to_sigset(compat_sigset_word *set32, | ||
59 | unsigned long *set64) | ||
60 | { | ||
61 | set64[0] = (unsigned long) set32[0] | ((unsigned long) set32[1] << 32); | ||
62 | } | ||
63 | |||
51 | int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from) | 64 | int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from) |
52 | { | 65 | { |
53 | int err; | 66 | int err; |
@@ -281,10 +294,12 @@ COMPAT_SYSCALL_DEFINE0(sigreturn) | |||
281 | { | 294 | { |
282 | struct pt_regs *regs = task_pt_regs(current); | 295 | struct pt_regs *regs = task_pt_regs(current); |
283 | sigframe32 __user *frame = (sigframe32 __user *)regs->gprs[15]; | 296 | sigframe32 __user *frame = (sigframe32 __user *)regs->gprs[15]; |
297 | compat_sigset_t cset; | ||
284 | sigset_t set; | 298 | sigset_t set; |
285 | 299 | ||
286 | if (__copy_from_user(&set.sig, &frame->sc.oldmask, _SIGMASK_COPY_SIZE32)) | 300 | if (__copy_from_user(&cset.sig, &frame->sc.oldmask, _SIGMASK_COPY_SIZE32)) |
287 | goto badframe; | 301 | goto badframe; |
302 | sigset32_to_sigset(cset.sig, set.sig); | ||
288 | set_current_blocked(&set); | 303 | set_current_blocked(&set); |
289 | save_fpu_regs(); | 304 | save_fpu_regs(); |
290 | if (restore_sigregs32(regs, &frame->sregs)) | 305 | if (restore_sigregs32(regs, &frame->sregs)) |
@@ -302,10 +317,12 @@ COMPAT_SYSCALL_DEFINE0(rt_sigreturn) | |||
302 | { | 317 | { |
303 | struct pt_regs *regs = task_pt_regs(current); | 318 | struct pt_regs *regs = task_pt_regs(current); |
304 | rt_sigframe32 __user *frame = (rt_sigframe32 __user *)regs->gprs[15]; | 319 | rt_sigframe32 __user *frame = (rt_sigframe32 __user *)regs->gprs[15]; |
320 | compat_sigset_t cset; | ||
305 | sigset_t set; | 321 | sigset_t set; |
306 | 322 | ||
307 | if (__copy_from_user(&set, &frame->uc.uc_sigmask, sizeof(set))) | 323 | if (__copy_from_user(&cset, &frame->uc.uc_sigmask, sizeof(cset))) |
308 | goto badframe; | 324 | goto badframe; |
325 | sigset32_to_sigset(cset.sig, set.sig); | ||
309 | set_current_blocked(&set); | 326 | set_current_blocked(&set); |
310 | if (compat_restore_altstack(&frame->uc.uc_stack)) | 327 | if (compat_restore_altstack(&frame->uc.uc_stack)) |
311 | goto badframe; | 328 | goto badframe; |
@@ -377,7 +394,7 @@ static int setup_frame32(struct ksignal *ksig, sigset_t *set, | |||
377 | return -EFAULT; | 394 | return -EFAULT; |
378 | 395 | ||
379 | /* Create struct sigcontext32 on the signal stack */ | 396 | /* Create struct sigcontext32 on the signal stack */ |
380 | memcpy(&sc.oldmask, &set->sig, _SIGMASK_COPY_SIZE32); | 397 | sigset_to_sigset32(set->sig, sc.oldmask); |
381 | sc.sregs = (__u32)(unsigned long __force) &frame->sregs; | 398 | sc.sregs = (__u32)(unsigned long __force) &frame->sregs; |
382 | if (__copy_to_user(&frame->sc, &sc, sizeof(frame->sc))) | 399 | if (__copy_to_user(&frame->sc, &sc, sizeof(frame->sc))) |
383 | return -EFAULT; | 400 | return -EFAULT; |
@@ -438,6 +455,7 @@ static int setup_frame32(struct ksignal *ksig, sigset_t *set, | |||
438 | static int setup_rt_frame32(struct ksignal *ksig, sigset_t *set, | 455 | static int setup_rt_frame32(struct ksignal *ksig, sigset_t *set, |
439 | struct pt_regs *regs) | 456 | struct pt_regs *regs) |
440 | { | 457 | { |
458 | compat_sigset_t cset; | ||
441 | rt_sigframe32 __user *frame; | 459 | rt_sigframe32 __user *frame; |
442 | unsigned long restorer; | 460 | unsigned long restorer; |
443 | size_t frame_size; | 461 | size_t frame_size; |
@@ -485,11 +503,12 @@ static int setup_rt_frame32(struct ksignal *ksig, sigset_t *set, | |||
485 | store_sigregs(); | 503 | store_sigregs(); |
486 | 504 | ||
487 | /* Create ucontext on the signal stack. */ | 505 | /* Create ucontext on the signal stack. */ |
506 | sigset_to_sigset32(set->sig, cset.sig); | ||
488 | if (__put_user(uc_flags, &frame->uc.uc_flags) || | 507 | if (__put_user(uc_flags, &frame->uc.uc_flags) || |
489 | __put_user(0, &frame->uc.uc_link) || | 508 | __put_user(0, &frame->uc.uc_link) || |
490 | __compat_save_altstack(&frame->uc.uc_stack, regs->gprs[15]) || | 509 | __compat_save_altstack(&frame->uc.uc_stack, regs->gprs[15]) || |
491 | save_sigregs32(regs, &frame->uc.uc_mcontext) || | 510 | save_sigregs32(regs, &frame->uc.uc_mcontext) || |
492 | __copy_to_user(&frame->uc.uc_sigmask, set, sizeof(*set)) || | 511 | __copy_to_user(&frame->uc.uc_sigmask, &cset, sizeof(cset)) || |
493 | save_sigregs_ext32(regs, &frame->uc.uc_mcontext_ext)) | 512 | save_sigregs_ext32(regs, &frame->uc.uc_mcontext_ext)) |
494 | return -EFAULT; | 513 | return -EFAULT; |
495 | 514 | ||