doc: RCU scheduler spinlock rcu_read_unlock() restriction remains

Given RCU flavor consolidation, when rcu_read_unlock() is invoked with
interrupts disabled, the reporting of the corresponding quiescent state is
deferred until interrupts are re-enabled.  There was therefore some hope
that this would allow dropping the restriction against holding scheduler
spinlocks across an rcu_read_unlock() without disabling interrupts across
the entire corresponding RCU read-side critical section.  Unfortunately,
the need to quickly provide a quiescent state to expedited grace periods
sometimes requires a call to raise_softirq() during rcu_read_unlock()
execution.  Because raise_softirq() can sometimes acquire the scheduler
spinlocks, the restriction must remain in effect.  This commit therefore
updates the RCU requirements documentation accordingly.

Signed-off-by: Paul E. McKenney <paulmck@linux.ibm.com>
This commit is contained in:
Paul E. McKenney 2018-10-15 10:54:13 -07:00
parent 97949f0176
commit 97562c0181

View File

@ -2475,23 +2475,37 @@ for context-switch-heavy <tt>CONFIG_NO_HZ_FULL=y</tt> workloads,
but there is room for further improvement. but there is room for further improvement.
<p> <p>
In the past, it was forbidden to disable interrupts across an It is forbidden to hold any of scheduler's runqueue or priority-inheritance
<tt>rcu_read_unlock()</tt> unless that interrupt-disabled region spinlocks across an <tt>rcu_read_unlock()</tt> unless interrupts have been
of code also included the matching <tt>rcu_read_lock()</tt>. disabled across the entire RCU read-side critical section, that is,
Violating this restriction could result in deadlocks involving the up to and including the matching <tt>rcu_read_lock()</tt>.
scheduler's runqueue and priority-inheritance spinlocks. Violating this restriction can result in deadlocks involving these
This restriction was lifted when interrupt-disabled calls to scheduler spinlocks.
<tt>rcu_read_unlock()</tt> started deferring the reporting of There was hope that this restriction might be lifted when interrupt-disabled
the resulting RCU-preempt quiescent state until the end of that calls to <tt>rcu_read_unlock()</tt> started deferring the reporting of
the resulting RCU-preempt quiescent state until the end of the corresponding
interrupts-disabled region. interrupts-disabled region.
This deferred reporting means that the scheduler's runqueue and Unfortunately, timely reporting of the corresponding quiescent state
priority-inheritance locks cannot be held while reporting an RCU-preempt to expedited grace periods requires a call to <tt>raise_softirq()</tt>,
quiescent state, which lifts the earlier restriction, at least from which can acquire these scheduler spinlocks.
a deadlock perspective. In addition, real-time systems using RCU priority boosting
Unfortunately, real-time systems using RCU priority boosting may
need this restriction to remain in effect because deferred need this restriction to remain in effect because deferred
quiescent-state reporting also defers deboosting, which in turn quiescent-state reporting would also defer deboosting, which in turn
degrades real-time latencies. would degrade real-time latencies.
<p>
In theory, if a given RCU read-side critical section could be
guaranteed to be less than one second in duration, holding a scheduler
spinlock across that critical section's <tt>rcu_read_unlock()</tt>
would require only that preemption be disabled across the entire
RCU read-side critical section, not interrupts.
Unfortunately, given the possibility of vCPU preemption, long-running
interrupts, and so on, it is not possible in practice to guarantee
that a given RCU read-side critical section will complete in less than
one second.
Therefore, as noted above, if scheduler spinlocks are held across
a given call to <tt>rcu_read_unlock()</tt>, interrupts must be
disabled across the entire RCU read-side critical section.
<h3><a name="Tracing and RCU">Tracing and RCU</a></h3> <h3><a name="Tracing and RCU">Tracing and RCU</a></h3>