m68knommu: fix signal handling return path
The return from software signal handling pushes code on the stack that system calls to the kernels cleanup code. This is borrowed directly from the m68k linux signal handler. The rt signal case is not quite right for the restricted instruction set of the ColdFire parts. And neither the normal signal case or rt signal case properly flushes/pushes the appropriate cache lines. Rework the return path to just call back through some code fragments in the kernel proper (with no MMU in the way we can do this). No cache problems, and less code overall. Original patch submitted by Wilson Callan <wcallan@savantav.com> Greg fixed the rt signal return path to use the proper system call Signed-off-by: Greg Ungerer <gerg@uclinux.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
10ed2195bb
commit
2d7f16d6d1
2 changed files with 13 additions and 17 deletions
|
@ -32,6 +32,7 @@
|
||||||
#include <asm/segment.h>
|
#include <asm/segment.h>
|
||||||
#include <asm/asm-offsets.h>
|
#include <asm/asm-offsets.h>
|
||||||
#include <asm/entry.h>
|
#include <asm/entry.h>
|
||||||
|
#include <asm/unistd.h>
|
||||||
|
|
||||||
.text
|
.text
|
||||||
|
|
||||||
|
@ -140,3 +141,11 @@ ENTRY(sys_rt_sigreturn)
|
||||||
RESTORE_SWITCH_STACK
|
RESTORE_SWITCH_STACK
|
||||||
rts
|
rts
|
||||||
|
|
||||||
|
ENTRY(ret_from_user_signal)
|
||||||
|
moveq #__NR_sigreturn,%d0
|
||||||
|
trap #0
|
||||||
|
|
||||||
|
ENTRY(ret_from_user_rt_signal)
|
||||||
|
move #__NR_rt_sigreturn,%d0
|
||||||
|
trap #0
|
||||||
|
|
||||||
|
|
|
@ -51,6 +51,8 @@
|
||||||
|
|
||||||
#define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
|
#define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
|
||||||
|
|
||||||
|
void ret_from_user_signal(void);
|
||||||
|
void ret_from_user_rt_signal(void);
|
||||||
asmlinkage int do_signal(sigset_t *oldset, struct pt_regs *regs);
|
asmlinkage int do_signal(sigset_t *oldset, struct pt_regs *regs);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -539,10 +541,6 @@ static inline int rt_setup_ucontext(struct ucontext *uc, struct pt_regs *regs)
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void push_cache (unsigned long vaddr)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void *
|
static inline void *
|
||||||
get_sigframe(struct k_sigaction *ka, struct pt_regs *regs, size_t frame_size)
|
get_sigframe(struct k_sigaction *ka, struct pt_regs *regs, size_t frame_size)
|
||||||
{
|
{
|
||||||
|
@ -586,16 +584,11 @@ static void setup_frame (int sig, struct k_sigaction *ka,
|
||||||
err |= copy_to_user (&frame->sc, &context, sizeof(context));
|
err |= copy_to_user (&frame->sc, &context, sizeof(context));
|
||||||
|
|
||||||
/* Set up to return from userspace. */
|
/* Set up to return from userspace. */
|
||||||
err |= __put_user(frame->retcode, &frame->pretcode);
|
err |= __put_user((void *) ret_from_user_signal, &frame->pretcode);
|
||||||
/* moveq #,d0; trap #0 */
|
|
||||||
err |= __put_user(0x70004e40 + (__NR_sigreturn << 16),
|
|
||||||
(long *)(frame->retcode));
|
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
goto give_sigsegv;
|
goto give_sigsegv;
|
||||||
|
|
||||||
push_cache ((unsigned long) &frame->retcode);
|
|
||||||
|
|
||||||
/* Set up registers for signal handler */
|
/* Set up registers for signal handler */
|
||||||
wrusp ((unsigned long) frame);
|
wrusp ((unsigned long) frame);
|
||||||
regs->pc = (unsigned long) ka->sa.sa_handler;
|
regs->pc = (unsigned long) ka->sa.sa_handler;
|
||||||
|
@ -655,17 +648,11 @@ static void setup_rt_frame (int sig, struct k_sigaction *ka, siginfo_t *info,
|
||||||
err |= copy_to_user (&frame->uc.uc_sigmask, set, sizeof(*set));
|
err |= copy_to_user (&frame->uc.uc_sigmask, set, sizeof(*set));
|
||||||
|
|
||||||
/* Set up to return from userspace. */
|
/* Set up to return from userspace. */
|
||||||
err |= __put_user(frame->retcode, &frame->pretcode);
|
err |= __put_user((void *) ret_from_user_rt_signal, &frame->pretcode);
|
||||||
/* moveq #,d0; notb d0; trap #0 */
|
|
||||||
err |= __put_user(0x70004600 + ((__NR_rt_sigreturn ^ 0xff) << 16),
|
|
||||||
(long *)(frame->retcode + 0));
|
|
||||||
err |= __put_user(0x4e40, (short *)(frame->retcode + 4));
|
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
goto give_sigsegv;
|
goto give_sigsegv;
|
||||||
|
|
||||||
push_cache ((unsigned long) &frame->retcode);
|
|
||||||
|
|
||||||
/* Set up registers for signal handler */
|
/* Set up registers for signal handler */
|
||||||
wrusp ((unsigned long) frame);
|
wrusp ((unsigned long) frame);
|
||||||
regs->pc = (unsigned long) ka->sa.sa_handler;
|
regs->pc = (unsigned long) ka->sa.sa_handler;
|
||||||
|
|
Loading…
Reference in a new issue