aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPaul E. McKenney <paulmck@linux.vnet.ibm.com>2015-01-21 18:26:03 -0500
committerPaul E. McKenney <paulmck@linux.vnet.ibm.com>2015-03-03 14:16:00 -0500
commitd24209bb689e2c7f7418faec9b4a948e922d24da (patch)
tree40db37902a30acb68f5733e666f25ea7b52e8485
parent6629240575992a6f0d18c46f5160b34527b0e501 (diff)
rcu: Improve diagnostics for blocked critical sections in irq
If an RCU read-side critical section occurs within an interrupt handler or a softirq handler, it cannot have been preempted. Therefore, there is a check in rcu_read_unlock_special() checking for this error. However, when this check triggers, it lacks diagnostic information. This commit therefore moves rcu_read_unlock()'s lockdep annotation to follow the call to __rcu_read_unlock() and changes rcu_read_unlock_special()'s WARN_ON_ONCE() to an lockdep_rcu_suspicious() in order to locate where the offending RCU read-side critical section began. In addition, the value of the ->rcu_read_unlock_special field is printed. Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
-rw-r--r--include/linux/lockdep.h7
-rw-r--r--include/linux/rcupdate.h2
-rw-r--r--kernel/rcu/tree_plugin.h8
3 files changed, 14 insertions, 3 deletions
diff --git a/include/linux/lockdep.h b/include/linux/lockdep.h
index 74ab23176e9b..066ba4157541 100644
--- a/include/linux/lockdep.h
+++ b/include/linux/lockdep.h
@@ -531,8 +531,13 @@ do { \
531# define might_lock_read(lock) do { } while (0) 531# define might_lock_read(lock) do { } while (0)
532#endif 532#endif
533 533
534#ifdef CONFIG_PROVE_RCU 534#ifdef CONFIG_LOCKDEP
535void lockdep_rcu_suspicious(const char *file, const int line, const char *s); 535void lockdep_rcu_suspicious(const char *file, const int line, const char *s);
536#else
537static inline void
538lockdep_rcu_suspicious(const char *file, const int line, const char *s)
539{
540}
536#endif 541#endif
537 542
538#endif /* __LINUX_LOCKDEP_H */ 543#endif /* __LINUX_LOCKDEP_H */
diff --git a/include/linux/rcupdate.h b/include/linux/rcupdate.h
index 3e6afed51051..70b896e16f19 100644
--- a/include/linux/rcupdate.h
+++ b/include/linux/rcupdate.h
@@ -942,9 +942,9 @@ static inline void rcu_read_unlock(void)
942{ 942{
943 rcu_lockdep_assert(rcu_is_watching(), 943 rcu_lockdep_assert(rcu_is_watching(),
944 "rcu_read_unlock() used illegally while idle"); 944 "rcu_read_unlock() used illegally while idle");
945 rcu_lock_release(&rcu_lock_map);
946 __release(RCU); 945 __release(RCU);
947 __rcu_read_unlock(); 946 __rcu_read_unlock();
947 rcu_lock_release(&rcu_lock_map); /* Keep acq info for rls diags. */
948} 948}
949 949
950/** 950/**
diff --git a/kernel/rcu/tree_plugin.h b/kernel/rcu/tree_plugin.h
index 0a571e9a0f1d..8a33920b8845 100644
--- a/kernel/rcu/tree_plugin.h
+++ b/kernel/rcu/tree_plugin.h
@@ -334,7 +334,13 @@ void rcu_read_unlock_special(struct task_struct *t)
334 } 334 }
335 335
336 /* Hardware IRQ handlers cannot block, complain if they get here. */ 336 /* Hardware IRQ handlers cannot block, complain if they get here. */
337 if (WARN_ON_ONCE(in_irq() || in_serving_softirq())) { 337 if (in_irq() || in_serving_softirq()) {
338 lockdep_rcu_suspicious(__FILE__, __LINE__,
339 "rcu_read_unlock() from irq or softirq with blocking in critical section!!!\n");
340 pr_alert("->rcu_read_unlock_special: %#x (b: %d, nq: %d)\n",
341 t->rcu_read_unlock_special.s,
342 t->rcu_read_unlock_special.b.blocked,
343 t->rcu_read_unlock_special.b.need_qs);
338 local_irq_restore(flags); 344 local_irq_restore(flags);
339 return; 345 return;
340 } 346 }