aboutsummaryrefslogtreecommitdiffstats
path: root/arch/mips/kernel/signal32.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/mips/kernel/signal32.c')
-rw-r--r--arch/mips/kernel/signal32.c27
1 files changed, 24 insertions, 3 deletions
diff --git a/arch/mips/kernel/signal32.c b/arch/mips/kernel/signal32.c
index 19bbef001959..20013b6fe725 100644
--- a/arch/mips/kernel/signal32.c
+++ b/arch/mips/kernel/signal32.c
@@ -220,6 +220,18 @@ static int setup_sigcontext32(struct pt_regs *regs,
220 return err; 220 return err;
221} 221}
222 222
223static int
224check_and_restore_fp_context32(struct sigcontext32 __user *sc)
225{
226 int err, sig;
227
228 err = sig = fpcsr_pending(&sc->sc_fpc_csr);
229 if (err > 0)
230 err = 0;
231 err |= restore_fp_context32(sc);
232 return err ?: sig;
233}
234
223static int restore_sigcontext32(struct pt_regs *regs, 235static int restore_sigcontext32(struct pt_regs *regs,
224 struct sigcontext32 __user *sc) 236 struct sigcontext32 __user *sc)
225{ 237{
@@ -255,7 +267,8 @@ static int restore_sigcontext32(struct pt_regs *regs,
255 if (used_math()) { 267 if (used_math()) {
256 /* restore fpu context if we have used it before */ 268 /* restore fpu context if we have used it before */
257 own_fpu(); 269 own_fpu();
258 err |= restore_fp_context32(sc); 270 if (!err)
271 err = check_and_restore_fp_context32(sc);
259 } else { 272 } else {
260 /* signal handler may have used FPU. Give it up. */ 273 /* signal handler may have used FPU. Give it up. */
261 lose_fpu(); 274 lose_fpu();
@@ -508,6 +521,7 @@ asmlinkage void sys32_sigreturn(nabi_no_regargs struct pt_regs regs)
508{ 521{
509 struct sigframe32 __user *frame; 522 struct sigframe32 __user *frame;
510 sigset_t blocked; 523 sigset_t blocked;
524 int sig;
511 525
512 frame = (struct sigframe32 __user *) regs.regs[29]; 526 frame = (struct sigframe32 __user *) regs.regs[29];
513 if (!access_ok(VERIFY_READ, frame, sizeof(*frame))) 527 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
@@ -521,8 +535,11 @@ asmlinkage void sys32_sigreturn(nabi_no_regargs struct pt_regs regs)
521 recalc_sigpending(); 535 recalc_sigpending();
522 spin_unlock_irq(&current->sighand->siglock); 536 spin_unlock_irq(&current->sighand->siglock);
523 537
524 if (restore_sigcontext32(&regs, &frame->sf_sc)) 538 sig = restore_sigcontext32(&regs, &frame->sf_sc);
539 if (sig < 0)
525 goto badframe; 540 goto badframe;
541 else if (sig)
542 force_sig(sig, current);
526 543
527 /* 544 /*
528 * Don't let your children do this ... 545 * Don't let your children do this ...
@@ -545,6 +562,7 @@ asmlinkage void sys32_rt_sigreturn(nabi_no_regargs struct pt_regs regs)
545 sigset_t set; 562 sigset_t set;
546 stack_t st; 563 stack_t st;
547 s32 sp; 564 s32 sp;
565 int sig;
548 566
549 frame = (struct rt_sigframe32 __user *) regs.regs[29]; 567 frame = (struct rt_sigframe32 __user *) regs.regs[29];
550 if (!access_ok(VERIFY_READ, frame, sizeof(*frame))) 568 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
@@ -558,8 +576,11 @@ asmlinkage void sys32_rt_sigreturn(nabi_no_regargs struct pt_regs regs)
558 recalc_sigpending(); 576 recalc_sigpending();
559 spin_unlock_irq(&current->sighand->siglock); 577 spin_unlock_irq(&current->sighand->siglock);
560 578
561 if (restore_sigcontext32(&regs, &frame->rs_uc.uc_mcontext)) 579 sig = restore_sigcontext32(&regs, &frame->rs_uc.uc_mcontext);
580 if (sig < 0)
562 goto badframe; 581 goto badframe;
582 else if (sig)
583 force_sig(sig, current);
563 584
564 /* The ucontext contains a stack32_t, so we must convert! */ 585 /* The ucontext contains a stack32_t, so we must convert! */
565 if (__get_user(sp, &frame->rs_uc.uc_stack.ss_sp)) 586 if (__get_user(sp, &frame->rs_uc.uc_stack.ss_sp))