Concurrency Primer
Concurrency Primer
Concurrency Primer
Matt Kline
May 4, 2018
Abstract
Seasoned programmers are familiar with tools like mutexes, semaphores, and condition
variables. But what makes them work? How do we write concurrent code when we can’t use
them, like when we’re working below the operating system in an embedded environment, or
when we can’t block due to hard time constraints? And since your system transforms your
code into things you didn’t write, running in orders you never asked for, how do multithreaded
programs work at all? Concurrency—especially on modern hardware—is a complicated and
unintuitive topic, but let’s try to cover some fundamentals.
Contents
1. Background . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 2
2. Enforcing law and order . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3
3. Atomicity . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3
4. Arbitrarily-sized “atomic” types . . . . . . . . . . . . . . . . . . . . . . . . . 3
5. Read-modify-write . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
5.1. Exchange . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
5.2. Test and set . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
5.3. Fetch and… . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
5.4. Compare and swap . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
6. Atomic operations as building blocks . . . . . . . . . . . . . . . . . . . . . . . 5
7. Sequential consistency on weakly-ordered hardware . . . . . . . . . . . . . . . 5
8. Implementing atomic read-modify-write operations with LL/SC instructions . . 6
8.1. Spurious LL/SC failures . . . . . . . . . . . . . . . . . . . . . . . . . . 6
9. Do we always need sequentially consistent operations? . . . . . . . . . . . . . 7
10. Memory orderings . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
10.1. Acquire and release . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
10.2. Relaxed . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
10.3. Acquire-Release . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
10.4. Consume . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
10.5. hc svnt dracones . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
11. Hardware convergence . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
12. Cache effects and false sharing . . . . . . . . . . . . . . . . . . . . . . . . . . 10
13. If concurrency is the question, volatile is not the answer. . . . . . . . . . . . 10
14. Atomic fusion . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
15. Takeaways . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
Additional Resources . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
Contributing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
Colophon . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
1
1. Background improve locality.† Variables can be assigned to the same mem-
ory location if they’re never used in overlapping time frames.
Modern computers run several instruction sequences con- Instructions can be executed speculatively, before a branch is
currently. We call them different names depending on the taken, then undone if the compiler guessed incorrectly.‡
context—processes, threads, tasks, interrupt service routines,
Even if we used a compiler that didn’t reorder our code,
and so on—but many of the same principles apply across the
we’d still be in trouble, since our hardware does it too! Mod-
board. On single-core machines, these sequences take turns,
ern cpu designs handle incoming instructions in a much more
sharing the cpu in short slices of time. On multiprocessors,
complicated fashion than traditional pipelined approaches like
several can run in parallel, each on its own core.
the one shown in Figure 1. They contain multiple data paths,
While computer scientists have invented many useful ab-
each for different types of instructions, and schedulers which
stractions, these instruction streams (let’s call them threads
reorder and route instructions through these paths.
from here on out for the sake of brevity) ultimately interact
with one another by sharing bits of state. For this to work, we
must be able to reason about the order of the reads and writes
communicating threads make to memory. Consider a simple
example where thread A shares an integer with others. It writes
the value to some variable, then sets a flag to instruct other
threads to read whatever it just stored. As code, this might Figure 1: A traditional five-stage cpu pipeline with fetch, decode, execute,
resemble: memory access, and write-back stages. Modern designs are
much more complicated, often reordering instructions on the fly.
int v; Image courtesy of Wikipedia.
bool v_ready = false;
It’s also easy to make naïve assumptions about how mem-
void threadA() ory works. If we imagine a multiprocessor, we might think of
{ something resembling Figure 2, where each core takes turns
// Write the value performing reads and writes to the system’s memory.
// and set its ready flag.
v = 42;
v_ready = true;
}
void threadB()
{
// Await a value change and read it.
while (!v_ready) { /* spin */ }
const int my_v = v; Figure 2: An idealized multiprocessor where cores take sequential turns
// Do something with my_v... accessing a single shared set of memory.
}
This is almost never the case. While processor speeds have in-
Our system must guarantee that other threads observe A’s creased exponentially over the past decades, ram hasn’t been
write to v_ready only after A’s write to v. (If another thread able to keep up, creating an ever-widening gulf between the
can “see” v_ready change before it sees v change, our com- time it takes to run an instruction and the time needed to re-
munication scheme can’t work.) trieve data from main memory. Hardware manufacturers have
This appears to be an incredibly simple guarantee to pro- compensated by placing an increasing number of hierarchical
vide, but nothing is as it seems. For starters, any compiler caches directly on the cpu die. Each core also usually has a
worth its salt will happily modify and reorder your code to take store buffer that handles pending writes while subsequent in-
better advantage of the hardware it runs on. So long as the re- structions are executed. Keeping this memory system coherent,
sulting instructions run to the same effect for the current thread, so that writes made by one core are observable by others, even
reads and writes can be moved to avoid pipeline stalls* or to if those cores use different caches, is quite challenging.
*Most cpu designs execute parts of several instructions in parallel to increase their throughput (see Figure 1). When the result of an instruction is needed
by another instruction in the pipeline, the cpu may need to suspend forward progress, or stall, until that result is ready.
† ram is not read in single bytes, but in chunks called cache lines. If variables that are used together can be placed on the same cache line, they will be read
and written all at once. This usually provides a massive speedup, but as we’ll see in §12, can bite us when a line must be shared between cores.
‡ This is especially common when using profile-guided optimization.
2
types in <stdatomic.h> or <atomic>, respectively. They
look and act just like the integer types they mirror (e.g.,
bool → atomic_bool, int → atomic_int, etc.), but
the compiler ensures that other loads and stores aren’t
reordered around their reads and writes. By using an
atomic Boolean, v = 42 is now guaranteed to happen before
v_ready = true in thread A, just as my_v = v must occur
after reading v_ready in thread B.
Formally, these types provide a single total modification or-
der such that, “[…] the result of any execution is the same as if
the reads and writes occurred in some order, and the operations
of each individual processor appear in this sequence in the
Figure 3: A common memory hierarchy for modern multiprocessors order specified by its program.” This model, defined by Leslie
Lamport in 1979, is called sequential consistency. Informally,
The net effect of these complications is that there is no the important takeaway is that sequentially consistent reads
consistent concept of “now” in a multithreaded program, espe- and writes act as rendezvous points for threads. By ensuring
cially one running on a multiprocessor. Attaining some sense that other operations cannot move “past” them, we know that
of order so that threads can communicate is a team effort of anything thread A did before writing to an atomic variable—
hardware manufacturers, compiler writers, language design- such as assigning 42 to v before writing to v_ready—can be
ers, and application developers. Let’s explore what we can do, observed by another thread that reads the atomic variable.
and what tools we will need.
3. Atomicity
2. Enforcing law and order
Our focus so far on ordering sidestepped the other vital ingre-
Creating order in our programs requires a different approach dient for inter-thread communication: atomicity. Something is
on each cpu architecture. Until alarmingly recently, systems atomic if it cannot be divided into smaller parts. To see why
languages like C and C++ offered no help here, so developers reads and writes must have this quality to share data between
needed assembly for safe inter-thread communication. Fortu- threads, let’s see what problems we might encounter if they
nately, the 2011 iso standards of both languages introduced did not.
tools for the task. So long as the programmer uses them cor- Consider a program with two threads. One thread pro-
rectly, the compiler will prevent reorderings—both by the op- cesses some list of files and increments a counter each time
timizer, and by hardware—that cause data races.* it finishes working on one of them. The other thread handles
Let’s return to our example from before. For it to work the user interface, and will periodically read the counter to
as-desired, we need to use an atomic type for the “ready” flag: update a progress bar. If that counter is a 64-bit integer, we
int v = 0; have a problem on 32-bit machines, since two loads or stores
std::atomic_bool v_ready(false); are needed to read or write the entire value. If we’re having
a particularly unlucky time, the first thread could be halfway
void threadA() through writing the counter when the second thread reads it,
{ receiving an incorrect value. These unfortunate occasions are
v = 42; called torn reads and writes.
v_ready = true; If reads and writes to shared data are atomic, however, our
} problem disappears. We can also see that, compared to the
difficulties of establishing order, ensuring atomicity is fairly
straightforward: make sure that variables used for thread syn-
void threadB() chronization are no larger than the architecture’s word size.
{
while (!v_ready) { /* spin */ }
const int my_v = v; 4. Arbitrarily-sized “atomic” types
// Do something with my_v...
} Along with atomic_int and friends, C++ provides the tem-
plate std::atomic<T> for defining arbitrary atomic types.
The C and C++ standard libraries define a series of these C, lacking a similar language feature but wanting to provide
*The ISO C11 standard lifted its concurrency facilities, almost verbatim, from the C++11 standard. Everything you see here should be identical in both
languages, barring some arguably cleaner syntax in C++.
3
the same functionality, added an _Atomic keyword. Run- offer a type dedicated to this purpose, called atomic_flag.
ning counter to what we just discussed, any type can be made We could use it to build a spinlock:
“atomic”, even if it is larger than the target architecture’s word
size. In these cases, the compiler and the language runtime std::atomic_flag af;
library automatically surround the variable’s reads and writes
with locks. For situations where this is unacceptable,* you can void lock()
add an assertion: {
while (af.test_and_set()) { /* spin */ }
std::atomic<Foo> bar;
ASSERT(bar.is_lock_free()); }
5. Read-modify-write
Loads and stores are all well and good, but sometimes we need 5.3. Fetch and…
to read a value, modify it, and write it back in a single atomic
step. There are a few common read-modify-write (rmw) op- We can also read a value, perform some basic mathematical
erations. In C++, they’re represented as member functions of operation on it (addition, subtraction, bitwise and, or, xor),
std::atomic<T>. In C, they’re freestanding functions. and return its previous value. You might have noticed that
in our exchange example, the worker thread’s additions must
5.1. Exchange also be atomic, or else we could run into a race where:
The simplest atomic rmw operation is an exchange: the cur-
rent value is read and replaced with a new one. To see where 1. The worker thread loads the current counter value and
this might be useful, let’s tweak our example from §3. Instead adds one.
of displaying the total number of processed files, we might
want to show how many were processed each second. To do
so, we’ll have the ui thread zero the counter each time it is
2. Before that thread can store the value back, the ui thread
read. Even if these reads and writes are atomic, we could still
zeroes the counter.
run into the following race condition:
1. The ui thread reads the counter.
3. The worker now performs its store, as if the counter was
2. Before the ui thread has the chance to zero it, the worker never cleared.
thread increments it again.
3. The ui thread now zeroes the counter, and the previous
increment is lost.
5.4. Compare and swap
If the ui thread exchanges the current value of the counter
with zero atomically, the race disappears.
Finally, we have compare-and-swap (cas), sometimes called
compare-and-exchange. It allows us to conditionally exchange a
5.2. Test and set
value if its previous value matches some expected one. In C
Test-and-set works on a Boolean value: we read it, set it to and C++, cas resembles the following, if it were all executed
true, and provide the value it held beforehand. C and C++ atomically:
*…which are quite often, since we’re often using atomic operations to avoid locks in the first place.
† The language standards permit atomic types to be sometimes lock-free. This might be necessary for architectures that don’t guarantee atomicity for
unaligned reads and writes.
4
template <typename T> Blocking synchronization methods are usually simpler to
bool atomic<T>::compare_exchange_strong( reason about, but they can make threads pause for arbitrary
T& expected, T desired) amounts of time. For example, consider a mutex, which en-
{ sures that threads take turns holding exclusive access to shared
if (*this == expected) { data. If some thread locks the mutex and another attempts to
*this = desired; do the same, the second thread must wait—or block—until
return true; the first thread releases the lock, however long that may be.
} Blocking mechanisms are also susceptible to deadlock and live-
else { lock—situations where the entire system “gets stuck” due to
expected = *this; threads waiting for one another.
return false; In contrast, lockless approaches ensure that the system is
} always making forward progress. They are non-blocking since
} no thread can cause another to wait indefinitely. Consider an
You might be perplexed by the _strong suffix. Is there a audio streaming program, or an embedded system where a sen-
“weak” cas? Yes, but hold onto that thought—we’ll talk sor triggers an interrupt service routine (isr) when new data
about it in §8.1. arrives. We need lock-free algorithms and data structures to
Let’s say we have some long-running piece of work that communicate between threads in these systems, since block-
we might want to cancel from a ui thread. We’ll give it three ing for arbitrary amounts of time would break them. (In the
states: idle, running, and cancelled, and write a loop that exits first case, the user’s audio would begin to stutter if sound data
when it is cancelled. isn’t provided at the bitrate it is consumed. In the second,
subsequent sensor inputs could be missed if the isr does not
enum class TaskState : int8_t { complete as quickly as possible.)
Idle, Running, Cancelled It’s important to point out that lockless algorithms are not
}; somehow better or faster than lock-based ones. They are just
different tools designed to serve different needs. We should
std::atomic<TaskState> ts; also note that algorithms aren’t automatically lock-free if they
only use atomic operations. Our primitive spinlock from §5.2
void taskLoop() is still a blocking algorithm even though it doesn’t use any os-
{ provided syscalls to put the blocked thread to sleep.*
ts = TaskState::Running; Of course, there are situations where blocking and non-
while (ts == TaskState::Running) {
blocking approaches could both work.† If performance is a
// Do good work.
concern, profile! How well a given synchronization method
}
performs depends on a number of factors, ranging from the
}
number of threads at play to the specifics of your cpu hard-
If we only want to set ts to Cancelled when it’s currently ware. And as always, consider the tradeoffs you make between
Running, but do nothing if it’s already Idle, we could cas: complexity and performance—lockless programming is a per-
ilous art.
bool cancel()
{
auto expected = TaskState::Running; 7. Sequential consistency on weakly-ordered hardware
return ts.compare_exchange_strong(
expected, TaskState::Cancelled); As mentioned in §2, different hardware architectures provide
} different ordering guarantees, or memory models. For exam-
ple, x64 is relatively strongly-ordered, and can be trusted to
preserve some system-wide order of loads and stores in most
6. Atomic operations as building blocks cases. Other architectures like arm are more weakly-ordered,
so one shouldn’t assume that loads and stores are executed in
Atomic loads, stores, and rmw operations are the building program order unless the cpu is given special instructions—
blocks for all the concurrency tools we use. It’s useful to split called memory barriers—to not shuffle them around.
those tools into two camps: blocking and lockless. It’s helpful to look at how atomic operations work in a
*Putting a blocked thread to sleep is often an optimization, since the operating system’s scheduler can run other threads on the cpu until the sleeping one
is unblocked. Some concurrency libraries even offer hybrid locks which spin briefly, then sleep. (This avoids the cost of context switching away from the
current thread if it is blocked for less than the spin length, but avoids wasting cpu time in a long-running loop.)
† You may also hear of wait-free algorithms—they are a subset of lock-free ones which are guaranteed to complete in some bounded number of steps.
5
weakly-ordered system, both to better understand what’s hap- void incFoo() { ++foo; }
pening in hardware, and to see why the C and C++ models were
designed as they were.* Let’s examine arm, since it’s straight- compiles to:
forward and widely-used. Consider the simplest atomic opera- incFoo:
tions: loads and stores. Given some atomic_int foo, ldr r3, <&foo>
dmb
loop:
getFoo: ldrex r2, [r3] // LL foo
int getFoo() ldr r3, <&foo> adds r2, r2, #1 // Increment
{ dmb strex r1, r2, [r3] // SC
becomes
return foo; ldr r0, [r3, #0] cmp r1, #0 // Check the SC result.
} dmb bne loop // Loop if the SC failed.
bx lr dmb
bx lr
6
must emit nested loops: an inner one to protect us from spuri- 10. Memory orderings
ous sc failures, and an outer one which repeatedly loads and
multiplies foo until no other thread has modified it. With By default, all atomic operations—including loads, stores, and
compare_exchange_weak, the compiler is free to generate a the various flavors of rmw—are sequentially consistent. But
single loop instead, since we don’t care about the difference be- this is only one of several orderings that we can give them.
tween spurious failures and “normal” ones caused by another We’ll examine each of them in turn, but a full list, along with
thread modifying foo. the enumerations that the C and C++ api uses, is:
All of our examples so far have used sequentially consistent • Release (memory_order_release)
reads and writes to prevent memory accesses from being re-
arranged in ways that break our code. We’ve also seen how • Relaxed (memory_order_relaxed)
weakly-ordered architectures like arm use a pair of memory
barriers to provide this guarantee. As you might expect, these • Acquire-Release (memory_order_acq_rel)
barriers can have a non-trivial impact on performance. After
all, they inhibit optimizations that your compiler and hardware • Consume (memory_order_consume)
would otherwise make.
What if we could avoid some of this slowdown? Consider To specify one of these orderings, you provide it as an optional
some simple case like the spinlock from §5.2. Between the argument that we’ve slyly failed to mention so far:*
lock() and unlock() calls, we have a critical section where
void lock()
we can safely modify shared state protected by the lock. Out-
{
side this critical section, we only read and write to things that
while (af.test_and_set(
aren’t shared with other threads.
memory_order_acquire)) { /* spin */ }
deepThought.calculate(); // non-shared }
It’s vital that reads and writes to the shared memory we’re int i = foo.load(memory_order_acquire);
protecting don’t move outside the critical section. But the op-
posite isn’t true—the compiler and hardware could move as Compare-and-swap operations are a bit odd in that they have
much as they desire into the critical section without causing two orderings: one for when the cas succeeds, and one for
any trouble. We have no problem with the following if it is when it fails:
somehow faster:
while (!foo.compare_exchange_weak(
lock(); // Lock; critical section begins expected, expected * by,
deepThought.calculate(); // non-shared memory_order_seq_cst, // On success
sharedState.subject = memory_order_relaxed)) // On failure
"Life, the universe and everything"; { /* empty loop */ }
sharedState.answer = 42;
demolishEarth(vogons); // non-shared With the syntax out of the way, let’s look at what these
unlock(); // Unlock; critical section ends orderings are and how we can use them. As it turns out, al-
most all of the examples we’ve seen so far don’t actually need
So, how do we tell the compiler as much? sequentially consistent operations.
*C, being C, defines separate functions for cases where you want to specify an ordering. exchange() becomes exchange_explicit(), a cas becomes
compare_exchange_strong_explicit(), and so on.
7
10.1. Acquire and release 10.2. Relaxed
int acquireFoo()
{
return foo.load(memory_order_acquire);
}
void releaseFoo(int i)
{
foo.store(i, memory_order_release);
Figure 4: Relaxed atomic operations circa 1946
}
become: This might seem like a rare occurrence, but is surprisingly com-
mon. Recall our examples from §3 and §5 where some worker
thread increments a counter which is read by a ui thread
acquireFoo: releaseFoo: to show progress. That counter could be incremented with
ldr r3, <&foo> ldr r3, <&foo> atomic_fetch_add() using memory_order_relaxed. All
ldr r0, [r3, #0] dmb we need is atomicity—nothing is synchronized by the counter.
dmb str r0, [r3, #0] Relaxed reads and writes are also useful for sharing flags
bx lr bx lr between threads. Consider some thread that loops until told
to exit:
8
void atomicMultiply(int by) The difference between acq_rel and seq_cst is gen-
{ erally whether the operation is required to participate
int expected = foo.load(memory_order_relaxed); in the single global order of sequentially consistent op-
erations.
while (!foo.compare_exchange_weak( In other words, acquire-release provides order relative to the
expected, expected * by, variable being load-acquired and store-released, whereas se-
memory_order_release, quentially consistent operation provides some global order
memory_order_relaxed)) across the entire program. If the distinction still seems hazy,
{ /* empty loop */ } you’re not alone. Boehm continues with,
}
This has subtle and unintuitive effects. The [barriers]
All of the loads can be relaxed, as we don’t need to enforce any in the current standard may be the most experts-only
sort of ordering until we’ve successfully modified our value. construct we have in the language.
The initial load of expected isn’t even strictly necessary—it
just saves us a loop iteration if no other thread modifies foo 10.4. Consume
before the cas.
Last but not least, we have memory_order_consume. Con-
sider a scenario where data is rarely changed, but frequently
10.3. Acquire-Release read by multiple threads. Perhaps it is a pointer in the kernel to
memory_order_acq_rel is used with atomic rmw opera- information about peripherals plugged into the machine. This
tions that need to both load-acquire and store-release a value. data will change very infrequently, so it makes sense to opti-
A typical example involves thread-safe reference counting, like mize reads as much as possible. Given what we know so far,
in C++’s shared_ptr: the best we can do is:
std::atomic<PeripheralData*> peripherals;
atomic_int refCount;
// Writers:
void inc() PeripheralData* p = kAllocate(sizeof(*p));
{ populateWithNewDeviceData(p);
refCount.fetch_add(1, memory_order_relaxed); peripherals.store(p, memory_order_release);
}
// Readers:
PeripheralData* p =
void dec()
peripherals.load(memory_order_acquire);
{
if (p != nullptr) {
if (refCount.fetch_sub(1,
doSomethingWith(p->keyboards);
memory_order_acq_rel) == 1) {
}
// No more references, delete the data.
Since we want to optimize readers as much as possible,
}
it would be quite nice if we could avoid a memory barrier
}
on weakly-ordered systems. As it turns out, we usually can.
Order doesn’t matter when incrementing the reference Since the data we examine (p->keyboards) is dependent on
count since no action is taken as a result. However, when we the value of p, most platforms—even weakly-ordered ones—
decrement, we must ensure that: cannot reorder the initial load (p = peripherals) to take
place after its use (p->keyboards).† So long as we convince
1. All reads and writes to the referenced object occur before the compiler not to make any similar speculations, we’re in the
the count reaches zero. clear. This is what memory_order_consume is for. Change
2. Deletion occurs after the reference count drops to zero.* readers to:
PeripheralData* p =
Curious readers might be wondering about the difference peripherals.load(memory_order_consume);
between acquire-release and sequentially consistent operations. if (p != nullptr) {
To quote Hans Boehm, chair of the ISO C++ Concurrency doSomethingWith(p->keyboards);
Study Group, }
*This can be optimized even further by making the acquire barrier only occur conditionally, when the reference count is zero. Standalone barriers are
outside the scope of this paper, since they’re almost always pessimal compared to a combined load-acquire or store-release, but you can see an example
here: http://www.boost.org/doc/libs/release/doc/html/atomic/usage_examples.html.
† Much to everybody’s chagrin, this isn’t the case on some extremely weakly-ordered architectures like DEC Alpha.
9
and an arm compiler could emit: 12. Cache effects and false sharing
As if all of this wasn’t enough to keep rattling around in your
ldr r3, &peripherals head, modern hardware gives us one more wrinkle. Recall that
ldr r3, [r3] memory is transferred between main ram and the cpu in
// Look ma, no barrier! chunks called cache lines. These lines are also the smallest
cbz r3, was_null // Check for null unit transferred between cores and their respective caches—if
ldr r0, [r3, #4] // Load p->keyboards one core writes a value and another core reads it, the entire line
b doSomethingWith(Keyboards*) containing that value must be transferred from the first core’s
was_null: cache(s) to the second core’s in order to keep their “view” of
... memory coherent.
This can have a surprising performance impact. Consider a
Sadly, the emphasis here is on could. Figuring out what readers-writer lock, which avoids races by ensuring that shared
constitutes a “dependency” between expressions isn’t as trivial data has one writer or any number of readers, but never both
as one might hope,* so all compilers currently convert consume at the same time. At its core, it resembles the following:
operations to acquires. struct RWLock {
int readers;
bool hasWriter; // Zero or one writers
10.5. hc svnt dracones };
Non-sequentially consistent orderings have many subtleties, Writers must block until readers reaches zero, but readers
and a slight mistake can cause elusive Heisenbugs that only oc- can take the lock with an atomic rmw operation whenever
cur sometimes, on some platforms. Before reaching for them, hasWriter is false.
ask yourself: Naïvely, it seems like this offers a huge performance win
over exclusive locks (e.g., mutexes, spinlocks, etc.) for cases
Am I using a well-known and understood pattern where we read the shared data more often than we write, but
(such as the ones shown above)? this assumption fails to consider the effects of cache coher-
ence. If multiple readers—each running on a different core—
Are the operations in a tight loop? simultaneously take the lock, the cache line containing it will
“ping-pong” between those cores’ caches. Unless critical sec-
tions are very large, resolving this contention will likely take
Does every microsecond count here?
more time than the critical sections themselves,† even though
If the answer isn’t yes for at least one of these, default to se- no blocking is required by the algorithm.
quentially consistent operations. Otherwise, be sure to give This slowdown is even more insidious when it occurs be-
your code extra review and testing. tween unrelated variables that happen to be placed on the same
cache line. When one designs concurrent data structures or
algorithms, this false sharing must be taken into account. One
way to avoid it is to pad atomic variables with a cache line of
11. Hardware convergence
unshared data, but this is obviously a large space-time tradeoff.
Those familiar with the platform may have noticed that all
arm assembly shown here is from the seventh version of the 13. If concurrency is the question, volatile is not
architecture. Excitingly, the current (eighth) generation offers the answer.
a massive improvement for lockless code. Since most pro-
gramming languages have converged on the memory model Before we go, we should lay a common misconception sur-
we’ve been exploring, armv8 processors offer dedicated load- rounding the volatile keyword to rest. Perhaps because
acquire and store-release instructions, lda and stl. We can of how it worked in older compilers and hardware, or due to
use them to implement everything we’ve discussed here with- its different meaning in languages like Java and C#,‡ some be-
out resorting to memory barriers. Hopefully, future cpu ar- lieve that the keyword is useful for building concurrency tools.
chitectures will follow suit. Except for one specific case (see §14), this is false.
*Even the experts in the iso committee’s concurrency study group, sg1, came away with different understandings. See n4036 for the gory details.
Proposed solutions are explored in p0190r3 and p0462r1.
† On some systems, a cache miss can cost more than two orders of magnitude than an atomic rmw operation. See Paul E. McKenney’s talk from
CppCon 2017 for more details.
‡ Unlike in C and C++, volatile does enforce ordering in those languages.
10
The purpose of volatile is to inform the compiler that a Since relaxed loads provide no ordering guarantees, the com-
value can be changed by something besides the program we’re piler is free to unroll the loop as much as it pleases, perhaps
executing. This is useful for memory-mapped i/o (mmio), into:
where the system hardware translates reads and writes to cer-
tain addresses into instructions for the devices connected to while (tmp = foo.load(memory_order_relaxed)) {
the cpu. (This is how most machines ultimately interact with doSomething(tmp);
the outside world.) This implies two guarantees: doSomething(tmp);
doSomething(tmp);
1. The compiler will not elide what it otherwise sees as doSomething(tmp);
“unnecessary” loads and stores. For example, if I had }
some function:
In some cases, “fusing” reads or writes like this is unacceptable,
void write(int* t)
so we must prevent it with volatile casts or incantations
{
like asm volatile("" ::: "memory").* The Linux kernel
*t = 2;
provides READ_ONCE() and WRITE_ONCE() macros for this
*t = 42;
exact purpose.†
}
15. Takeaways
the compiler would normally optimize it to:
void write(int* t) { *t = 42; } We’ve only scratched the surface here, but hopefully you now
know:
*t = 2 is usually assumed to be a dead store that does
nothing. But, if t points to some mmio register, it’s not • Why compilers and cpu hardware reorder loads and
safe to make this assumption—each write could have stores.
some effect on the hardware it’s interacting with.
• Why we need special tools to prevent these reorderings
2. The compiler will not reorder volatile reads and to communicate between threads.
writes with respect to other volatile ones for simi-
• How we can guarantee sequential consistency in our pro-
lar reasons.
grams.
These rules don’t give us the atomicity or order we need • Atomic read-modify-write operations.
for safe inter-thread communication. Notice that the second
guarantee only prevents volatile operations from being re- • How atomic operations can be implemented on weakly-
ordered in relation to each other—the compiler is still free to ordered hardware, and what implications this can have
rearrange all other “normal” loads and stores around them. for a language-level api.
And even if we set that problem aside, volatile does not
emit memory barriers on weakly-ordered hardware. The key- • How we can carefully optimize lockless code using alter-
word only works as a synchronization mechanism if both your native memory orderings.
compiler and your hardware perform no reordering. Don’t bet • How false sharing can impact the performance of con-
on that. current memory access.
Finally, one should realize that while atomic operations do pre- • How to prevent the compiler from fusing atomic opera-
vent certain optimizations, they aren’t somehow immune to all tions in undesirable ways.
of them. The optimizer can do fairly mundane things, such as
replacing foo.fetch_and(0) with foo = 0, but it can also To learn more, see the additional resources below, or exam-
produce surprising results. Consider: ine lock-free data structures and algorithms, such as a single-
producer/single-consumer (sp/sc) queue or read-copy-update
while (tmp = foo.load(memory_order_relaxed)) { (rcu).‡
doSomething(tmp);
} Good luck and godspeed!
*See https://stackoverflow.com/a/14983432.
† See n4374 and the kernel’s compiler.h for details.
‡ See the Linux Weekly News article, What is RCU, Fundamentally? for an introduction.
11
Additional Resources
C++ atomics, from basic to advanced. What do they really do? by Fedor Pikus, a hour-long talk
on this topic.
atomic<> Weapons: The C++11 Memory Model and Modern Hardware by Herb Sutter, a
three-hour talk that provides a deeper dive. Also the source of figures 2 and 3.
Futexes are Tricky, a paper by Ulrich Drepper on how mutexes and other synchronization
primitives can be built in Linux using atomic operations and syscalls.
Is Parallel Programming Hard,And, If So, What Can You Do About It?, by Paul E. McKenney, an
incredibly comprehensive book covering parallel data structures and algorithms, transactional
memory, cache coherence protocols, cpu architecture specifics, and more.
Memory Barriers: a Hardware View for Software Hackers, an older but much shorter piece by
McKenney explaining how memory barriers are implemented in the Linux kernel on various
architectures.
No Sane Compiler Would Optimize Atomics, a discussion of how atomic operations are handled
by current optimizers. Available as a writeup, n4455, and as a CppCon talk.
cppreference.com, an excellent reference for the C and C++ memory model and atomic api.
Matt Godbolt’s Compiler Explorer, an online tool that provides live, color-coded disassembly
using compilers and flags of your choosing. Fantastic for examining what compilers emit for
various atomic operations on different architectures.
Contributing
Contributions are welcome! Sources and history are available on Gitlab and Github. This
paper is prepared in LATEX—if you’re not familiar with it, feel free to contact the author (via
email, by opening an issue, etc.) in lieu of pull requests.
This paper is licensed under a Creative Commons Attribution-ShareAlike 4.0 International Li-
cense. The legalese can be found through https://creativecommons.org/licenses/
by-sa/4.0/, but in short, you are free to copy, redistribute, translate, or otherwise trans-
form this paper so long as you give appropriate credit, indicate if changes were made, and
release your version under this same license.
Colophon
This guide was typeset using LuaLATEX in Matthew Butterick’s Equity, with code in Matthias
Tellen’s mononoki. The title is set in Neue Haas Grotesk, a Helvetica restoration by
Christian Schwartz.
12