aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/exit.c21
1 files changed, 9 insertions, 12 deletions
diff --git a/kernel/exit.c b/kernel/exit.c
index c3691cbc220a..da28745f7c38 100644
--- a/kernel/exit.c
+++ b/kernel/exit.c
@@ -163,27 +163,17 @@ static void delayed_put_task_struct(struct rcu_head *rhp)
163 put_task_struct(container_of(rhp, struct task_struct, rcu)); 163 put_task_struct(container_of(rhp, struct task_struct, rcu));
164} 164}
165 165
166/*
167 * Do final ptrace-related cleanup of a zombie being reaped.
168 *
169 * Called with write_lock(&tasklist_lock) held.
170 */
171static void ptrace_release_task(struct task_struct *p)
172{
173 BUG_ON(!list_empty(&p->ptraced));
174 ptrace_unlink(p);
175 BUG_ON(!list_empty(&p->ptrace_entry));
176}
177 166
178void release_task(struct task_struct * p) 167void release_task(struct task_struct * p)
179{ 168{
180 struct task_struct *leader; 169 struct task_struct *leader;
181 int zap_leader; 170 int zap_leader;
182repeat: 171repeat:
172 tracehook_prepare_release_task(p);
183 atomic_dec(&p->user->processes); 173 atomic_dec(&p->user->processes);
184 proc_flush_task(p); 174 proc_flush_task(p);
185 write_lock_irq(&tasklist_lock); 175 write_lock_irq(&tasklist_lock);
186 ptrace_release_task(p); 176 tracehook_finish_release_task(p);
187 __exit_signal(p); 177 __exit_signal(p);
188 178
189 /* 179 /*
@@ -205,6 +195,13 @@ repeat:
205 * that case. 195 * that case.
206 */ 196 */
207 zap_leader = task_detached(leader); 197 zap_leader = task_detached(leader);
198
199 /*
200 * This maintains the invariant that release_task()
201 * only runs on a task in EXIT_DEAD, just for sanity.
202 */
203 if (zap_leader)
204 leader->exit_state = EXIT_DEAD;
208 } 205 }
209 206
210 write_unlock_irq(&tasklist_lock); 207 write_unlock_irq(&tasklist_lock);