diff options
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/exit.c | 25 |
1 files changed, 8 insertions, 17 deletions
diff --git a/kernel/exit.c b/kernel/exit.c index 0c06b9efae3b..7a8311422930 100644 --- a/kernel/exit.c +++ b/kernel/exit.c | |||
@@ -732,19 +732,15 @@ static void exit_mm(struct task_struct * tsk) | |||
732 | } | 732 | } |
733 | 733 | ||
734 | /* | 734 | /* |
735 | * Return nonzero if @parent's children should reap themselves. | 735 | * Called with irqs disabled, returns true if childs should reap themselves. |
736 | * | ||
737 | * Called with write_lock_irq(&tasklist_lock) held. | ||
738 | */ | 736 | */ |
739 | static int ignoring_children(struct task_struct *parent) | 737 | static int ignoring_children(struct sighand_struct *sigh) |
740 | { | 738 | { |
741 | int ret; | 739 | int ret; |
742 | struct sighand_struct *psig = parent->sighand; | 740 | spin_lock(&sigh->siglock); |
743 | unsigned long flags; | 741 | ret = (sigh->action[SIGCHLD-1].sa.sa_handler == SIG_IGN) || |
744 | spin_lock_irqsave(&psig->siglock, flags); | 742 | (sigh->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDWAIT); |
745 | ret = (psig->action[SIGCHLD-1].sa.sa_handler == SIG_IGN || | 743 | spin_unlock(&sigh->siglock); |
746 | (psig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDWAIT)); | ||
747 | spin_unlock_irqrestore(&psig->siglock, flags); | ||
748 | return ret; | 744 | return ret; |
749 | } | 745 | } |
750 | 746 | ||
@@ -757,7 +753,6 @@ static int ignoring_children(struct task_struct *parent) | |||
757 | static void ptrace_exit(struct task_struct *parent, struct list_head *dead) | 753 | static void ptrace_exit(struct task_struct *parent, struct list_head *dead) |
758 | { | 754 | { |
759 | struct task_struct *p, *n; | 755 | struct task_struct *p, *n; |
760 | int ign = -1; | ||
761 | 756 | ||
762 | list_for_each_entry_safe(p, n, &parent->ptraced, ptrace_entry) { | 757 | list_for_each_entry_safe(p, n, &parent->ptraced, ptrace_entry) { |
763 | __ptrace_unlink(p); | 758 | __ptrace_unlink(p); |
@@ -779,12 +774,8 @@ static void ptrace_exit(struct task_struct *parent, struct list_head *dead) | |||
779 | if (!task_detached(p) && thread_group_empty(p)) { | 774 | if (!task_detached(p) && thread_group_empty(p)) { |
780 | if (!same_thread_group(p->real_parent, parent)) | 775 | if (!same_thread_group(p->real_parent, parent)) |
781 | do_notify_parent(p, p->exit_signal); | 776 | do_notify_parent(p, p->exit_signal); |
782 | else { | 777 | else if (ignoring_children(parent->sighand)) |
783 | if (ign < 0) | 778 | p->exit_signal = -1; |
784 | ign = ignoring_children(parent); | ||
785 | if (ign) | ||
786 | p->exit_signal = -1; | ||
787 | } | ||
788 | } | 779 | } |
789 | 780 | ||
790 | if (task_detached(p)) { | 781 | if (task_detached(p)) { |