Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
4de5f89e
Commit
4de5f89e
authored
Jun 06, 2017
by
Paul E. McKenney
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
doc: Update RCU documentation
Signed-off-by:
Paul E. McKenney
<
paulmck@linux.vnet.ibm.com
>
parent
f99bcb2c
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
111 additions
and
49 deletions
+111
-49
Documentation/RCU/checklist.txt
Documentation/RCU/checklist.txt
+85
-36
Documentation/RCU/rcu.txt
Documentation/RCU/rcu.txt
+3
-6
Documentation/RCU/rcubarrier.txt
Documentation/RCU/rcubarrier.txt
+5
-0
Documentation/RCU/torture.txt
Documentation/RCU/torture.txt
+14
-6
Documentation/RCU/whatisRCU.txt
Documentation/RCU/whatisRCU.txt
+4
-1
No files found.
Documentation/RCU/checklist.txt
View file @
4de5f89e
...
@@ -23,6 +23,14 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -23,6 +23,14 @@ over a rather long period of time, but improvements are always welcome!
Yet another exception is where the low real-time latency of RCU's
Yet another exception is where the low real-time latency of RCU's
read-side primitives is critically important.
read-side primitives is critically important.
One final exception is where RCU readers are used to prevent
the ABA problem (https://en.wikipedia.org/wiki/ABA_problem)
for lockless updates. This does result in the mildly
counter-intuitive situation where rcu_read_lock() and
rcu_read_unlock() are used to protect updates, however, this
approach provides the same potential simplifications that garbage
collectors do.
1. Does the update code have proper mutual exclusion?
1. Does the update code have proper mutual exclusion?
RCU does allow -readers- to run (almost) naked, but -writers- must
RCU does allow -readers- to run (almost) naked, but -writers- must
...
@@ -40,7 +48,9 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -40,7 +48,9 @@ over a rather long period of time, but improvements are always welcome!
explain how this single task does not become a major bottleneck on
explain how this single task does not become a major bottleneck on
big multiprocessor machines (for example, if the task is updating
big multiprocessor machines (for example, if the task is updating
information relating to itself that other tasks can read, there
information relating to itself that other tasks can read, there
by definition can be no bottleneck).
by definition can be no bottleneck). Note that the definition
of "large" has changed significantly: Eight CPUs was "large"
in the year 2000, but a hundred CPUs was unremarkable in 2017.
2. Do the RCU read-side critical sections make proper use of
2. Do the RCU read-side critical sections make proper use of
rcu_read_lock() and friends? These primitives are needed
rcu_read_lock() and friends? These primitives are needed
...
@@ -55,6 +65,12 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -55,6 +65,12 @@ over a rather long period of time, but improvements are always welcome!
Disabling of preemption can serve as rcu_read_lock_sched(), but
Disabling of preemption can serve as rcu_read_lock_sched(), but
is less readable.
is less readable.
Letting RCU-protected pointers "leak" out of an RCU read-side
critical section is every bid as bad as letting them leak out
from under a lock. Unless, of course, you have arranged some
other means of protection, such as a lock or a reference count
-before- letting them out of the RCU read-side critical section.
3. Does the update code tolerate concurrent accesses?
3. Does the update code tolerate concurrent accesses?
The whole point of RCU is to permit readers to run without
The whole point of RCU is to permit readers to run without
...
@@ -81,7 +97,7 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -81,7 +97,7 @@ over a rather long period of time, but improvements are always welcome!
c. Make updates appear atomic to readers. For example,
c. Make updates appear atomic to readers. For example,
pointer updates to properly aligned fields will
pointer updates to properly aligned fields will
appear atomic, as will individual atomic primitives.
appear atomic, as will individual atomic primitives.
Sequences of perations performed under a lock will -not-
Sequences of
o
perations performed under a lock will -not-
appear to be atomic to RCU readers, nor will sequences
appear to be atomic to RCU readers, nor will sequences
of multiple atomic primitives.
of multiple atomic primitives.
...
@@ -168,8 +184,8 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -168,8 +184,8 @@ over a rather long period of time, but improvements are always welcome!
5. If call_rcu(), or a related primitive such as call_rcu_bh(),
5. If call_rcu(), or a related primitive such as call_rcu_bh(),
call_rcu_sched(), or call_srcu() is used, the callback function
call_rcu_sched(), or call_srcu() is used, the callback function
must be written to be called from softirq context. In particular,
will be called from softirq context. In particular, it cannot
it cannot
block.
block.
6. Since synchronize_rcu() can block, it cannot be called from
6. Since synchronize_rcu() can block, it cannot be called from
any sort of irq context. The same rule applies for
any sort of irq context. The same rule applies for
...
@@ -178,11 +194,14 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -178,11 +194,14 @@ over a rather long period of time, but improvements are always welcome!
synchronize_sched_expedite(), and synchronize_srcu_expedited().
synchronize_sched_expedite(), and synchronize_srcu_expedited().
The expedited forms of these primitives have the same semantics
The expedited forms of these primitives have the same semantics
as the non-expedited forms, but expediting is both expensive
as the non-expedited forms, but expediting is both expensive and
and unfriendly to real-time workloads. Use of the expedited
(with the exception of synchronize_srcu_expedited()) unfriendly
primitives should be restricted to rare configuration-change
to real-time workloads. Use of the expedited primitives should
operations that would not normally be undertaken while a real-time
be restricted to rare configuration-change operations that would
workload is running.
not normally be undertaken while a real-time workload is running.
However, real-time workloads can use rcupdate.rcu_normal kernel
boot parameter to completely disable expedited grace periods,
though this might have performance implications.
In particular, if you find yourself invoking one of the expedited
In particular, if you find yourself invoking one of the expedited
primitives repeatedly in a loop, please do everyone a favor:
primitives repeatedly in a loop, please do everyone a favor:
...
@@ -193,11 +212,6 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -193,11 +212,6 @@ over a rather long period of time, but improvements are always welcome!
of the system, especially to real-time workloads running on
of the system, especially to real-time workloads running on
the rest of the system.
the rest of the system.
In addition, it is illegal to call the expedited forms from
a CPU-hotplug notifier, or while holding a lock that is acquired
by a CPU-hotplug notifier. Failing to observe this restriction
will result in deadlock.
7. If the updater uses call_rcu() or synchronize_rcu(), then the
7. If the updater uses call_rcu() or synchronize_rcu(), then the
corresponding readers must use rcu_read_lock() and
corresponding readers must use rcu_read_lock() and
rcu_read_unlock(). If the updater uses call_rcu_bh() or
rcu_read_unlock(). If the updater uses call_rcu_bh() or
...
@@ -321,7 +335,7 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -321,7 +335,7 @@ over a rather long period of time, but improvements are always welcome!
Similarly, disabling preemption is not an acceptable substitute
Similarly, disabling preemption is not an acceptable substitute
for rcu_read_lock(). Code that attempts to use preemption
for rcu_read_lock(). Code that attempts to use preemption
disabling where it should be using rcu_read_lock() will break
disabling where it should be using rcu_read_lock() will break
in
real-time
kernel builds.
in
CONFIG_PREEMPT=y
kernel builds.
If you want to wait for interrupt handlers, NMI handlers, and
If you want to wait for interrupt handlers, NMI handlers, and
code under the influence of preempt_disable(), you instead
code under the influence of preempt_disable(), you instead
...
@@ -356,23 +370,22 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -356,23 +370,22 @@ over a rather long period of time, but improvements are always welcome!
not the case, a self-spawning RCU callback would prevent the
not the case, a self-spawning RCU callback would prevent the
victim CPU from ever going offline.)
victim CPU from ever going offline.)
14. SRCU (srcu_read_lock(), srcu_read_unlock(), srcu_dereference(),
14. Unlike other forms of RCU, it -is- permissible to block in an
synchronize_srcu(), synchronize_srcu_expedited(), and call_srcu())
SRCU read-side critical section (demarked by srcu_read_lock()
may only be invoked from process context. Unlike other forms of
and srcu_read_unlock()), hence the "SRCU": "sleepable RCU".
RCU, it -is- permissible to block in an SRCU read-side critical
Please note that if you don't need to sleep in read-side critical
section (demarked by srcu_read_lock() and srcu_read_unlock()),
sections, you should be using RCU rather than SRCU, because RCU
hence the "SRCU": "sleepable RCU". Please note that if you
is almost always faster and easier to use than is SRCU.
don't need to sleep in read-side critical sections, you should be
using RCU rather than SRCU, because RCU is almost always faster
Also unlike other forms of RCU, explicit initialization and
and easier to use than is SRCU.
cleanup is required either at build time via DEFINE_SRCU()
or DEFINE_STATIC_SRCU() or at runtime via init_srcu_struct()
Also unlike other forms of RCU, explicit initialization
and cleanup_srcu_struct(). These last two are passed a
and cleanup is required via init_srcu_struct() and
"struct srcu_struct" that defines the scope of a given
cleanup_srcu_struct(). These are passed a "struct srcu_struct"
SRCU domain. Once initialized, the srcu_struct is passed
that defines the scope of a given SRCU domain. Once initialized,
to srcu_read_lock(), srcu_read_unlock() synchronize_srcu(),
the srcu_struct is passed to srcu_read_lock(), srcu_read_unlock()
synchronize_srcu_expedited(), and call_srcu(). A given
synchronize_srcu(), synchronize_srcu_expedited(), and call_srcu().
synchronize_srcu() waits only for SRCU read-side critical
A given synchronize_srcu() waits only for SRCU read-side critical
sections governed by srcu_read_lock() and srcu_read_unlock()
sections governed by srcu_read_lock() and srcu_read_unlock()
calls that have been passed the same srcu_struct. This property
calls that have been passed the same srcu_struct. This property
is what makes sleeping read-side critical sections tolerable --
is what makes sleeping read-side critical sections tolerable --
...
@@ -390,10 +403,16 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -390,10 +403,16 @@ over a rather long period of time, but improvements are always welcome!
Therefore, SRCU should be used in preference to rw_semaphore
Therefore, SRCU should be used in preference to rw_semaphore
only in extremely read-intensive situations, or in situations
only in extremely read-intensive situations, or in situations
requiring SRCU's read-side deadlock immunity or low read-side
requiring SRCU's read-side deadlock immunity or low read-side
realtime latency.
realtime latency. You should also consider percpu_rw_semaphore
when you need lightweight readers.
Note that, rcu_assign_pointer() relates to SRCU just as it does
SRCU's expedited primitive (synchronize_srcu_expedited())
to other forms of RCU.
never sends IPIs to other CPUs, so it is easier on
real-time workloads than is synchronize_rcu_expedited(),
synchronize_rcu_bh_expedited() or synchronize_sched_expedited().
Note that rcu_dereference() and rcu_assign_pointer() relate to
SRCU just as they do to other forms of RCU.
15. The whole point of call_rcu(), synchronize_rcu(), and friends
15. The whole point of call_rcu(), synchronize_rcu(), and friends
is to wait until all pre-existing readers have finished before
is to wait until all pre-existing readers have finished before
...
@@ -435,3 +454,33 @@ over a rather long period of time, but improvements are always welcome!
...
@@ -435,3 +454,33 @@ over a rather long period of time, but improvements are always welcome!
These debugging aids can help you find problems that are
These debugging aids can help you find problems that are
otherwise extremely difficult to spot.
otherwise extremely difficult to spot.
18. If you register a callback using call_rcu(), call_rcu_bh(),
call_rcu_sched(), or call_srcu(), and pass in a function defined
within a loadable module, then it in necessary to wait for
all pending callbacks to be invoked after the last invocation
and before unloading that module. Note that it is absolutely
-not- sufficient to wait for a grace period! The current (say)
synchronize_rcu() implementation waits only for all previous
callbacks registered on the CPU that synchronize_rcu() is running
on, but it is -not- guaranteed to wait for callbacks registered
on other CPUs.
You instead need to use one of the barrier functions:
o call_rcu() -> rcu_barrier()
o call_rcu_bh() -> rcu_barrier_bh()
o call_rcu_sched() -> rcu_barrier_sched()
o call_srcu() -> srcu_barrier()
However, these barrier functions are absolutely -not- guaranteed
to wait for a grace period. In fact, if there are no call_rcu()
callbacks waiting anywhere in the system, rcu_barrier() is within
its rights to return immediately.
So if you need to wait for both an RCU grace period and for
all pre-existing call_rcu() callbacks, you will need to execute
both rcu_barrier() and synchronize_rcu(), if necessary, using
something like workqueues to to execute them concurrently.
See rcubarrier.txt for more information.
Documentation/RCU/rcu.txt
View file @
4de5f89e
...
@@ -76,15 +76,12 @@ o I hear that RCU is patented? What is with that?
...
@@ -76,15 +76,12 @@ o I hear that RCU is patented? What is with that?
Of these, one was allowed to lapse by the assignee, and the
Of these, one was allowed to lapse by the assignee, and the
others have been contributed to the Linux kernel under GPL.
others have been contributed to the Linux kernel under GPL.
There are now also LGPL implementations of user-level RCU
There are now also LGPL implementations of user-level RCU
available (http://l
ttng.org/?q=node/18
).
available (http://l
iburcu.org/
).
o I hear that RCU needs work in order to support realtime kernels?
o I hear that RCU needs work in order to support realtime kernels?
This work is largely completed. Realtime-friendly RCU can be
Realtime-friendly RCU can be enabled via the CONFIG_PREEMPT_RCU
enabled via the CONFIG_PREEMPT_RCU kernel configuration
kernel configuration parameter.
parameter. However, work is in progress for enabling priority
boosting of preempted RCU read-side critical sections. This is
needed if you have CPU-bound realtime threads.
o Where can I find more information on RCU?
o Where can I find more information on RCU?
...
...
Documentation/RCU/rcubarrier.txt
View file @
4de5f89e
...
@@ -263,6 +263,11 @@ Quick Quiz #2: What happens if CPU 0's rcu_barrier_func() executes
...
@@ -263,6 +263,11 @@ Quick Quiz #2: What happens if CPU 0's rcu_barrier_func() executes
are delayed for a full grace period? Couldn't this result in
are delayed for a full grace period? Couldn't this result in
rcu_barrier() returning prematurely?
rcu_barrier() returning prematurely?
The current rcu_barrier() implementation is more complex, due to the need
to avoid disturbing idle CPUs (especially on battery-powered systems)
and the need to minimally disturb non-idle CPUs in real-time systems.
However, the code above illustrates the concepts.
rcu_barrier() Summary
rcu_barrier() Summary
...
...
Documentation/RCU/torture.txt
View file @
4de5f89e
...
@@ -276,15 +276,17 @@ o "Free-Block Circulation": Shows the number of torture structures
...
@@ -276,15 +276,17 @@ o "Free-Block Circulation": Shows the number of torture structures
somehow gets incremented farther than it should.
somehow gets incremented farther than it should.
Different implementations of RCU can provide implementation-specific
Different implementations of RCU can provide implementation-specific
additional information. For example, SRCU provides the following
additional information. For example,
Tree
SRCU provides the following
additional line:
additional line:
srcu
-torture: per-CPU(idx=1): 0(0,1) 1(0,1) 2(0,0) 3(0,1
)
srcu
d-torture: Tree SRCU per-CPU(idx=0): 0(35,-21) 1(-4,24) 2(1,1) 3(-26,20) 4(28,-47) 5(-9,4) 6(-10,14) 7(-14,11) T(1,6
)
This line shows the per-CPU counter state. The numbers in parentheses are
This line shows the per-CPU counter state, in this case for Tree SRCU
the values of the "old" and "current" counters for the corresponding CPU.
using a dynamically allocated srcu_struct (hence "srcud-" rather than
The "idx" value maps the "old" and "current" values to the underlying
"srcu-"). The numbers in parentheses are the values of the "old" and
array, and is useful for debugging.
"current" counters for the corresponding CPU. The "idx" value maps the
"old" and "current" values to the underlying array, and is useful for
debugging. The final "T" entry contains the totals of the counters.
USAGE
USAGE
...
@@ -304,3 +306,9 @@ checked for such errors. The "rmmod" command forces a "SUCCESS",
...
@@ -304,3 +306,9 @@ checked for such errors. The "rmmod" command forces a "SUCCESS",
"FAILURE", or "RCU_HOTPLUG" indication to be printk()ed. The first
"FAILURE", or "RCU_HOTPLUG" indication to be printk()ed. The first
two are self-explanatory, while the last indicates that while there
two are self-explanatory, while the last indicates that while there
were no RCU failures, CPU-hotplug problems were detected.
were no RCU failures, CPU-hotplug problems were detected.
However, the tools/testing/selftests/rcutorture/bin/kvm.sh script
provides better automation, including automatic failure analysis.
It assumes a qemu/kvm-enabled platform, and runs guest OSes out of initrd.
See tools/testing/selftests/rcutorture/doc/initrd.txt for instructions
on setting up such an initrd.
Documentation/RCU/whatisRCU.txt
View file @
4de5f89e
...
@@ -890,6 +890,8 @@ SRCU: Critical sections Grace period Barrier
...
@@ -890,6 +890,8 @@ SRCU: Critical sections Grace period Barrier
srcu_read_lock_held
srcu_read_lock_held
SRCU: Initialization/cleanup
SRCU: Initialization/cleanup
DEFINE_SRCU
DEFINE_STATIC_SRCU
init_srcu_struct
init_srcu_struct
cleanup_srcu_struct
cleanup_srcu_struct
...
@@ -913,7 +915,8 @@ a. Will readers need to block? If so, you need SRCU.
...
@@ -913,7 +915,8 @@ a. Will readers need to block? If so, you need SRCU.
b. What about the -rt patchset? If readers would need to block
b. What about the -rt patchset? If readers would need to block
in an non-rt kernel, you need SRCU. If readers would block
in an non-rt kernel, you need SRCU. If readers would block
in a -rt kernel, but not in a non-rt kernel, SRCU is not
in a -rt kernel, but not in a non-rt kernel, SRCU is not
necessary.
necessary. (The -rt patchset turns spinlocks into sleeplocks,
hence this distinction.)
c. Do you need to treat NMI handlers, hardirq handlers,
c. Do you need to treat NMI handlers, hardirq handlers,
and code segments with preemption disabled (whether
and code segments with preemption disabled (whether
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment