Commit fbe98686 authored by Al Viro's avatar Al Viro Committed by Richard Weinberger

um: no need to play with save_sp in signal frame setup anymore

Signed-off-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
Signed-off-by: default avatarRichard Weinberger <richard@nod.at>
parent c7ea591c
...@@ -215,8 +215,7 @@ static int copy_sc_from_user(struct pt_regs *regs, ...@@ -215,8 +215,7 @@ static int copy_sc_from_user(struct pt_regs *regs,
} }
static int copy_sc_to_user(struct sigcontext __user *to, static int copy_sc_to_user(struct sigcontext __user *to,
struct _fpstate __user *to_fp, struct pt_regs *regs, struct _fpstate __user *to_fp, struct pt_regs *regs)
unsigned long sp)
{ {
struct sigcontext sc; struct sigcontext sc;
struct faultinfo * fi = &current->thread.arch.faultinfo; struct faultinfo * fi = &current->thread.arch.faultinfo;
...@@ -230,7 +229,7 @@ static int copy_sc_to_user(struct sigcontext __user *to, ...@@ -230,7 +229,7 @@ static int copy_sc_to_user(struct sigcontext __user *to,
sc.di = REGS_EDI(regs->regs.gp); sc.di = REGS_EDI(regs->regs.gp);
sc.si = REGS_ESI(regs->regs.gp); sc.si = REGS_ESI(regs->regs.gp);
sc.bp = REGS_EBP(regs->regs.gp); sc.bp = REGS_EBP(regs->regs.gp);
sc.sp = sp; sc.sp = REGS_SP(regs->regs.gp);
sc.bx = REGS_EBX(regs->regs.gp); sc.bx = REGS_EBX(regs->regs.gp);
sc.dx = REGS_EDX(regs->regs.gp); sc.dx = REGS_EDX(regs->regs.gp);
sc.cx = REGS_ECX(regs->regs.gp); sc.cx = REGS_ECX(regs->regs.gp);
...@@ -291,7 +290,7 @@ static int copy_ucontext_to_user(struct ucontext __user *uc, ...@@ -291,7 +290,7 @@ static int copy_ucontext_to_user(struct ucontext __user *uc,
err |= put_user(current->sas_ss_sp, &uc->uc_stack.ss_sp); err |= put_user(current->sas_ss_sp, &uc->uc_stack.ss_sp);
err |= put_user(sas_ss_flags(sp), &uc->uc_stack.ss_flags); err |= put_user(sas_ss_flags(sp), &uc->uc_stack.ss_flags);
err |= put_user(current->sas_ss_size, &uc->uc_stack.ss_size); err |= put_user(current->sas_ss_size, &uc->uc_stack.ss_size);
err |= copy_sc_to_user(&uc->uc_mcontext, fp, &current->thread.regs, sp); err |= copy_sc_to_user(&uc->uc_mcontext, fp, &current->thread.regs);
err |= copy_to_user(&uc->uc_sigmask, set, sizeof(*set)); err |= copy_to_user(&uc->uc_sigmask, set, sizeof(*set));
return err; return err;
} }
...@@ -324,7 +323,6 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig, ...@@ -324,7 +323,6 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig,
{ {
struct sigframe __user *frame; struct sigframe __user *frame;
void __user *restorer; void __user *restorer;
unsigned long save_sp = PT_REGS_SP(regs);
int err = 0; int err = 0;
/* This is the same calculation as i386 - ((sp + 4) & 15) == 0 */ /* This is the same calculation as i386 - ((sp + 4) & 15) == 0 */
...@@ -337,19 +335,9 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig, ...@@ -337,19 +335,9 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig,
if (ka->sa.sa_flags & SA_RESTORER) if (ka->sa.sa_flags & SA_RESTORER)
restorer = ka->sa.sa_restorer; restorer = ka->sa.sa_restorer;
/* Update SP now because the page fault handler refuses to extend
* the stack if the faulting address is too far below the current
* SP, which frame now certainly is. If there's an error, the original
* value is restored on the way out.
* When writing the sigcontext to the stack, we have to write the
* original value, so that's passed to copy_sc_to_user, which does
* the right thing with it.
*/
PT_REGS_SP(regs) = (unsigned long) frame;
err |= __put_user(restorer, &frame->pretcode); err |= __put_user(restorer, &frame->pretcode);
err |= __put_user(sig, &frame->sig); err |= __put_user(sig, &frame->sig);
err |= copy_sc_to_user(&frame->sc, NULL, regs, save_sp); err |= copy_sc_to_user(&frame->sc, NULL, regs);
err |= __put_user(mask->sig[0], &frame->sc.oldmask); err |= __put_user(mask->sig[0], &frame->sc.oldmask);
if (_NSIG_WORDS > 1) if (_NSIG_WORDS > 1)
err |= __copy_to_user(&frame->extramask, &mask->sig[1], err |= __copy_to_user(&frame->extramask, &mask->sig[1],
...@@ -367,7 +355,7 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig, ...@@ -367,7 +355,7 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig,
err |= __put_user(0x80cd, (short __user *)(frame->retcode+6)); err |= __put_user(0x80cd, (short __user *)(frame->retcode+6));
if (err) if (err)
goto err; return err;
PT_REGS_SP(regs) = (unsigned long) frame; PT_REGS_SP(regs) = (unsigned long) frame;
PT_REGS_IP(regs) = (unsigned long) ka->sa.sa_handler; PT_REGS_IP(regs) = (unsigned long) ka->sa.sa_handler;
...@@ -378,10 +366,6 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig, ...@@ -378,10 +366,6 @@ int setup_signal_stack_sc(unsigned long stack_top, int sig,
if ((current->ptrace & PT_DTRACE) && (current->ptrace & PT_PTRACED)) if ((current->ptrace & PT_DTRACE) && (current->ptrace & PT_PTRACED))
ptrace_notify(SIGTRAP); ptrace_notify(SIGTRAP);
return 0; return 0;
err:
PT_REGS_SP(regs) = save_sp;
return err;
} }
int setup_signal_stack_si(unsigned long stack_top, int sig, int setup_signal_stack_si(unsigned long stack_top, int sig,
...@@ -390,7 +374,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -390,7 +374,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
{ {
struct rt_sigframe __user *frame; struct rt_sigframe __user *frame;
void __user *restorer; void __user *restorer;
unsigned long save_sp = PT_REGS_SP(regs);
int err = 0; int err = 0;
stack_top &= -8UL; stack_top &= -8UL;
...@@ -402,16 +385,13 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -402,16 +385,13 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
if (ka->sa.sa_flags & SA_RESTORER) if (ka->sa.sa_flags & SA_RESTORER)
restorer = ka->sa.sa_restorer; restorer = ka->sa.sa_restorer;
/* See comment above about why this is here */
PT_REGS_SP(regs) = (unsigned long) frame;
err |= __put_user(restorer, &frame->pretcode); err |= __put_user(restorer, &frame->pretcode);
err |= __put_user(sig, &frame->sig); err |= __put_user(sig, &frame->sig);
err |= __put_user(&frame->info, &frame->pinfo); err |= __put_user(&frame->info, &frame->pinfo);
err |= __put_user(&frame->uc, &frame->puc); err |= __put_user(&frame->uc, &frame->puc);
err |= copy_siginfo_to_user(&frame->info, info); err |= copy_siginfo_to_user(&frame->info, info);
err |= copy_ucontext_to_user(&frame->uc, &frame->fpstate, mask, err |= copy_ucontext_to_user(&frame->uc, &frame->fpstate, mask,
save_sp); PT_REGS_SP(regs));
/* /*
* This is movl $,%eax ; int $0x80 * This is movl $,%eax ; int $0x80
...@@ -425,8 +405,9 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -425,8 +405,9 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
err |= __put_user(0x80cd, (short __user *)(frame->retcode+5)); err |= __put_user(0x80cd, (short __user *)(frame->retcode+5));
if (err) if (err)
goto err; return err;
PT_REGS_SP(regs) = (unsigned long) frame;
PT_REGS_IP(regs) = (unsigned long) ka->sa.sa_handler; PT_REGS_IP(regs) = (unsigned long) ka->sa.sa_handler;
PT_REGS_EAX(regs) = (unsigned long) sig; PT_REGS_EAX(regs) = (unsigned long) sig;
PT_REGS_EDX(regs) = (unsigned long) &frame->info; PT_REGS_EDX(regs) = (unsigned long) &frame->info;
...@@ -435,10 +416,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -435,10 +416,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
if ((current->ptrace & PT_DTRACE) && (current->ptrace & PT_PTRACED)) if ((current->ptrace & PT_DTRACE) && (current->ptrace & PT_PTRACED))
ptrace_notify(SIGTRAP); ptrace_notify(SIGTRAP);
return 0; return 0;
err:
PT_REGS_SP(regs) = save_sp;
return err;
} }
long sys_sigreturn(struct pt_regs regs) long sys_sigreturn(struct pt_regs regs)
......
...@@ -68,7 +68,7 @@ static int copy_sc_from_user(struct pt_regs *regs, ...@@ -68,7 +68,7 @@ static int copy_sc_from_user(struct pt_regs *regs,
static int copy_sc_to_user(struct sigcontext __user *to, static int copy_sc_to_user(struct sigcontext __user *to,
struct _fpstate __user *to_fp, struct pt_regs *regs, struct _fpstate __user *to_fp, struct pt_regs *regs,
unsigned long mask, unsigned long sp) unsigned long mask)
{ {
struct faultinfo * fi = &current->thread.arch.faultinfo; struct faultinfo * fi = &current->thread.arch.faultinfo;
struct sigcontext sc; struct sigcontext sc;
...@@ -81,11 +81,7 @@ static int copy_sc_to_user(struct sigcontext __user *to, ...@@ -81,11 +81,7 @@ static int copy_sc_to_user(struct sigcontext __user *to,
PUTREG(DI, di); PUTREG(DI, di);
PUTREG(SI, si); PUTREG(SI, si);
PUTREG(BP, bp); PUTREG(BP, bp);
/* PUTREG(SP, sp);
* Must use original RSP, which is passed in, rather than what's in
* signal frame.
*/
sc.sp = sp;
PUTREG(BX, bx); PUTREG(BX, bx);
PUTREG(DX, dx); PUTREG(DX, dx);
PUTREG(CX, cx); PUTREG(CX, cx);
...@@ -141,7 +137,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -141,7 +137,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
siginfo_t *info, sigset_t *set) siginfo_t *info, sigset_t *set)
{ {
struct rt_sigframe __user *frame; struct rt_sigframe __user *frame;
unsigned long save_sp = PT_REGS_RSP(regs);
int err = 0; int err = 0;
struct task_struct *me = current; struct task_struct *me = current;
...@@ -159,26 +154,15 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -159,26 +154,15 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
goto out; goto out;
} }
/*
* Update SP now because the page fault handler refuses to extend
* the stack if the faulting address is too far below the current
* SP, which frame now certainly is. If there's an error, the original
* value is restored on the way out.
* When writing the sigcontext to the stack, we have to write the
* original value, so that's passed to copy_sc_to_user, which does
* the right thing with it.
*/
PT_REGS_RSP(regs) = (unsigned long) frame;
/* Create the ucontext. */ /* Create the ucontext. */
err |= __put_user(0, &frame->uc.uc_flags); err |= __put_user(0, &frame->uc.uc_flags);
err |= __put_user(0, &frame->uc.uc_link); err |= __put_user(0, &frame->uc.uc_link);
err |= __put_user(me->sas_ss_sp, &frame->uc.uc_stack.ss_sp); err |= __put_user(me->sas_ss_sp, &frame->uc.uc_stack.ss_sp);
err |= __put_user(sas_ss_flags(save_sp), err |= __put_user(sas_ss_flags(PT_REGS_RSP(regs)),
&frame->uc.uc_stack.ss_flags); &frame->uc.uc_stack.ss_flags);
err |= __put_user(me->sas_ss_size, &frame->uc.uc_stack.ss_size); err |= __put_user(me->sas_ss_size, &frame->uc.uc_stack.ss_size);
err |= copy_sc_to_user(&frame->uc.uc_mcontext, &frame->fpstate, regs, err |= copy_sc_to_user(&frame->uc.uc_mcontext, &frame->fpstate, regs,
set->sig[0], save_sp); set->sig[0]);
err |= __put_user(&frame->fpstate, &frame->uc.uc_mcontext.fpstate); err |= __put_user(&frame->fpstate, &frame->uc.uc_mcontext.fpstate);
if (sizeof(*set) == 16) { if (sizeof(*set) == 16) {
__put_user(set->sig[0], &frame->uc.uc_sigmask.sig[0]); __put_user(set->sig[0], &frame->uc.uc_sigmask.sig[0]);
...@@ -197,10 +181,10 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -197,10 +181,10 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
err |= __put_user(ka->sa.sa_restorer, &frame->pretcode); err |= __put_user(ka->sa.sa_restorer, &frame->pretcode);
else else
/* could use a vstub here */ /* could use a vstub here */
goto restore_sp; return err;
if (err) if (err)
goto restore_sp; return err;
/* Set up registers for signal handler */ /* Set up registers for signal handler */
{ {
...@@ -209,6 +193,7 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -209,6 +193,7 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
sig = ed->signal_invmap[sig]; sig = ed->signal_invmap[sig];
} }
PT_REGS_RSP(regs) = (unsigned long) frame;
PT_REGS_RDI(regs) = sig; PT_REGS_RDI(regs) = sig;
/* In case the signal handler was declared without prototypes */ /* In case the signal handler was declared without prototypes */
PT_REGS_RAX(regs) = 0; PT_REGS_RAX(regs) = 0;
...@@ -222,10 +207,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig, ...@@ -222,10 +207,6 @@ int setup_signal_stack_si(unsigned long stack_top, int sig,
PT_REGS_RIP(regs) = (unsigned long) ka->sa.sa_handler; PT_REGS_RIP(regs) = (unsigned long) ka->sa.sa_handler;
out: out:
return err; return err;
restore_sp:
PT_REGS_RSP(regs) = save_sp;
return err;
} }
long sys_rt_sigreturn(struct pt_regs *regs) long sys_rt_sigreturn(struct pt_regs *regs)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment