aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/exit.c
diff options
context:
space:
mode:
authorOleg Nesterov <oleg@redhat.com>2011-06-23 13:06:50 -0400
committerOleg Nesterov <oleg@redhat.com>2011-06-27 14:30:08 -0400
commit45cdf5cc0703c537194588c63d53bad1f2539d36 (patch)
tree579573d2cbe054d767a9f6b575b88836586fb7ce /kernel/exit.c
parent53c8f9f199b239668e6b1a907735ee323a0d1ccd (diff)
kill tracehook_notify_death()
Kill tracehook_notify_death(), reimplement the logic in its caller, exit_notify(). Also, change the exec_id's check to use thread_group_leader() instead of task_detached(), this is more clear. This logic only applies to the exiting leader, a sub-thread must never change its exit_signal. Note: when the traced group leader exits the exit_signal-or-SIGCHLD logic looks really strange: - we notify the tracer even if !thread_group_empty() but do_wait(WEXITED) can't work until all threads exit - if the tracer is real_parent, it is not clear why can't we use ->exit_signal event if !thread_group_empty() -v2: do not try to fix the 2nd oddity to avoid the subtle behavior change mixed with reorganization, suggested by Tejun. Signed-off-by: Oleg Nesterov <oleg@redhat.com> Reviewed-by: Tejun Heo <tj@kernel.org>
Diffstat (limited to 'kernel/exit.c')
-rw-r--r--kernel/exit.c21
1 files changed, 13 insertions, 8 deletions
diff --git a/kernel/exit.c b/kernel/exit.c
index 34d135f4fccc..bb08e938ca74 100644
--- a/kernel/exit.c
+++ b/kernel/exit.c
@@ -819,9 +819,7 @@ static void forget_original_parent(struct task_struct *father)
819 */ 819 */
820static void exit_notify(struct task_struct *tsk, int group_dead) 820static void exit_notify(struct task_struct *tsk, int group_dead)
821{ 821{
822 int signal;
823 bool autoreap; 822 bool autoreap;
824 void *cookie;
825 823
826 /* 824 /*
827 * This does two things: 825 * This does two things:
@@ -852,16 +850,23 @@ static void exit_notify(struct task_struct *tsk, int group_dead)
852 * we have changed execution domain as these two values started 850 * we have changed execution domain as these two values started
853 * the same after a fork. 851 * the same after a fork.
854 */ 852 */
855 if (tsk->exit_signal != SIGCHLD && !task_detached(tsk) && 853 if (thread_group_leader(tsk) && tsk->exit_signal != SIGCHLD &&
856 (tsk->parent_exec_id != tsk->real_parent->self_exec_id || 854 (tsk->parent_exec_id != tsk->real_parent->self_exec_id ||
857 tsk->self_exec_id != tsk->parent_exec_id)) 855 tsk->self_exec_id != tsk->parent_exec_id))
858 tsk->exit_signal = SIGCHLD; 856 tsk->exit_signal = SIGCHLD;
859 857
860 signal = tracehook_notify_death(tsk, &cookie, group_dead); 858 if (unlikely(tsk->ptrace)) {
861 if (signal >= 0) 859 int sig = thread_group_leader(tsk) &&
862 autoreap = do_notify_parent(tsk, signal); 860 thread_group_empty(tsk) &&
863 else 861 !ptrace_reparented(tsk) ?
864 autoreap = (signal == DEATH_REAP); 862 tsk->exit_signal : SIGCHLD;
863 autoreap = do_notify_parent(tsk, sig);
864 } else if (thread_group_leader(tsk)) {
865 autoreap = thread_group_empty(tsk) &&
866 do_notify_parent(tsk, tsk->exit_signal);
867 } else {
868 autoreap = true;
869 }
865 870
866 tsk->exit_state = autoreap ? EXIT_DEAD : EXIT_ZOMBIE; 871 tsk->exit_state = autoreap ? EXIT_DEAD : EXIT_ZOMBIE;
867 872