]> git.baikalelectronics.ru Git - kernel.git/commit
tracing: Have branch tracer use recursive field of task struct
authorSteven Rostedt (Red Hat) <rostedt@goodmis.org>
Tue, 7 Jul 2015 19:05:03 +0000 (15:05 -0400)
committerSteven Rostedt <rostedt@goodmis.org>
Wed, 8 Jul 2015 15:53:45 +0000 (11:53 -0400)
commit10db0352a448a01125373a4ab187e59d4d46bcc7
tree1397e410bb96d75fe5de6689145e6a8ea250a7c5
parent0c9e0229f3c4e023815b9350b2095fd8ad5809e3
tracing: Have branch tracer use recursive field of task struct

Fengguang Wu's tests triggered a bug in the branch tracer's start up
test when CONFIG_DEBUG_PREEMPT set. This was because that config
adds some debug logic in the per cpu field, which calls back into
the branch tracer.

The branch tracer has its own recursive checks, but uses a per cpu
variable to implement it. If retrieving the per cpu variable calls
back into the branch tracer, you can see how things will break.

Instead of using a per cpu variable, use the trace_recursion field
of the current task struct. Simply set a bit when entering the
branch tracing and clear it when leaving. If the bit is set on
entry, just don't do the tracing.

There's also the case with lockdep, as the local_irq_save() called
before the recursion can also trigger code that can call back into
the function. Changing that to a raw_local_irq_save() will protect
that as well.

This prevents the recursion and the inevitable crash that follows.

Link: http://lkml.kernel.org/r/20150630141803.GA28071@wfg-t540p.sh.intel.com
Cc: stable@vger.kernel.org # 3.10+
Reported-by: Fengguang Wu <fengguang.wu@intel.com>
Tested-by: Fengguang Wu <fengguang.wu@intel.com>
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
kernel/trace/trace.h
kernel/trace/trace_branch.c