Commit cece79ae authored by Roland McGrath's avatar Roland McGrath Committed by Linus Torvalds

[PATCH] cleanup ptrace stops and remove notify_parent

This adds a new state TASK_TRACED that is used in place of TASK_STOPPED
when a thread stops because it is ptraced.  Now ptrace operations are only
permitted when the target is in TASK_TRACED state, not in TASK_STOPPED. 
This means that if a process is stopped normally by a job control signal
and then you PTRACE_ATTACH to it, you will have to send it a SIGCONT before
you can do any ptrace operations on it.  (The SIGCONT will be reported to
ptrace and then you can discard it instead of passing it through when you
call PTRACE_CONT et al.)

If a traced child gets orphaned while in TASK_TRACED state, it morphs into
TASK_STOPPED state.  This makes it again possible to resume or destroy the
process with SIGCONT or SIGKILL.

All non-signal tracing stops should now be done via ptrace_notify.  I've
updated the syscall tracing code in several architectures to do this
instead of replicating the work by hand.  I also fixed several that were
unnecessarily repeating some of the checks in ptrace_check_attach.  Calling
ptrace_check_attach alone is sufficient, and the old checks repeated before
are now incorrect, not just superfluous.

I've closed a race in ptrace_check_attach.  With this, we should have a
robust guarantee that when ptrace starts operating, the task will be in
TASK_TRACED state and won't come out of it.  This is because the only way
to resume from TASK_TRACED is via ptrace operations, and only the one
parent thread attached as the tracer can do those.

