Fengguang Wu discovered a crash that happened to be because of the branch

tracer (traces unlikely and likely branches) when enabled with certain
 debug options.
 
 What happened was that various debug options like lockdep and DEBUG_PREEMPT
 can cause parts of the branch tracer to recurse outside its recursion
 protection. In fact, part of its recursion protection used these features
 that caused the lockup. This cleans up the code a little and makes the
 recursion protection a bit more robust.
 -----BEGIN PGP SIGNATURE-----
 Version: GnuPG v1
 
 iQEcBAABAgAGBQJVovjwAAoJEEjnJuOKh9ldYacH/3VVIcBhRbuUmANZJSYLXSUh
 nEx6FOAfmla/7hH2dVSm8JmnbUzWyunttjbZ6/O8PFE1gumuvVrfWrIV8iwUe4J6
 6Z2KdbKd+3FpaSKEnX61UQ1cfR+1eFLOLH9CH5O4twVPyLzvI+NpaJQJaNoX0ywq
 vqsUMx63gKdGwhC6BhLi0t/xsTuuIgGjDAjuaF2yNZCuBw9UtziedxK5pveH2OFX
 G8dAVfP18aqsXaRMj1LUrm6wRUP0BD7B2v99jdUu2UHYTtmBzy8Vm6RfhJ4Gk8d7
 WnIkaBBU5iu75E9ec35MtA52zQ+8b8O/fpIQZgJgFRr7uaf+hvbXjF0UScWE6vU=
 =ujzT
 -----END PGP SIGNATURE-----

Merge tag 'trace-v4.2-rc1-fix' of git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-trace

Pull tracing fix from Steven Rostedt:
 "Fengguang Wu discovered a crash that happened to be because of the
  branch tracer (traces unlikely and likely branches) when enabled with
  certain debug options.

  What happened was that various debug options like lockdep and
  DEBUG_PREEMPT can cause parts of the branch tracer to recurse outside
  its recursion protection.  In fact, part of its recursion protection
  used these features that caused the lockup.  This cleans up the code a
  little and makes the recursion protection a bit more robust"

* tag 'trace-v4.2-rc1-fix' of git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-trace:
  tracing: Have branch tracer use recursive field of task struct
This commit is contained in:
Linus Torvalds 2015-07-15 11:14:10 -07:00
commit 7558009751
2 changed files with 11 additions and 7 deletions

View File

@ -444,6 +444,7 @@ enum {
TRACE_CONTROL_BIT, TRACE_CONTROL_BIT,
TRACE_BRANCH_BIT,
/* /*
* Abuse of the trace_recursion. * Abuse of the trace_recursion.
* As we need a way to maintain state if we are tracing the function * As we need a way to maintain state if we are tracing the function

View File

@ -36,9 +36,12 @@ probe_likely_condition(struct ftrace_branch_data *f, int val, int expect)
struct trace_branch *entry; struct trace_branch *entry;
struct ring_buffer *buffer; struct ring_buffer *buffer;
unsigned long flags; unsigned long flags;
int cpu, pc; int pc;
const char *p; const char *p;
if (current->trace_recursion & TRACE_BRANCH_BIT)
return;
/* /*
* I would love to save just the ftrace_likely_data pointer, but * I would love to save just the ftrace_likely_data pointer, but
* this code can also be used by modules. Ugly things can happen * this code can also be used by modules. Ugly things can happen
@ -49,10 +52,10 @@ probe_likely_condition(struct ftrace_branch_data *f, int val, int expect)
if (unlikely(!tr)) if (unlikely(!tr))
return; return;
local_irq_save(flags); raw_local_irq_save(flags);
cpu = raw_smp_processor_id(); current->trace_recursion |= TRACE_BRANCH_BIT;
data = per_cpu_ptr(tr->trace_buffer.data, cpu); data = this_cpu_ptr(tr->trace_buffer.data);
if (atomic_inc_return(&data->disabled) != 1) if (atomic_read(&data->disabled))
goto out; goto out;
pc = preempt_count(); pc = preempt_count();
@ -81,8 +84,8 @@ probe_likely_condition(struct ftrace_branch_data *f, int val, int expect)
__buffer_unlock_commit(buffer, event); __buffer_unlock_commit(buffer, event);
out: out:
atomic_dec(&data->disabled); current->trace_recursion &= ~TRACE_BRANCH_BIT;
local_irq_restore(flags); raw_local_irq_restore(flags);
} }
static inline static inline