Re: [patch] More ptrace fixes for 2.5.33

From: Ingo Molnar (mingo@elte.hu)
Date: Sat Sep 07 2002 - 11:27:33 EST


On Fri, 6 Sep 2002, Daniel Jacobowitz wrote:

> Thanks for your collected resync patch this morning. Here are the
> changes I have on top of BK-curr + that patch:
>
> - Fix some bugs I introduced in zap_thread
> - Improve the check for traced children in sys_wait4
> - Fix parent links when using CLONE_PTRACE

(your patch did not apply cleanly because it was put together from two
different-depth patchfiles it appears - the attached patch is the same
patch fixed up.)

        Ingo

--- linux/kernel/exit.c.orig Sat Sep 7 18:25:57 2002
+++ linux/kernel/exit.c Sat Sep 7 18:26:31 2002
@@ -449,15 +449,19 @@
 
 static inline void zap_thread(task_t *p, task_t *father, int traced)
 {
- /* If we were tracing the thread, release it; otherwise preserve the
- ptrace links. */
+ /* If someone else is tracing this thread, preserve the ptrace links. */
         if (unlikely(traced)) {
                 task_t *trace_task = p->parent;
+ int ptrace_flag = p->ptrace;
+ BUG_ON (ptrace_flag == 0);
+
                 __ptrace_unlink(p);
- p->ptrace = 1;
+ p->ptrace = ptrace_flag;
                 __ptrace_link(p, trace_task);
         } else {
- p->ptrace = 0;
+ /* Otherwise, if we were tracing this thread, untrace it. */
+ ptrace_unlink (p);
+
                 list_del_init(&p->sibling);
                 p->parent = p->real_parent;
                 list_add_tail(&p->sibling, &p->parent->children);
@@ -646,6 +650,41 @@
         do_exit((error_code&0xff)<<8);
 }
 
+static inline int eligible_child(pid_t pid, int options, task_t *p)
+{
+ if (pid>0) {
+ if (p->pid != pid)
+ return 0;
+ } else if (!pid) {
+ if (p->pgrp != current->pgrp)
+ return 0;
+ } else if (pid != -1) {
+ if (p->pgrp != -pid)
+ return 0;
+ }
+
+ /*
+ * Do not consider detached threads that are
+ * not ptraced:
+ */
+ if (p->exit_signal == -1 && !p->ptrace)
+ return 0;
+
+ /* Wait for all children (clone and not) if __WALL is set;
+ * otherwise, wait for clone children *only* if __WCLONE is
+ * set; otherwise, wait for non-clone children *only*. (Note:
+ * A "clone" child here is one that reports to its parent
+ * using a signal other than SIGCHLD.) */
+ if (((p->exit_signal != SIGCHLD) ^ ((options & __WCLONE) != 0))
+ && !(options & __WALL))
+ return 0;
+
+ if (security_ops->task_wait(p))
+ return 0;
+
+ return 1;
+}
+
 asmlinkage long sys_wait4(pid_t pid,unsigned int * stat_addr, int options, struct rusage * ru)
 {
         int flag, retval;
@@ -666,34 +705,8 @@
                 struct list_head *_p;
                 list_for_each(_p,&tsk->children) {
                         p = list_entry(_p,struct task_struct,sibling);
- if (pid>0) {
- if (p->pid != pid)
- continue;
- } else if (!pid) {
- if (p->pgrp != current->pgrp)
- continue;
- } else if (pid != -1) {
- if (p->pgrp != -pid)
- continue;
- }
- /*
- * Do not consider detached threads that are
- * not ptraced:
- */
- if (p->exit_signal == -1 && !p->ptrace)
- continue;
- /* Wait for all children (clone and not) if __WALL is set;
- * otherwise, wait for clone children *only* if __WCLONE is
- * set; otherwise, wait for non-clone children *only*. (Note:
- * A "clone" child here is one that reports to its parent
- * using a signal other than SIGCHLD.) */
- if (((p->exit_signal != SIGCHLD) ^ ((options & __WCLONE) != 0))
- && !(options & __WALL))
+ if (!eligible_child(pid, options, p))
                                 continue;
-
- if (security_ops->task_wait(p))
- continue;
-
                         flag = 1;
                         switch (p->state) {
                         case TASK_STOPPED:
@@ -738,12 +751,21 @@
                                 continue;
                         }
                 }
+ if (!flag) {
+ list_for_each (_p,&tsk->ptrace_children) {
+ p = list_entry(_p,struct task_struct,ptrace_list);
+ if (!eligible_child(pid, options, p))
+ continue;
+ flag = 1;
+ break;
+ }
+ }
                 if (options & __WNOTHREAD)
                         break;
                 tsk = next_thread(tsk);
         } while (tsk != current);
         read_unlock(&tasklist_lock);
- if (flag || !list_empty(&current->ptrace_children)) {
+ if (flag) {
                 retval = 0;
                 if (options & WNOHANG)
                         goto end_wait4;
--- linux/kernel/fork.c.orig Sat Sep 7 18:25:48 2002
+++ linux/kernel/fork.c Sat Sep 7 18:26:41 2002
@@ -836,13 +836,11 @@
         write_lock_irq(&tasklist_lock);
 
         /* CLONE_PARENT re-uses the old parent */
- p->real_parent = current->real_parent;
- p->parent = current->parent;
- if (!(clone_flags & CLONE_PARENT)) {
+ if (clone_flags & CLONE_PARENT)
+ p->real_parent = current->real_parent;
+ else
                 p->real_parent = current;
- if (!(p->ptrace & PT_PTRACED))
- p->parent = current;
- }
+ p->parent = p->real_parent;
 
         if (clone_flags & CLONE_THREAD) {
                 p->tgid = current->tgid;
@@ -850,7 +848,8 @@
         }
 
         SET_LINKS(p);
- ptrace_link(p, p->parent);
+ if (p->ptrace & PT_PTRACED)
+ __ptrace_link(p, current->parent);
         hash_pid(p);
         nr_threads++;
         write_unlock_irq(&tasklist_lock);

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/



This archive was generated by hypermail 2b29 : Sat Sep 07 2002 - 22:00:32 EST