This patch also cleans up the do_notify_parent and do_notify_parent_cldstop
code so that the dead and stopped cases are completely disjoint.  The
notify_parent function is gone.
Signed-off-by: default avatarRoland McGrath <roland@redhat.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent cfec5c01
...@@ -792,11 +792,8 @@ asmlinkage void syscall_trace(int why, struct pt_regs *regs) ...@@ -792,11 +792,8 @@ asmlinkage void syscall_trace(int why, struct pt_regs *regs)
/* the 0x80 provides a way for the tracing parent to distinguish /* the 0x80 provides a way for the tracing parent to distinguish
between a syscall stop and SIGTRAP delivery */ between a syscall stop and SIGTRAP delivery */
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -729,11 +729,8 @@ asmlinkage void syscall_trace(int why, struct pt_regs *regs) ...@@ -729,11 +729,8 @@ asmlinkage void syscall_trace(int why, struct pt_regs *regs)
/* the 0x80 provides a way for the tracing parent to distinguish /* the 0x80 provides a way for the tracing parent to distinguish
between a syscall stop and SIGTRAP delivery */ between a syscall stop and SIGTRAP delivery */
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -85,17 +85,8 @@ sys_ptrace(long request, long pid, long addr, long data) ...@@ -85,17 +85,8 @@ sys_ptrace(long request, long pid, long addr, long data)
goto out_tsk; goto out_tsk;
} }
ret = -ESRCH; ret = ptrace_check_attach(child, request == PTRACE_KILL);
if (ret < 0)
if (!(child->ptrace & PT_PTRACED))
goto out_tsk;
if (child->state != TASK_STOPPED) {
if (request != PTRACE_KILL)
goto out_tsk;
}
if (child->parent != current)
goto out_tsk; goto out_tsk;
switch (request) { switch (request) {
......
...@@ -89,13 +89,6 @@ asmlinkage int sys_ptrace(long request, long pid, long addr, long data) ...@@ -89,13 +89,6 @@ asmlinkage int sys_ptrace(long request, long pid, long addr, long data)
ret = ptrace_attach(child); ret = ptrace_attach(child);
goto out_tsk; goto out_tsk;
} }
ret = -ESRCH;
if (!(child->ptrace & PT_PTRACED))
goto out_tsk;
if (child->state != TASK_STOPPED) {
if (request != PTRACE_KILL)
goto out_tsk;
}
ret = ptrace_check_attach(child, request == PTRACE_KILL); ret = ptrace_check_attach(child, request == PTRACE_KILL);
if (ret < 0) if (ret < 0)
goto out_tsk; goto out_tsk;
...@@ -270,10 +263,8 @@ asmlinkage void syscall_trace(void) ...@@ -270,10 +263,8 @@ asmlinkage void syscall_trace(void)
return; return;
if (!(current->ptrace & PT_PTRACED)) if (!(current->ptrace & PT_PTRACED))
return; return;
current->exit_code = SIGTRAP; ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
current->state = TASK_STOPPED; ? 0x80 : 0));
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -379,11 +379,8 @@ asmlinkage void syscall_trace(void) ...@@ -379,11 +379,8 @@ asmlinkage void syscall_trace(void)
if (!current->thread.work.delayed_trace && if (!current->thread.work.delayed_trace &&
!current->thread.work.syscall_trace) !current->thread.work.syscall_trace)
return; return;
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -133,13 +133,6 @@ asmlinkage int sys_ptrace(long request, long pid, long addr, long data) ...@@ -133,13 +133,6 @@ asmlinkage int sys_ptrace(long request, long pid, long addr, long data)
ret = ptrace_attach(child); ret = ptrace_attach(child);
goto out_tsk; goto out_tsk;
} }
ret = -ESRCH;
if (!(child->ptrace & PT_PTRACED))
goto out_tsk;
if (child->state != TASK_STOPPED) {
if (request != PTRACE_KILL)
goto out_tsk;
}
ret = ptrace_check_attach(child, request == PTRACE_KILL); ret = ptrace_check_attach(child, request == PTRACE_KILL);
if (ret < 0) if (ret < 0)
goto out_tsk; goto out_tsk;
...@@ -376,10 +369,8 @@ asmlinkage void syscall_trace(void) ...@@ -376,10 +369,8 @@ asmlinkage void syscall_trace(void)
return; return;
if (!(current->ptrace & PT_PTRACED)) if (!(current->ptrace & PT_PTRACED))
return; return;
current->exit_code = SIGTRAP; ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
current->state = TASK_STOPPED; ? 0x80 : 0));
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -404,11 +404,8 @@ void syscall_trace(void) ...@@ -404,11 +404,8 @@ void syscall_trace(void)
return; return;
if (!(current->ptrace & PT_PTRACED)) if (!(current->ptrace & PT_PTRACED))
return; return;
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -311,11 +311,8 @@ asmlinkage void syscall_trace(void) ...@@ -311,11 +311,8 @@ asmlinkage void syscall_trace(void)
if (!(tsk->ptrace & PT_PTRACED)) if (!(tsk->ptrace & PT_PTRACED))
return; return;
tsk->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
tsk->state = TASK_STOPPED;
notify_parent(tsk, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -614,12 +614,9 @@ asmlinkage void syscall_trace(void) ...@@ -614,12 +614,9 @@ asmlinkage void syscall_trace(void)
return; return;
if (!(current->ptrace & PT_PTRACED)) if (!(current->ptrace & PT_PTRACED))
return; return;
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0);
current->state = TASK_STOPPED;
current->thread.flags ^= MAGIC_CONSTANT; current->thread.flags ^= MAGIC_CONSTANT;
notify_parent(current, SIGCHLD); ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
schedule(); ? 0x80 : 0));
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -627,11 +627,8 @@ asmlinkage void syscall_trace(void) ...@@ -627,11 +627,8 @@ asmlinkage void syscall_trace(void)
return; return;
if (!(current->ptrace & PT_PTRACED)) if (!(current->ptrace & PT_PTRACED))
return; return;
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
......
...@@ -147,14 +147,8 @@ int sys_ptrace(long request, long pid, long addr, long data) ...@@ -147,14 +147,8 @@ int sys_ptrace(long request, long pid, long addr, long data)
rval = ptrace_attach(child); rval = ptrace_attach(child);
goto out_tsk; goto out_tsk;
} }
rval = -ESRCH; ret = ptrace_check_attach(child, request == PTRACE_KILL);
if (!(child->ptrace & PT_PTRACED)) if (ret < 0)
goto out_tsk;
if (child->state != TASK_STOPPED) {
if (request != PTRACE_KILL)
goto out_tsk;
}
if (child->parent != current)
goto out_tsk; goto out_tsk;
switch (request) { switch (request) {
...@@ -269,11 +263,8 @@ asmlinkage void syscall_trace(void) ...@@ -269,11 +263,8 @@ asmlinkage void syscall_trace(void)
return; return;
/* The 0x80 provides a way for the tracing parent to distinguish /* The 0x80 provides a way for the tracing parent to distinguish
between a syscall stop and SIGTRAP delivery */ between a syscall stop and SIGTRAP delivery */
current->exit_code = SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD) ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
? 0x80 : 0); ? 0x80 : 0));
current->state = TASK_STOPPED;
notify_parent(current, SIGCHLD);
schedule();
/* /*
* this isn't the same as continuing with a signal, but it will do * this isn't the same as continuing with a signal, but it will do
* for normal use. strace only continues with a signal if the * for normal use. strace only continues with a signal if the
......
...@@ -130,8 +130,9 @@ static const char *task_state_array[] = { ...@@ -130,8 +130,9 @@ static const char *task_state_array[] = {
"S (sleeping)", /* 1 */ "S (sleeping)", /* 1 */
"D (disk sleep)", /* 2 */ "D (disk sleep)", /* 2 */
"T (stopped)", /* 4 */ "T (stopped)", /* 4 */
"Z (zombie)", /* 8 */ "T (tracing stop)", /* 8 */
"X (dead)" /* 16 */ "Z (zombie)", /* 16 */
"X (dead)" /* 32 */
}; };
static inline const char * get_task_state(struct task_struct *tsk) static inline const char * get_task_state(struct task_struct *tsk)
...@@ -141,7 +142,8 @@ static inline const char * get_task_state(struct task_struct *tsk) ...@@ -141,7 +142,8 @@ static inline const char * get_task_state(struct task_struct *tsk)
TASK_UNINTERRUPTIBLE | TASK_UNINTERRUPTIBLE |
TASK_ZOMBIE | TASK_ZOMBIE |
TASK_DEAD | TASK_DEAD |
TASK_STOPPED); TASK_STOPPED |
TASK_TRACED);
const char **p = &task_state_array[0]; const char **p = &task_state_array[0];
while (state) { while (state) {
......
...@@ -287,7 +287,8 @@ static int proc_root_link(struct inode *inode, struct dentry **dentry, struct vf ...@@ -287,7 +287,8 @@ static int proc_root_link(struct inode *inode, struct dentry **dentry, struct vf
#define MAY_PTRACE(task) \ #define MAY_PTRACE(task) \
(task == current || \ (task == current || \
(task->parent == current && \ (task->parent == current && \
(task->ptrace & PT_PTRACED) && task->state == TASK_STOPPED && \ (task->ptrace & PT_PTRACED) && \
(task->state == TASK_STOPPED || task->state == TASK_TRACED) && \
security_ptrace(current,task) == 0)) security_ptrace(current,task) == 0))
static int may_ptrace_attach(struct task_struct *task) static int may_ptrace_attach(struct task_struct *task)
......
...@@ -106,8 +106,9 @@ extern unsigned long nr_iowait(void); ...@@ -106,8 +106,9 @@ extern unsigned long nr_iowait(void);
#define TASK_INTERRUPTIBLE 1 #define TASK_INTERRUPTIBLE 1
#define TASK_UNINTERRUPTIBLE 2 #define TASK_UNINTERRUPTIBLE 2
#define TASK_STOPPED 4 #define TASK_STOPPED 4
#define TASK_ZOMBIE 8 #define TASK_TRACED 8
#define TASK_DEAD 16 #define TASK_ZOMBIE 16
#define TASK_DEAD 32
#define __set_task_state(tsk, state_value) \ #define __set_task_state(tsk, state_value) \
do { (tsk)->state = (state_value); } while (0) do { (tsk)->state = (state_value); } while (0)
...@@ -738,7 +739,6 @@ extern int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp); ...@@ -738,7 +739,6 @@ extern int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp);
extern int kill_pg_info(int, struct siginfo *, pid_t); extern int kill_pg_info(int, struct siginfo *, pid_t);
extern int kill_sl_info(int, struct siginfo *, pid_t); extern int kill_sl_info(int, struct siginfo *, pid_t);
extern int kill_proc_info(int, struct siginfo *, pid_t); extern int kill_proc_info(int, struct siginfo *, pid_t);
extern void notify_parent(struct task_struct *, int);
extern void do_notify_parent(struct task_struct *, int); extern void do_notify_parent(struct task_struct *, int);
extern void force_sig(int, struct task_struct *); extern void force_sig(int, struct task_struct *);
extern void force_sig_specific(int, struct task_struct *); extern void force_sig_specific(int, struct task_struct *);
......
...@@ -555,6 +555,14 @@ static inline void reparent_thread(task_t *p, task_t *father, int traced) ...@@ -555,6 +555,14 @@ static inline void reparent_thread(task_t *p, task_t *father, int traced)
if (p->state == TASK_ZOMBIE && p->exit_signal != -1 && if (p->state == TASK_ZOMBIE && p->exit_signal != -1 &&
thread_group_empty(p)) thread_group_empty(p))
do_notify_parent(p, p->exit_signal); do_notify_parent(p, p->exit_signal);
else if (p->state == TASK_TRACED) {
/*
* If it was at a trace stop, turn it into
* a normal stop since it's no longer being
* traced.
*/
p->state = TASK_STOPPED;
}
} }
/* /*
...@@ -1164,7 +1172,7 @@ static int wait_task_stopped(task_t *p, int delayed_group_leader, int noreap, ...@@ -1164,7 +1172,7 @@ static int wait_task_stopped(task_t *p, int delayed_group_leader, int noreap,
* race with the TASK_ZOMBIE case. * race with the TASK_ZOMBIE case.
*/ */
exit_code = xchg(&p->exit_code, 0); exit_code = xchg(&p->exit_code, 0);
if (unlikely(p->state > TASK_STOPPED)) { if (unlikely(p->state >= TASK_ZOMBIE)) {
/* /*
* The task resumed and then died. Let the next iteration * The task resumed and then died. Let the next iteration
* catch it in TASK_ZOMBIE. Note that exit_code might * catch it in TASK_ZOMBIE. Note that exit_code might
...@@ -1245,6 +1253,10 @@ static long do_wait(pid_t pid, int options, struct siginfo __user *infop, ...@@ -1245,6 +1253,10 @@ static long do_wait(pid_t pid, int options, struct siginfo __user *infop,
flag = 1; flag = 1;
switch (p->state) { switch (p->state) {
case TASK_TRACED:
if (!(p->ptrace & PT_PTRACED))
continue;
/*FALLTHROUGH*/
case TASK_STOPPED: case TASK_STOPPED:
if (!(options & WUNTRACED) && if (!(options & WUNTRACED) &&
!(p->ptrace & PT_PTRACED)) !(p->ptrace & PT_PTRACED))
......
...@@ -25,7 +25,8 @@ static inline int freezeable(struct task_struct * p) ...@@ -25,7 +25,8 @@ static inline int freezeable(struct task_struct * p)
(p->flags & PF_NOFREEZE) || (p->flags & PF_NOFREEZE) ||
(p->state == TASK_ZOMBIE) || (p->state == TASK_ZOMBIE) ||
(p->state == TASK_DEAD) || (p->state == TASK_DEAD) ||
(p->state == TASK_STOPPED)) (p->state == TASK_STOPPED) ||
(p->state == TASK_TRACED))
return 0; return 0;
return 1; return 1;
} }
...@@ -70,6 +71,7 @@ int freeze_processes(void) ...@@ -70,6 +71,7 @@ int freeze_processes(void)
if (!freezeable(p)) if (!freezeable(p))
continue; continue;
if ((p->flags & PF_FROZEN) || if ((p->flags & PF_FROZEN) ||
(p->state == TASK_TRACED) ||
(p->state == TASK_STOPPED)) (p->state == TASK_STOPPED))
continue; continue;
......
...@@ -55,6 +55,15 @@ void __ptrace_unlink(task_t *child) ...@@ -55,6 +55,15 @@ void __ptrace_unlink(task_t *child)
REMOVE_LINKS(child); REMOVE_LINKS(child);
child->parent = child->real_parent; child->parent = child->real_parent;
SET_LINKS(child); SET_LINKS(child);
if (child->state == TASK_TRACED) {
/*
* Turn a tracing stop into a normal stop now,
* since with no tracer there would be no way
* to wake it up with SIGCONT or SIGKILL.
*/
child->state = TASK_STOPPED;
}
} }
/* /*
...@@ -62,20 +71,28 @@ void __ptrace_unlink(task_t *child) ...@@ -62,20 +71,28 @@ void __ptrace_unlink(task_t *child)
*/ */
int ptrace_check_attach(struct task_struct *child, int kill) int ptrace_check_attach(struct task_struct *child, int kill)
{ {
if (!(child->ptrace & PT_PTRACED)) int ret = -ESRCH;
return -ESRCH;
if (child->parent != current) /*
return -ESRCH; * We take the read lock around doing both checks to close a
* possible race where someone else was tracing our child and
* detached between these two checks. After this locked check,
* we are sure that this is our traced child and that can only
* be changed by us so it's not changing right after this.
*/
read_lock(&tasklist_lock);
if ((child->ptrace & PT_PTRACED) && child->parent == current)
ret = 0;
read_unlock(&tasklist_lock);
if (!kill) { if (!ret && !kill) {
if (child->state != TASK_STOPPED) if (child->state != TASK_TRACED)
return -ESRCH; return -ESRCH;
wait_task_inactive(child); wait_task_inactive(child);
} }
/* All systems go.. */ /* All systems go.. */
return 0; return ret;
} }
int ptrace_attach(struct task_struct *task) int ptrace_attach(struct task_struct *task)
...@@ -281,15 +298,13 @@ static int ptrace_setoptions(struct task_struct *child, long data) ...@@ -281,15 +298,13 @@ static int ptrace_setoptions(struct task_struct *child, long data)
static int ptrace_getsiginfo(struct task_struct *child, siginfo_t __user * data) static int ptrace_getsiginfo(struct task_struct *child, siginfo_t __user * data)
{ {
if (child->last_siginfo == NULL) BUG_ON(child->last_siginfo == NULL);
return -EINVAL;
return copy_siginfo_to_user(data, child->last_siginfo); return copy_siginfo_to_user(data, child->last_siginfo);
} }
static int ptrace_setsiginfo(struct task_struct *child, siginfo_t __user * data) static int ptrace_setsiginfo(struct task_struct *child, siginfo_t __user * data)
{ {
if (child->last_siginfo == NULL) BUG_ON(child->last_siginfo == NULL);
return -EINVAL;
if (copy_from_user(child->last_siginfo, data, sizeof (siginfo_t)) != 0) if (copy_from_user(child->last_siginfo, data, sizeof (siginfo_t)) != 0)
return -EFAULT; return -EFAULT;
return 0; return 0;
...@@ -322,24 +337,3 @@ int ptrace_request(struct task_struct *child, long request, ...@@ -322,24 +337,3 @@ int ptrace_request(struct task_struct *child, long request,
return ret; return ret;
} }
void ptrace_notify(int exit_code)
{
BUG_ON (!(current->ptrace & PT_PTRACED));
/* Let the debugger run. */
current->exit_code = exit_code;
set_current_state(TASK_STOPPED);
notify_parent(current, SIGCHLD);
schedule();
/*
* Signals sent while we were stopped might set TIF_SIGPENDING.
*/
spin_lock_irq(&current->sighand->siglock);
recalc_sigpending();
spin_unlock_irq(&current->sighand->siglock);
}
EXPORT_SYMBOL(ptrace_notify);
...@@ -1258,7 +1258,7 @@ static int try_to_wake_up(task_t * p, unsigned int state, int sync) ...@@ -1258,7 +1258,7 @@ static int try_to_wake_up(task_t * p, unsigned int state, int sync)
int fastcall wake_up_process(task_t * p) int fastcall wake_up_process(task_t * p)
{ {
return try_to_wake_up(p, TASK_STOPPED | return try_to_wake_up(p, TASK_STOPPED | TASK_TRACED |
TASK_INTERRUPTIBLE | TASK_UNINTERRUPTIBLE, 0); TASK_INTERRUPTIBLE | TASK_UNINTERRUPTIBLE, 0);
} }
...@@ -3679,7 +3679,7 @@ static void show_task(task_t * p) ...@@ -3679,7 +3679,7 @@ static void show_task(task_t * p)
task_t *relative; task_t *relative;
unsigned state; unsigned state;
unsigned long free = 0; unsigned long free = 0;
static const char *stat_nam[] = { "R", "S", "D", "T", "Z", "W" }; static const char *stat_nam[] = { "R", "S", "D", "T", "t", "Z", "X" };
printk("%-13.13s ", p->comm); printk("%-13.13s ", p->comm);
state = p->state ? __ffs(p->state) + 1 : 0; state = p->state ? __ffs(p->state) + 1 : 0;
......
...@@ -636,7 +636,8 @@ static int check_kill_permission(int sig, struct siginfo *info, ...@@ -636,7 +636,8 @@ static int check_kill_permission(int sig, struct siginfo *info,
/* forward decl */ /* forward decl */
static void do_notify_parent_cldstop(struct task_struct *tsk, static void do_notify_parent_cldstop(struct task_struct *tsk,
struct task_struct *parent); struct task_struct *parent,
int why);
/* /*
* Handle magic process-wide effects of stop/continue signals. * Handle magic process-wide effects of stop/continue signals.
...@@ -681,11 +682,13 @@ static void handle_stop_signal(int sig, struct task_struct *p) ...@@ -681,11 +682,13 @@ static void handle_stop_signal(int sig, struct task_struct *p)
p->signal->stop_state = 1; p->signal->stop_state = 1;
spin_unlock(&p->sighand->siglock); spin_unlock(&p->sighand->siglock);
if (p->ptrace & PT_PTRACED) if (p->ptrace & PT_PTRACED)
do_notify_parent_cldstop(p, p->parent); do_notify_parent_cldstop(p, p->parent,
CLD_STOPPED);
else else
do_notify_parent_cldstop( do_notify_parent_cldstop(
p->group_leader, p->group_leader,
p->group_leader->real_parent); p->group_leader->real_parent,
CLD_STOPPED);
spin_lock(&p->sighand->siglock); spin_lock(&p->sighand->siglock);
} }
rm_from_queue(SIG_KERNEL_STOP_MASK, &p->signal->shared_pending); rm_from_queue(SIG_KERNEL_STOP_MASK, &p->signal->shared_pending);
...@@ -727,11 +730,13 @@ static void handle_stop_signal(int sig, struct task_struct *p) ...@@ -727,11 +730,13 @@ static void handle_stop_signal(int sig, struct task_struct *p)
p->signal->group_exit_code = 0; p->signal->group_exit_code = 0;
spin_unlock(&p->sighand->siglock); spin_unlock(&p->sighand->siglock);
if (p->ptrace & PT_PTRACED) if (p->ptrace & PT_PTRACED)
do_notify_parent_cldstop(p, p->parent); do_notify_parent_cldstop(p, p->parent,
CLD_CONTINUED);
else else
do_notify_parent_cldstop( do_notify_parent_cldstop(
p->group_leader, p->group_leader,
p->group_leader->real_parent); p->group_leader->real_parent,
CLD_CONTINUED);
spin_lock(&p->sighand->siglock); spin_lock(&p->sighand->siglock);
} }
} }
...@@ -899,10 +904,19 @@ force_sig_specific(int sig, struct task_struct *t) ...@@ -899,10 +904,19 @@ force_sig_specific(int sig, struct task_struct *t)
static void static void
__group_complete_signal(int sig, struct task_struct *p, unsigned int mask) __group_complete_signal(int sig, struct task_struct *p)
{ {
unsigned int mask;
struct task_struct *t; struct task_struct *t;
/*
* Don't bother zombies and stopped tasks (but
* SIGKILL will punch through stopped state)
*/
mask = TASK_DEAD | TASK_ZOMBIE | TASK_TRACED;
if (sig != SIGKILL)
mask |= TASK_STOPPED;
/* /*
* Now find a thread we can wake up to take the signal off the queue. * Now find a thread we can wake up to take the signal off the queue.
* *
...@@ -1004,7 +1018,6 @@ __group_complete_signal(int sig, struct task_struct *p, unsigned int mask) ...@@ -1004,7 +1018,6 @@ __group_complete_signal(int sig, struct task_struct *p, unsigned int mask)
static int static int
__group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p)
{ {
unsigned int mask;
int ret = 0; int ret = 0;
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
...@@ -1027,14 +1040,6 @@ __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) ...@@ -1027,14 +1040,6 @@ __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p)
/* This is a non-RT signal and we already have one queued. */ /* This is a non-RT signal and we already have one queued. */
return ret; return ret;
/*
* Don't bother zombies and stopped tasks (but
* SIGKILL will punch through stopped state)
*/
mask = TASK_DEAD | TASK_ZOMBIE;
if (sig != SIGKILL)
mask |= TASK_STOPPED;
/* /*
* Put this signal on the shared-pending queue, or fail with EAGAIN. * Put this signal on the shared-pending queue, or fail with EAGAIN.
* We always use the shared queue for process-wide signals, * We always use the shared queue for process-wide signals,
...@@ -1044,7 +1049,7 @@ __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) ...@@ -1044,7 +1049,7 @@ __group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p)
if (unlikely(ret)) if (unlikely(ret))
return ret; return ret;
__group_complete_signal(sig, p, mask); __group_complete_signal(sig, p);
return 0; return 0;
} }
...@@ -1401,7 +1406,6 @@ int ...@@ -1401,7 +1406,6 @@ int
send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p) send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p)
{ {
unsigned long flags; unsigned long flags;
unsigned int mask;
int ret = 0; int ret = 0;
BUG_ON(!(q->flags & SIGQUEUE_PREALLOC)); BUG_ON(!(q->flags & SIGQUEUE_PREALLOC));
...@@ -1426,13 +1430,6 @@ send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p) ...@@ -1426,13 +1430,6 @@ send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p)
q->info.si_overrun++; q->info.si_overrun++;
goto out; goto out;
} }
/*
* Don't bother zombies and stopped tasks (but
* SIGKILL will punch through stopped state)
*/
mask = TASK_DEAD | TASK_ZOMBIE;
if (sig != SIGKILL)
mask |= TASK_STOPPED;
/* /*
* Put this signal on the shared-pending queue. * Put this signal on the shared-pending queue.
...@@ -1443,7 +1440,7 @@ send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p) ...@@ -1443,7 +1440,7 @@ send_group_sigqueue(int sig, struct sigqueue *q, struct task_struct *p)
list_add_tail(&q->list, &p->signal->shared_pending.list); list_add_tail(&q->list, &p->signal->shared_pending.list);
sigaddset(&p->signal->shared_pending.signal, sig); sigaddset(&p->signal->shared_pending.signal, sig);
__group_complete_signal(sig, p, mask); __group_complete_signal(sig, p);
out: out:
spin_unlock_irqrestore(&p->sighand->siglock, flags); spin_unlock_irqrestore(&p->sighand->siglock, flags);
read_unlock(&tasklist_lock); read_unlock(&tasklist_lock);
...@@ -1476,19 +1473,22 @@ static void __wake_up_parent(struct task_struct *p, ...@@ -1476,19 +1473,22 @@ static void __wake_up_parent(struct task_struct *p,
} }
/* /*
* Let a parent know about a status change of a child. * Let a parent know about the death of a child.
* For a stopped/continued status change, use do_notify_parent_cldstop instead.
*/ */
void do_notify_parent(struct task_struct *tsk, int sig) void do_notify_parent(struct task_struct *tsk, int sig)
{ {
struct siginfo info; struct siginfo info;
unsigned long flags; unsigned long flags;
int why, status;
struct sighand_struct *psig; struct sighand_struct *psig;
if (sig == -1) if (sig == -1)
BUG(); BUG();
/* do_notify_parent_cldstop should have been called instead. */
BUG_ON(tsk->state & (TASK_STOPPED|TASK_TRACED));
BUG_ON(!tsk->ptrace && BUG_ON(!tsk->ptrace &&
(tsk->group_leader != tsk || !thread_group_empty(tsk))); (tsk->group_leader != tsk || !thread_group_empty(tsk)));
...@@ -1502,34 +1502,19 @@ void do_notify_parent(struct task_struct *tsk, int sig) ...@@ -1502,34 +1502,19 @@ void do_notify_parent(struct task_struct *tsk, int sig)
info.si_stime = tsk->stime + tsk->signal->stime; info.si_stime = tsk->stime + tsk->signal->stime;
k_getrusage(tsk, RUSAGE_BOTH, &info.si_rusage); k_getrusage(tsk, RUSAGE_BOTH, &info.si_rusage);
status = tsk->exit_code & 0x7f; info.si_status = tsk->exit_code & 0x7f;
why = SI_KERNEL; /* shouldn't happen */ if (tsk->exit_code & 0x80)
switch (tsk->state) { info.si_code = CLD_DUMPED;
case TASK_STOPPED: else if (tsk->exit_code & 0x7f)
/* FIXME -- can we deduce CLD_TRAPPED or CLD_CONTINUED? */ info.si_code = CLD_KILLED;
if (tsk->ptrace & PT_PTRACED) else {
why = CLD_TRAPPED; info.si_code = CLD_EXITED;
else info.si_status = tsk->exit_code >> 8;
why = CLD_STOPPED;
break;
default:
if (tsk->exit_code & 0x80)
why = CLD_DUMPED;
else if (tsk->exit_code & 0x7f)
why = CLD_KILLED;
else {
why = CLD_EXITED;
status = tsk->exit_code >> 8;
}
break;
} }
info.si_code = why;
info.si_status = status;
psig = tsk->parent->sighand; psig = tsk->parent->sighand;
spin_lock_irqsave(&psig->siglock, flags); spin_lock_irqsave(&psig->siglock, flags);
if (sig == SIGCHLD && tsk->state != TASK_STOPPED && if (sig == SIGCHLD &&
(psig->action[SIGCHLD-1].sa.sa_handler == SIG_IGN || (psig->action[SIGCHLD-1].sa.sa_handler == SIG_IGN ||
(psig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDWAIT))) { (psig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDWAIT))) {
/* /*
...@@ -1557,26 +1542,9 @@ void do_notify_parent(struct task_struct *tsk, int sig) ...@@ -1557,26 +1542,9 @@ void do_notify_parent(struct task_struct *tsk, int sig)
spin_unlock_irqrestore(&psig->siglock, flags); spin_unlock_irqrestore(&psig->siglock, flags);
} }
/*
* We need the tasklist lock because it's the only
* thing that protects out "parent" pointer.
*
* exit.c calls "do_notify_parent()" directly, because
* it already has the tasklist lock.
*/
void
notify_parent(struct task_struct *tsk, int sig)
{
if (sig != -1) {
read_lock(&tasklist_lock);
do_notify_parent(tsk, sig);
read_unlock(&tasklist_lock);
}
}
static void static void
do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent) do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent,
int why)
{ {
struct siginfo info; struct siginfo info;
unsigned long flags; unsigned long flags;
...@@ -1592,14 +1560,20 @@ do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent) ...@@ -1592,14 +1560,20 @@ do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent)
info.si_stime = tsk->stime; info.si_stime = tsk->stime;
k_getrusage(tsk, RUSAGE_BOTH, &info.si_rusage); k_getrusage(tsk, RUSAGE_BOTH, &info.si_rusage);
info.si_status = (tsk->signal ? tsk->signal->group_exit_code : info.si_code = why;
tsk->exit_code) & 0x7f; switch (why) {
if (info.si_status == 0) { case CLD_CONTINUED:
info.si_status = SIGCONT; info.si_status = SIGCONT;
info.si_code = CLD_CONTINUED; break;
} else { case CLD_STOPPED:
info.si_code = CLD_STOPPED; info.si_status = tsk->signal->group_exit_code & 0x7f;
} break;
case CLD_TRAPPED:
info.si_status = tsk->exit_code & 0x7f;
break;
default:
BUG();
}
sighand = parent->sighand; sighand = parent->sighand;
spin_lock_irqsave(&sighand->siglock, flags); spin_lock_irqsave(&sighand->siglock, flags);
...@@ -1613,6 +1587,68 @@ do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent) ...@@ -1613,6 +1587,68 @@ do_notify_parent_cldstop(struct task_struct *tsk, struct task_struct *parent)
spin_unlock_irqrestore(&sighand->siglock, flags); spin_unlock_irqrestore(&sighand->siglock, flags);
} }
/*
* This must be called with current->sighand->siglock held.
*
* This should be the path for all ptrace stops.
* We always set current->last_siginfo while stopped here.
* That makes it a way to test a stopped process for
* being ptrace-stopped vs being job-control-stopped.
*/
static void ptrace_stop(int exit_code, siginfo_t *info)
{
BUG_ON(!(current->ptrace & PT_PTRACED));
/*
* If there is a group stop in progress,
* we must participate in the bookkeeping.
*/
if (current->signal->group_stop_count > 0)
--current->signal->group_stop_count;
current->last_siginfo = info;
current->exit_code = exit_code;
/* Let the debugger run. */
set_current_state(TASK_TRACED);
spin_unlock_irq(&current->sighand->siglock);
read_lock(&tasklist_lock);
do_notify_parent_cldstop(current, current->parent, CLD_TRAPPED);
read_unlock(&tasklist_lock);
schedule();
/*
* We are back. Now reacquire the siglock before touching
* last_siginfo, so that we are sure to have synchronized with
* any signal-sending on another CPU that wants to examine it.
*/
spin_lock_irq(&current->sighand->siglock);
current->last_siginfo = NULL;
/*
* Queued signals ignored us while we were stopped for tracing.
* So check for any that we should take before resuming user mode.
*/
recalc_sigpending();
}
void ptrace_notify(int exit_code)
{
siginfo_t info;
BUG_ON((exit_code & (0x7f | ~0xffff)) != SIGTRAP);
memset(&info, 0, sizeof info);
info.si_signo = SIGTRAP;
info.si_code = exit_code;
info.si_pid = current->pid;
info.si_uid = current->uid;
/* Let the debugger run. */
spin_lock_irq(&current->sighand->siglock);
ptrace_stop(exit_code, &info);
spin_unlock_irq(&current->sighand->siglock);
}
#ifndef HAVE_ARCH_GET_SIGNAL_TO_DELIVER #ifndef HAVE_ARCH_GET_SIGNAL_TO_DELIVER
...@@ -1626,13 +1662,15 @@ finish_stop(int stop_count) ...@@ -1626,13 +1662,15 @@ finish_stop(int stop_count)
*/ */
if (stop_count < 0 || (current->ptrace & PT_PTRACED)) { if (stop_count < 0 || (current->ptrace & PT_PTRACED)) {
read_lock(&tasklist_lock); read_lock(&tasklist_lock);
do_notify_parent_cldstop(current, current->parent); do_notify_parent_cldstop(current, current->parent,
CLD_STOPPED);
read_unlock(&tasklist_lock); read_unlock(&tasklist_lock);
} }
else if (stop_count == 0) { else if (stop_count == 0) {
read_lock(&tasklist_lock); read_lock(&tasklist_lock);
do_notify_parent_cldstop(current->group_leader, do_notify_parent_cldstop(current->group_leader,
current->group_leader->real_parent); current->group_leader->real_parent,
CLD_STOPPED);
read_unlock(&tasklist_lock); read_unlock(&tasklist_lock);
} }
...@@ -1815,25 +1853,10 @@ int get_signal_to_deliver(siginfo_t *info, struct k_sigaction *return_ka, ...@@ -1815,25 +1853,10 @@ int get_signal_to_deliver(siginfo_t *info, struct k_sigaction *return_ka,
if ((current->ptrace & PT_PTRACED) && signr != SIGKILL) { if ((current->ptrace & PT_PTRACED) && signr != SIGKILL) {
ptrace_signal_deliver(regs, cookie); ptrace_signal_deliver(regs, cookie);
/*
* If there is a group stop in progress,
* we must participate in the bookkeeping.
*/
if (current->signal->group_stop_count > 0)
--current->signal->group_stop_count;
/* Let the debugger run. */ /* Let the debugger run. */
current->exit_code = signr; ptrace_stop(signr, info);
current->last_siginfo = info;
set_current_state(TASK_STOPPED);
spin_unlock_irq(&current->sighand->siglock);
notify_parent(current, SIGCHLD);
schedule();
current->last_siginfo = NULL;
/* We're back. Did the debugger cancel the sig? */ /* We're back. Did the debugger cancel the sig? */
spin_lock_irq(&current->sighand->siglock);
signr = current->exit_code; signr = current->exit_code;
if (signr == 0) if (signr == 0)
continue; continue;
...@@ -1964,7 +1987,7 @@ EXPORT_SYMBOL(kill_proc); ...@@ -1964,7 +1987,7 @@ EXPORT_SYMBOL(kill_proc);
EXPORT_SYMBOL(kill_proc_info); EXPORT_SYMBOL(kill_proc_info);
EXPORT_SYMBOL(kill_sl); EXPORT_SYMBOL(kill_sl);
EXPORT_SYMBOL(kill_sl_info); EXPORT_SYMBOL(kill_sl_info);
EXPORT_SYMBOL(notify_parent); EXPORT_SYMBOL(ptrace_notify);
EXPORT_SYMBOL(send_sig); EXPORT_SYMBOL(send_sig);
EXPORT_SYMBOL(send_sig_info); EXPORT_SYMBOL(send_sig_info);
EXPORT_SYMBOL(send_group_sig_info); EXPORT_SYMBOL(send_group_sig_info);
......
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