trace, lockdep: manual preempt count adding for local_bh_disable
authorSteven Rostedt <srostedt@redhat.com>
Fri, 23 Jan 2009 00:01:40 +0000 (19:01 -0500)
committerIngo Molnar <mingo@elte.hu>
Fri, 23 Jan 2009 10:10:57 +0000 (11:10 +0100)
commit7e49fcce1bdadd723ae6a0b3b324c4daced61563
treea2bf1a143ed33ca01612dfab1fb7c993c467cdb0
parentb06a830183b610c0a88c29a92feb7991a867ab46
trace, lockdep: manual preempt count adding for local_bh_disable

Impact: fix to preempt trace triggering lockdep check_flag failure

In local_bh_disable, the use of add_preempt_count causes the
preempt tracer to start recording the time preemption is off.
But because it already modified the preempt_count to show
softirqs disabled, and before it called the lockdep code to
handle this, it causes a state that lockdep can not handle.

The preempt tracer will reset the ring buffer on start of a trace,
and the ring buffer reset code does a spin_lock_irqsave. This
calls into lockdep and lockdep will fail when it detects the
invalid state of having softirqs disabled but the internal
current->softirqs_enabled is still set.

The fix is to manually add the SOFTIRQ_OFFSET to preempt count
and call the preempt tracer code outside the lockdep critical
area.

Thanks to Peter Zijlstra for suggesting this solution.

Signed-off-by: Steven Rostedt <srostedt@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
include/linux/sched.h
kernel/sched.c
kernel/softirq.c