Commit 0b8ad905 authored by Matthew Dempsky's avatar Matthew Dempsky Committed by Ben Hutchings

ptrace: fix fork event messages across pid namespaces

commit 4e52365f upstream.

When tracing a process in another pid namespace, it's important for fork
event messages to contain the child's pid as seen from the tracer's pid
namespace, not the parent's.  Otherwise, the tracer won't be able to
correlate the fork event with later SIGTRAP signals it receives from the
child.

We still risk a race condition if a ptracer from a different pid
namespace attaches after we compute the pid_t value.  However, sending a
bogus fork event message in this unlikely scenario is still a vast
improvement over the status quo where we always send bogus fork event
messages to debuggers in a different pid namespace than the forking
process.
Signed-off-by: default avatarMatthew Dempsky <mdempsky@chromium.org>
Acked-by: default avatarOleg Nesterov <oleg@redhat.com>
Cc: Kees Cook <keescook@chromium.org>
Cc: Julien Tinnes <jln@chromium.org>
Cc: Roland McGrath <mcgrathr@chromium.org>
Cc: Jan Kratochvil <jan.kratochvil@redhat.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
[bwh: Backported to 3.2: adjust context]
Signed-off-by: default avatarBen Hutchings <ben@decadent.org.uk>
parent 6df2529a
...@@ -112,6 +112,7 @@ ...@@ -112,6 +112,7 @@
#include <linux/compiler.h> /* For unlikely. */ #include <linux/compiler.h> /* For unlikely. */
#include <linux/sched.h> /* For struct task_struct. */ #include <linux/sched.h> /* For struct task_struct. */
#include <linux/pid_namespace.h> /* For task_active_pid_ns. */
extern long arch_ptrace(struct task_struct *child, long request, extern long arch_ptrace(struct task_struct *child, long request,
...@@ -203,6 +204,37 @@ static inline void ptrace_event(int event, unsigned long message) ...@@ -203,6 +204,37 @@ static inline void ptrace_event(int event, unsigned long message)
} }
} }
/**
* ptrace_event_pid - possibly stop for a ptrace event notification
* @event: %PTRACE_EVENT_* value to report
* @pid: process identifier for %PTRACE_GETEVENTMSG to return
*
* Check whether @event is enabled and, if so, report @event and @pid
* to the ptrace parent. @pid is reported as the pid_t seen from the
* the ptrace parent's pid namespace.
*
* Called without locks.
*/
static inline void ptrace_event_pid(int event, struct pid *pid)
{
/*
* FIXME: There's a potential race if a ptracer in a different pid
* namespace than parent attaches between computing message below and
* when we acquire tasklist_lock in ptrace_stop(). If this happens,
* the ptracer will get a bogus pid from PTRACE_GETEVENTMSG.
*/
unsigned long message = 0;
struct pid_namespace *ns;
rcu_read_lock();
ns = task_active_pid_ns(rcu_dereference(current->parent));
if (ns)
message = pid_nr_ns(pid, ns);
rcu_read_unlock();
ptrace_event(event, message);
}
/** /**
* ptrace_init_task - initialize ptrace state for a new child * ptrace_init_task - initialize ptrace state for a new child
* @child: new child task * @child: new child task
......
...@@ -1513,10 +1513,12 @@ long do_fork(unsigned long clone_flags, ...@@ -1513,10 +1513,12 @@ long do_fork(unsigned long clone_flags,
*/ */
if (!IS_ERR(p)) { if (!IS_ERR(p)) {
struct completion vfork; struct completion vfork;
struct pid *pid;
trace_sched_process_fork(current, p); trace_sched_process_fork(current, p);
nr = task_pid_vnr(p); pid = get_task_pid(p, PIDTYPE_PID);
nr = pid_vnr(pid);
if (clone_flags & CLONE_PARENT_SETTID) if (clone_flags & CLONE_PARENT_SETTID)
put_user(nr, parent_tidptr); put_user(nr, parent_tidptr);
...@@ -1540,14 +1542,16 @@ long do_fork(unsigned long clone_flags, ...@@ -1540,14 +1542,16 @@ long do_fork(unsigned long clone_flags,
/* forking complete and child started to run, tell ptracer */ /* forking complete and child started to run, tell ptracer */
if (unlikely(trace)) if (unlikely(trace))
ptrace_event(trace, nr); ptrace_event_pid(trace, pid);
if (clone_flags & CLONE_VFORK) { if (clone_flags & CLONE_VFORK) {
freezer_do_not_count(); freezer_do_not_count();
wait_for_completion(&vfork); wait_for_completion(&vfork);
freezer_count(); freezer_count();
ptrace_event(PTRACE_EVENT_VFORK_DONE, nr); ptrace_event_pid(PTRACE_EVENT_VFORK_DONE, pid);
} }
put_pid(pid);
} else { } else {
nr = PTR_ERR(p); nr = PTR_ERR(p);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment