diff options
author | Oleg Nesterov <oleg@redhat.com> | 2009-04-02 19:58:19 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-04-02 22:05:00 -0400 |
commit | 5dfc80be73dd0c212d2e6dd8dbf5afa07e680bbe (patch) | |
tree | ac173fb3fcfe2970781591f620f22d30f4ce090c | |
parent | 39c626ae47c469abdfd30c6e42eff884931380d6 (diff) |
forget_original_parent: do not abuse child->ptrace_entry
By discussion with Roland.
- Use ->sibling instead of ->ptrace_entry to chain the need to be
release_task'd childs. Nobody else can use ->sibling, this task
is EXIT_DEAD and nobody can find it on its own list.
- rename ptrace_dead to dead_childs.
- Now that we don't have the "parallel" untrace code, change back
reparent_thread() to return void, pass dead_childs as an argument.
Actually, I don't understand why do we notify /sbin/init when we
reparent a zombie, probably it is better to reap it unconditionally.
[akpm@linux-foundation.org: s/childs/children/]
Signed-off-by: Oleg Nesterov <oleg@redhat.com>
Cc: "Eric W. Biederman" <ebiederm@xmission.com>
Cc: "Metzger, Markus T" <markus.t.metzger@intel.com>
Cc: Roland McGrath <roland@redhat.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r-- | kernel/exit.c | 87 |
1 files changed, 41 insertions, 46 deletions
diff --git a/kernel/exit.c b/kernel/exit.c index 506693dfdd4e..029415d9f82e 100644 --- a/kernel/exit.c +++ b/kernel/exit.c | |||
@@ -726,46 +726,6 @@ static void exit_mm(struct task_struct * tsk) | |||
726 | mmput(mm); | 726 | mmput(mm); |
727 | } | 727 | } |
728 | 728 | ||
729 | /* Returns nonzero if the child should be released. */ | ||
730 | static int reparent_thread(struct task_struct *p, struct task_struct *father) | ||
731 | { | ||
732 | int dead; | ||
733 | |||
734 | if (p->pdeath_signal) | ||
735 | /* We already hold the tasklist_lock here. */ | ||
736 | group_send_sig_info(p->pdeath_signal, SEND_SIG_NOINFO, p); | ||
737 | |||
738 | list_move_tail(&p->sibling, &p->real_parent->children); | ||
739 | |||
740 | if (task_detached(p)) | ||
741 | return 0; | ||
742 | /* If this is a threaded reparent there is no need to | ||
743 | * notify anyone anything has happened. | ||
744 | */ | ||
745 | if (same_thread_group(p->real_parent, father)) | ||
746 | return 0; | ||
747 | |||
748 | /* We don't want people slaying init. */ | ||
749 | p->exit_signal = SIGCHLD; | ||
750 | |||
751 | /* If we'd notified the old parent about this child's death, | ||
752 | * also notify the new parent. | ||
753 | */ | ||
754 | dead = 0; | ||
755 | if (!p->ptrace && | ||
756 | p->exit_state == EXIT_ZOMBIE && thread_group_empty(p)) { | ||
757 | do_notify_parent(p, p->exit_signal); | ||
758 | if (task_detached(p)) { | ||
759 | p->exit_state = EXIT_DEAD; | ||
760 | dead = 1; | ||
761 | } | ||
762 | } | ||
763 | |||
764 | kill_orphaned_pgrp(p, father); | ||
765 | |||
766 | return dead; | ||
767 | } | ||
768 | |||
769 | /* | 729 | /* |
770 | * When we die, we re-parent all our children. | 730 | * When we die, we re-parent all our children. |
771 | * Try to give them to another thread in our thread | 731 | * Try to give them to another thread in our thread |
@@ -805,10 +765,46 @@ static struct task_struct *find_new_reaper(struct task_struct *father) | |||
805 | return pid_ns->child_reaper; | 765 | return pid_ns->child_reaper; |
806 | } | 766 | } |
807 | 767 | ||
768 | /* | ||
769 | * Any that need to be release_task'd are put on the @dead list. | ||
770 | */ | ||
771 | static void reparent_thread(struct task_struct *father, struct task_struct *p, | ||
772 | struct list_head *dead) | ||
773 | { | ||
774 | if (p->pdeath_signal) | ||
775 | group_send_sig_info(p->pdeath_signal, SEND_SIG_NOINFO, p); | ||
776 | |||
777 | list_move_tail(&p->sibling, &p->real_parent->children); | ||
778 | |||
779 | if (task_detached(p)) | ||
780 | return; | ||
781 | /* | ||
782 | * If this is a threaded reparent there is no need to | ||
783 | * notify anyone anything has happened. | ||
784 | */ | ||
785 | if (same_thread_group(p->real_parent, father)) | ||
786 | return; | ||
787 | |||
788 | /* We don't want people slaying init. */ | ||
789 | p->exit_signal = SIGCHLD; | ||
790 | |||
791 | /* If it has exited notify the new parent about this child's death. */ | ||
792 | if (!p->ptrace && | ||
793 | p->exit_state == EXIT_ZOMBIE && thread_group_empty(p)) { | ||
794 | do_notify_parent(p, p->exit_signal); | ||
795 | if (task_detached(p)) { | ||
796 | p->exit_state = EXIT_DEAD; | ||
797 | list_move_tail(&p->sibling, dead); | ||
798 | } | ||
799 | } | ||
800 | |||
801 | kill_orphaned_pgrp(p, father); | ||
802 | } | ||
803 | |||
808 | static void forget_original_parent(struct task_struct *father) | 804 | static void forget_original_parent(struct task_struct *father) |
809 | { | 805 | { |
810 | struct task_struct *p, *n, *reaper; | 806 | struct task_struct *p, *n, *reaper; |
811 | LIST_HEAD(ptrace_dead); | 807 | LIST_HEAD(dead_children); |
812 | 808 | ||
813 | exit_ptrace(father); | 809 | exit_ptrace(father); |
814 | 810 | ||
@@ -821,15 +817,14 @@ static void forget_original_parent(struct task_struct *father) | |||
821 | BUG_ON(p->ptrace); | 817 | BUG_ON(p->ptrace); |
822 | p->parent = p->real_parent; | 818 | p->parent = p->real_parent; |
823 | } | 819 | } |
824 | if (reparent_thread(p, father)) | 820 | reparent_thread(father, p, &dead_children); |
825 | list_add(&p->ptrace_entry, &ptrace_dead);; | ||
826 | } | 821 | } |
827 | |||
828 | write_unlock_irq(&tasklist_lock); | 822 | write_unlock_irq(&tasklist_lock); |
823 | |||
829 | BUG_ON(!list_empty(&father->children)); | 824 | BUG_ON(!list_empty(&father->children)); |
830 | 825 | ||
831 | list_for_each_entry_safe(p, n, &ptrace_dead, ptrace_entry) { | 826 | list_for_each_entry_safe(p, n, &dead_children, sibling) { |
832 | list_del_init(&p->ptrace_entry); | 827 | list_del_init(&p->sibling); |
833 | release_task(p); | 828 | release_task(p); |
834 | } | 829 | } |
835 | } | 830 | } |