Commit 83f7d932 authored by Jeff Dike's avatar Jeff Dike Committed by Linus Torvalds

[PATCH] uml: handle signal api

This first patch lays some groundwork needed by the others.

From: Bodo Stroesser - Change the interface to handle_signal so that it
doesn't take the system call return value as an argument and eliminate
its return value.

kern_do_signal also now doesn't return immediately after determining that
there is no signal to deliver.
Signed-off-by: default avatarJeff Dike <jdike@addtoit.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 63849d27
...@@ -38,16 +38,17 @@ EXPORT_SYMBOL(unblock_signals); ...@@ -38,16 +38,17 @@ EXPORT_SYMBOL(unblock_signals);
/* /*
* OK, we're invoking a handler * OK, we're invoking a handler
*/ */
static int handle_signal(struct pt_regs *regs, unsigned long signr, static void handle_signal(struct pt_regs *regs, unsigned long signr,
struct k_sigaction *ka, siginfo_t *info, struct k_sigaction *ka, siginfo_t *info,
sigset_t *oldset, int error) sigset_t *oldset)
{ {
__sighandler_t handler; __sighandler_t handler;
void (*restorer)(void); void (*restorer)(void);
unsigned long sp; unsigned long sp;
sigset_t save; sigset_t save;
int err, ret; int error, err, ret;
error = PT_REGS_SYSCALL_RET(&current->thread.regs);
ret = 0; ret = 0;
/* Always make any pending restarted system calls return -EINTR */ /* Always make any pending restarted system calls return -EINTR */
current_thread_info()->restart_block.fn = do_no_restart_syscall; current_thread_info()->restart_block.fn = do_no_restart_syscall;
...@@ -109,31 +110,25 @@ static int handle_signal(struct pt_regs *regs, unsigned long signr, ...@@ -109,31 +110,25 @@ static int handle_signal(struct pt_regs *regs, unsigned long signr,
else else
err = setup_signal_stack_sc(sp, signr, (unsigned long) handler, err = setup_signal_stack_sc(sp, signr, (unsigned long) handler,
restorer, regs, &save); restorer, regs, &save);
if(err) goto segv; if(err)
force_sigsegv(signr, current);
return(0);
segv:
force_sigsegv(signr, current);
return(1);
} }
static int kern_do_signal(struct pt_regs *regs, sigset_t *oldset, int error) static int kern_do_signal(struct pt_regs *regs, sigset_t *oldset)
{ {
struct k_sigaction ka_copy; struct k_sigaction ka_copy;
siginfo_t info; siginfo_t info;
int err, sig; int sig;
if (!oldset) if (!oldset)
oldset = &current->blocked; oldset = &current->blocked;
sig = get_signal_to_deliver(&info, &ka_copy, regs, NULL); sig = get_signal_to_deliver(&info, &ka_copy, regs, NULL);
if(sig == 0) if(sig > 0){
return(0); /* Whee! Actually deliver the signal. */
handle_signal(regs, sig, &ka_copy, &info, oldset);
/* Whee! Actually deliver the signal. */
err = handle_signal(regs, sig, &ka_copy, &info, oldset, error);
if(!err)
return(1); return(1);
}
/* Did we come from a system call? */ /* Did we come from a system call? */
if(PT_REGS_SYSCALL_NR(regs) >= 0){ if(PT_REGS_SYSCALL_NR(regs) >= 0){
...@@ -165,7 +160,7 @@ static int kern_do_signal(struct pt_regs *regs, sigset_t *oldset, int error) ...@@ -165,7 +160,7 @@ static int kern_do_signal(struct pt_regs *regs, sigset_t *oldset, int error)
int do_signal(int error) int do_signal(int error)
{ {
return(kern_do_signal(&current->thread.regs, NULL, error)); return(kern_do_signal(&current->thread.regs, NULL));
} }
/* /*
...@@ -182,10 +177,11 @@ int sys_sigsuspend(int history0, int history1, old_sigset_t mask) ...@@ -182,10 +177,11 @@ int sys_sigsuspend(int history0, int history1, old_sigset_t mask)
recalc_sigpending(); recalc_sigpending();
spin_unlock_irq(&current->sighand->siglock); spin_unlock_irq(&current->sighand->siglock);
PT_REGS_SYSCALL_RET(&current->thread.regs) = -EINTR;
while (1) { while (1) {
current->state = TASK_INTERRUPTIBLE; current->state = TASK_INTERRUPTIBLE;
schedule(); schedule();
if(kern_do_signal(&current->thread.regs, &saveset, -EINTR)) if(kern_do_signal(&current->thread.regs, &saveset))
return(-EINTR); return(-EINTR);
} }
} }
...@@ -208,10 +204,11 @@ int sys_rt_sigsuspend(sigset_t __user *unewset, size_t sigsetsize) ...@@ -208,10 +204,11 @@ int sys_rt_sigsuspend(sigset_t __user *unewset, size_t sigsetsize)
recalc_sigpending(); recalc_sigpending();
spin_unlock_irq(&current->sighand->siglock); spin_unlock_irq(&current->sighand->siglock);
PT_REGS_SYSCALL_RET(&current->thread.regs) = -EINTR;
while (1) { while (1) {
current->state = TASK_INTERRUPTIBLE; current->state = TASK_INTERRUPTIBLE;
schedule(); schedule();
if (kern_do_signal(&current->thread.regs, &saveset, -EINTR)) if (kern_do_signal(&current->thread.regs, &saveset))
return(-EINTR); return(-EINTR);
} }
} }
......
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