Upgrade to Pro — share decks privately, control downloads, hide ads and more …

ITT 2019 - Kavya Joshi - Let's talk locks!

ITT 2019 - Kavya Joshi - Let's talk locks!

Locks have a bad rap for “being slow” and yet, they’re used extensively in applications and under-the-hood. So, what gives? This talk resolves the dichotomy. We’ll explore when and why locks affect performance, delve into Go’s lock implementation as a case-study, and discuss strategies we can use when locks are actually a problem.

Istanbul Tech Talks

April 02, 2019
Tweet

More Decks by Istanbul Tech Talks

Other Decks in Technology

Transcript

  1. “locks are slow” …but they’re used everywhere. from schedulers to

    databases and web servers. lock contention causes ~10x latency latency (ms) time
  2. “locks are slow” …but they’re used everywhere. from schedulers to

    databases and web servers. lock contention causes ~10x latency latency (ms) time ?
  3. let’s analyze its performance! performance models for contention let’s build

    a lock! a tour through lock internals let’s use it, smartly! a few closing strategies
  4. our case-study Lock implementations are hardware, ISA, OS and language

    specific:
 
 We assume an x86_64 SMP machine running a modern Linux.
 We’ll look at the lock implementation in Go 1.12. CPU 0 CPU 1 cache cache interconnect memory simplified SMP system diagram
  5. use as you would threads 
 > go handle_request(r)
 but

    user-space threads:
 managed entirely by the Go runtime, not the operating system. The unit of concurrent execution: goroutines. a brief go primer
  6. use as you would threads 
 > go handle_request(r)
 but

    user-space threads:
 managed entirely by the Go runtime, not the operating system. The unit of concurrent execution: goroutines. a brief go primer Data shared between goroutines must be synchronized. One way is to use the blocking, non-recursive lock construct: > var mu sync.Mutex
 mu.Lock()
 … mu.Unlock()
  7. T1 running on CPU 1 T2 running on CPU 2

    func reader() {
 // Read a task
 t := tasks.get()
 
 // Do something with it.
 ... } func writer() {
 // Write to tasks
 tasks.put(t) } // track whether tasks is // available (0) or not (1) // shared ring buffer var tasks Tasks want: “mutual exclusion” only one thread has access to shared data at any given time
  8. func reader() {
 // Read a task
 t := tasks.get()


    
 // Do something with it.
 ... } func writer() {
 // Write to tasks
 tasks.put(t) } // track whether tasks is // available (0) or not (1) // shared ring buffer var tasks Tasks want: “mutual exclusion” …idea! use a flag? T1 running on CPU 1 T2 running on CPU 2
  9. // track whether tasks can be // accessed (0) or

    not (1) var flag int var tasks Tasks
  10. // track whether tasks can be // accessed (0) or

    not (1) var flag int var tasks Tasks func reader() {
 for { /* If flag is 0, can access tasks. */
 if flag == 0 {
 /* Set flag */ flag++ ...
 /* Unset flag */ flag-- return } /* Else, keep looping. */ 
 } } T1 running on CPU 1
  11. // track whether tasks can be // accessed (0) or

    not (1) var flag int var tasks Tasks func reader() {
 for { /* If flag is 0, can access tasks. */
 if flag == 0 {
 /* Set flag */ flag++ ...
 /* Unset flag */ flag-- return } /* Else, keep looping. */ 
 } } func writer() {
 for { /* If flag is 0, can access tasks. */
 if flag == 0 {
 /* Set flag */ flag++ ...
 /* Unset flag */ flag-- return } /* Else, keep looping. */ 
 } } T1 running on CPU 1 T2 running on CPU 2
  12. // track whether tasks can be // accessed (0) or

    not (1) var flag int var tasks Tasks func reader() {
 for { /* If flag is 0, can access tasks. */
 if flag == 0 {
 /* Set flag */ flag++ ...
 /* Unset flag */ flag-- return } /* Else, keep looping. */ 
 } } func writer() {
 for { /* If flag is 0, can access tasks. */
 if flag == 0 {
 /* Set flag */ flag++ ...
 /* Unset flag */ flag-- return } /* Else, keep looping. */ 
 } } T1 running on CPU 1 T2 running on CPU 2
  13. R R W flag++ if flag == 0 timeline of

    memory operations T1 running on CPU 1 T2 running on CPU 2 T2 may observe T1 ’s RMW half-complete
  14. atomicity A memory operation is non-atomic if it can be

    observed half-complete by another thread. An operation may be non-atomic because it:
 • uses multiple CPU instructions:
 operations on a large data structure; 
 compiler decisions.
 • use a single non-atomic CPU instruction:
 RMW instructions; unaligned loads and stores. > o := Order { id: 10, name: “yogi bear”, order: “pie”, count: 3, }
  15. atomicity A memory operation is non-atomic if it can be

    observed half-complete by another thread. An operation may be non-atomic because it:
 • uses multiple CPU instructions:
 operations on a large data structure; 
 compiler decisions.
 • uses a single non-atomic CPU instruction:
 RMW instructions; unaligned loads and stores. > flag++
  16. atomicity A memory operation is non-atomic if it can be

    observed half-complete by another thread. An operation may be non-atomic because it:
 • uses multiple CPU instructions:
 operations on a large data structure; 
 compiler decisions.
 • uses a single non-atomic CPU instruction:
 RMW instructions; unaligned loads and stores. > flag++ An atomic operation is an “indivisible” memory access. In x86_64, loads, stores that are 
 naturally aligned up to 64b.* guarantees the data item fits within a cache line;
 cache coherency guarantees a consistent view for a single cache line. * these are not the only guaranteed atomic operations.
  17. func reader() {
 for { /* If flag is 0,

    can access tasks. */
 if flag == 0 {
 /* Set flag */ flag = 1 t := tasks.get()
 ...
 /* Unset flag */ flag = 0 return } /* Else, keep looping. */ 
 } } T1 running on CPU 1
  18. the compiler may reorder operations. // Sets flag to 1

    & reads data. func reader() { flag = 1 t := tasks.get() ... flag = 0
  19. the processor may reorder operations. StoreLoad reordering load t before

    store flag = 1 // Sets flag to 1 & reads data. func reader() { flag = 1 t := tasks.get() ... flag = 0
  20. memory access reordering The compiler, processor can reorder memory operations

    to optimize execution. • The only cardinal rule is sequential consistency for single threaded programs.
 • Other guarantees about compiler reordering are captured by a 
 language’s memory model:
 C++, Go guarantee data-race free programs will be sequentially consistent. • For processor reordering, by the hardware memory model:
 x86_64 provides Total Store Ordering (TSO).
  21. memory access reordering The compiler, processor can reorder memory operations

    to optimize execution. • The only cardinal rule is sequential consistency for single threaded programs.
 • Other guarantees about compiler reordering are captured by a 
 language’s memory model:
 C++, Go guarantee data-race free programs will be sequentially consistent. • For processor reordering, by the hardware memory model:
 x86_64 provides Total Store Ordering (TSO).
  22. memory access reordering The compiler, processor can reorder memory operations

    to optimize execution. • The only cardinal rule is sequential consistency for single threaded programs.
 • Other guarantees about compiler reordering are captured by a 
 language’s memory model:
 C++, Go guarantee data-race free programs will be sequentially consistent. • For processor reordering, by the hardware memory model:
 x86_64 provides Total Store Ordering (TSO). a relaxed consistency model. most reorderings are invalid but StoreLoad is game;
 allows processor to hide the latency of writes.
  23. nope; not atomic and no memory order guarantees. …idea! use

    a flag? need a construct that provides atomicity and prevents memory reordering.
  24. nope; not atomic and no memory order guarantees. …idea! use

    a flag? need a construct that provides atomicity and prevents memory reordering. …the hardware provides!
  25. For guaranteed atomicity and to prevent memory reordering. special hardware

    instructions x86 example: XCHG (exchange) these instructions are called memory barriers. they prevent reordering by the compiler too. x86 example: MFENCE, LFENCE, SFENCE.
  26. special hardware instructions The x86 LOCK instruction prefix provides both.

    Used to prefix memory access instructions: LOCK ADD For guaranteed atomicity and to prevent memory reordering. } atomic operations in languages like Go: atomic.Add atomic.CompareAndSwap
  27. special hardware instructions The x86 LOCK instruction prefix provides both.

    Used to prefix memory access instructions: LOCK ADD For guaranteed atomicity and to prevent memory reordering. } atomic operations in languages like Go: atomic.Add atomic.CompareAndSwap LOCK CMPXCHG Atomic compare-and-swap (CAS) conditionally updates a variable:
 checks if it has the expected value and if so, changes it to the desired value.
  28. the CAS succeeded; we set flag to 1. flag was

    1 so our CAS failed; try again. var flag int var tasks Tasks func reader() { for { // Try to atomically CAS flag from 0 -> 1 if atomic.CompareAndSwap(&flag, 0, 1) { ... // Atomically set flag back to 0. atomic.Store(&flag, 0) return }
 // CAS failed, try again :) } } baby’s first lock
  29. var flag int var tasks Tasks func reader() { for

    { // Try to atomically CAS flag from 0 -> 1 if atomic.CompareAndSwap(&flag, 0, 1) { ... // Atomically set flag back to 0. atomic.Store(&flag, 0) return }
 // CAS failed, try again :) } } baby’s first lock: spinlocks This is a simplified spinlock. Spinlocks are used extensively in the Linux kernel. }
  30. cost of an atomic operation Run on a 12-core x86_64

    SMP machine.
 Atomic store to a C _Atomic int, 10M times in a tight loop. Measure average time taken per operation
 (from within the program). With 1 thread: ~13ns With 12 cpu-pinned threads: ~110ns threads are effectively serialized var flag int var tasks Tasks func reader() { for { // Try to atomically CAS flag from 0 -> 1 if atomic.CompareAndSwap(&flag, 0, 1) { ... // Atomically set flag back to 0. atomic.Store(&flag, 0) return }
 // CAS failed, try again :) } } spinlocks
  31. sweet. We have a scheme for mutual exclusion that provides

    atomicity and memory ordering guarantees.
  32. sweet. …but spinning for long durations is wasteful; it takes

    away CPU time from other threads. We have a scheme for mutual exclusion that provides atomicity and memory ordering guarantees.
  33. sweet. …but spinning for long durations is wasteful; it takes

    away CPU time from other threads. We have a scheme for mutual exclusion that provides atomicity and memory ordering guarantees. enter the operating system!
  34. Linux’s futex Interface and mechanism for userspace code to ask

    the kernel to suspend/ resume threads. futex syscall kernel-managed queue
  35. flag can be 0: unlocked
 1: locked 2: there’s a

    waiter var flag int var tasks Tasks
  36. set flag to 2 (there’s a waiter) flag can be

    0: unlocked
 1: locked 2: there’s a waiter futex syscall to tell the kernel to suspend us until flag changes. when we’re resumed, we’ll CAS again. var flag int var tasks Tasks func reader() { for { if atomic.CompareAndSwap(&flag, 0, 1) { ... }
 // CAS failed, set flag to sleeping. v := atomic.Xchg(&flag, 2) // and go to sleep. futex(&flag, FUTEX_WAIT, ...)
 } } T1 ’s CAS fails
 (because T2 has set the flag) T1
  37. in the kernel: keyA (from the userspace address:
 &flag) keyA

    T1 futex_q 1. arrange for thread to be resumed in the future:
 add an entry for this thread in the kernel queue for the address we care about
  38. in the kernel: keyA (from the userspace address:
 &flag) keyA

    T1 futex_q keyother Tother futex_q keyother hash(keyA ) 1. arrange for thread to be resumed in the future:
 add an entry for this thread in the kernel queue for the address we care about
  39. in the kernel: keyA (from the userspace address:
 &flag) keyA

    T1 futex_q keyother Tother futex_q keyother hash(keyA ) 1. arrange for thread to be resumed in the future:
 add an entry for this thread in the kernel queue for the address we care about 2. deschedule the calling thread to suspend it.
  40. T2 is done
 (accessing the shared data) T2 func writer()

    { for { if atomic.CompareAndSwap(&flag, 0, 1) { ... 
 // Set flag to unlocked. v := atomic.Xchg(&flag, 0) if v == 2 { // If there was a waiter, issue a wake up. futex(&flag, FUTEX_WAKE, ...) } return }
 v := atomic.Xchg(&flag, 2) futex(&flag, FUTEX_WAIT, …) } }
  41. T2 is done
 (accessing the shared data) T2 func writer()

    { for { if atomic.CompareAndSwap(&flag, 0, 1) { ... 
 // Set flag to unlocked. v := atomic.Xchg(&flag, 0) if v == 2 { // If there was a waiter, issue a wake up. futex(&flag, FUTEX_WAKE, ...) } return }
 v := atomic.Xchg(&flag, 2) futex(&flag, FUTEX_WAIT, …) } } if flag was 2, there’s at least one waiter futex syscall to tell the kernel to wake a waiter up.
  42. func writer() { for { if atomic.CompareAndSwap(&flag, 0, 1) {

    ... 
 // Set flag to unlocked. v := atomic.Xchg(&flag, 0) if v == 2 { // If there was a waiter, issue a wake up. futex(&flag, FUTEX_WAKE, ...) } return }
 v := atomic.Xchg(&flag, 2) futex(&flag, FUTEX_WAIT, …) } } if flag was 2, there’s at least one waiter futex syscall to tell the kernel to wake a waiter up. hashes the key walks the hash bucket’s futex queue finds the first thread waiting on the address schedules it to run again! } T2 is done
 (accessing the shared data) T2
  43. pretty convenient! pthread mutexes use futexes. That was a hella

    simplified futex. …but we still have a nice, lightweight primitive to build synchronization constructs.
  44. cost of a futex Run on a 12-core x86_64 SMP

    machine.
 Lock & unlock a pthread mutex 10M times in loop
 (lock, increment an integer, unlock).
 Measure average time taken per lock/unlock pair
 (from within the program). uncontended case (1 thread): ~13ns contended case (12 cpu-pinned threads): ~0.9us
  45. cost of a futex Run on a 12-core x86_64 SMP

    machine.
 Lock & unlock a pthread mutex 10M times in loop
 (lock, increment an integer, unlock).
 Measure average time taken per lock/unlock pair
 (from within the program). uncontended case (1 thread): ~13ns contended case (12 cpu-pinned threads): ~0.9us cost of the user-space atomic CAS = ~13ns } cost of the atomic CAS + syscall + thread context switch = ~0.9us }
  46. spinning vs. sleeping Spinning makes sense for short durations; it

    keeps the thread on the CPU. The trade-off is it uses CPU cycles not making progress. So at some point, it makes sense to pay the cost of the context switch to go to sleep. There are smart “hybrid” futexes:
 CAS-spin a small, fixed number of times —> if that didn’t lock, make the futex syscall. Example: the Go runtime’s futex implementation.
  47. spinning vs. sleeping Spinning makes sense for short durations; it

    keeps the thread on the CPU. The trade-off is it uses CPU cycles not making progress. So at some point, it makes sense to pay the cost of the context switch to go to sleep. There are smart “hybrid” futexes:
 CAS-spin a small, fixed number of times —> if that didn’t lock, make the futex syscall. Examples: the Go runtime’s futex implementation; a variant of the pthread_mutex.
  48. …can we do better for user-space threads? goroutines are user-space

    threads. The go runtime multiplexes them onto threads. lighter-weight and cheaper than threads:
 goroutine switches = ~tens of ns; 
 thread switches = ~a µs. CPU core g1 g6 g2 thread CPU core } OS scheduler Go scheduler }
  49. …can we do better for user-space threads? goroutines are user-space

    threads. The go runtime multiplexes them onto threads. lighter-weight and cheaper than threads:
 goroutine switches = ~tens of ns; 
 thread switches = ~a µs. CPU core g1 g6 g2 thread CPU core } OS scheduler Go scheduler } we can block the goroutine without blocking the underlying thread! to avoid the thread context switch cost.
  50. This is what the Go runtime’s semaphore does!
 The semaphore

    is conceptually very similar to futexes in Linux*, but it is used to 
 sleep/wake goroutines: a goroutine that blocks on a mutex is descheduled, but not the underlying thread. the goroutine wait queues are managed by the runtime, in user-space. * There are, of course, differences in implementation though.
  51. the goroutine wait queues are managed by the Go runtime,

    in user-space. var flag int var tasks Tasks func reader() { for { // Attempt to CAS flag. if atomic.CompareAndSwap(&flag, ...) { ... }
 // CAS failed; add G1 as a waiter for flag. root.queue() // and to sleep. futex(&flag, FUTEX_WAIT, ...) } } G1 ’s CAS fails
 (because G2 has set the flag) G1
  52. &flag (the userspace address) &flag G1 G3 G4 &other hash(&flag)

    } the top-level waitlist for a hash bucket is implemented as a treap } there’s a second-level wait queue 
 for each unique address the goroutine wait queues (in user-space, managed by the go runtime)
  53. the goroutine wait queues are managed by the Go runtime,

    in user-space. var flag int var tasks Tasks func reader() { for { // Attempt to CAS flag. if atomic.CompareAndSwap(&flag, ...) { ... }
 // CAS failed; add G1 as a waiter for flag. root.queue() // and suspend G1. gopark() } } G1 ’s CAS fails
 (because G2 has set the flag) G1 the Go runtime deschedules the goroutine; keeps the thread running!
  54. G2 ’s done
 (accessing the shared data) G2 func writer()

    { for { if atomic.CompareAndSwap(&flag, 0, 1) { ... 
 // Set flag to unlocked. atomic.Xadd(&flag, ...)
 
 // If there’s a waiter, reschedule it. waiter := root.dequeue(&flag) goready(waiter) return }
 root.queue() gopark() } } find the first waiter goroutine and reschedule it ]
  55. this is clever. Avoids the hefty thread context switch cost

    in the contended case,
 up to a point.
  56. this is clever. Avoids the hefty thread context switch cost

    in the contended case,
 up to a point. but…
  57. func reader() { for { if atomic.CompareAndSwap(&flag, ...) { ...

    }
 // CAS failed; add G1 as a waiter for flag. semaroot.queue() // and suspend G1. gopark() } } once G1 is resumed, 
 it will try to CAS again. Resumed goroutines have to compete with any other goroutines trying to CAS.
 
 They will likely lose:
 there’s a delay between when the flag was set to 0 and this goroutine was rescheduled.. G1
  58. Resumed goroutines have to compete with any other goroutines trying

    to CAS.
 
 They will likely lose:
 there’s a delay between when the flag was set to 0 and this goroutine was rescheduled.. // Set flag to unlocked. atomic.Xadd(&flag, …)
 
 // If there’s a waiter, reschedule it. waiter := root.dequeue(&flag) goready(waiter) return
  59. Resumed goroutines have to compete with any other goroutines trying

    to CAS.
 
 They will likely lose:
 there’s a delay between when the flag was set to 0 and this goroutine was rescheduled.. So, the semaphore implementation may end up:
 • unnecessarily resuming a waiter goroutine
 results in a goroutine context switch again.
 • cause goroutine starvation
 can result in long wait times, high tail latencies.
  60. Resumed goroutines have to compete with any other goroutines trying

    to CAS.
 
 They will likely lose:
 there’s a delay between when the flag was set to 0 and this goroutine was rescheduled.. So, the semaphore implementation may end up:
 • unnecessarily resuming a waiter goroutine
 results in a goroutine context switch again.
 • cause goroutine starvation
 can result in long wait times, high tail latencies. the sync.Mutex implementation adds a layer that fixes these.
  61. go’s sync.Mutex Additionally, it tracks extra state to: Is a

    hybrid lock that uses a semaphore to sleep / wake goroutines. prevent unnecessarily waking up a goroutine
 “There’s a goroutine actively trying to CAS”: An unlock in this case does not wake a waiter.
 prevent severe goroutine starvation “a waiter has been waiting”: If a waiter is resumed but loses the CAS again, it’s queued at the head of the wait queue.
 If a waiter fails to lock for 1ms, switch the mutex to “starvation mode”. prevent unnecessarily waking up a goroutine
 “There’s a goroutine actively trying to CAS”: An unlock in this case does not wake a waiter.
 prevent severe goroutine starvation “a waiter has been waiting”: If a waiter is resumed but loses the CAS again, it’s queued at the head of the wait queue.
 If a waiter fails to lock for 1ms, switch the mutex to “starvation mode”.
  62. go’s sync.Mutex Additionally, it tracks extra state to: Is a

    hybrid lock that uses a semaphore to sleep / wake goroutines. prevent unnecessarily waking up a goroutine
 “There’s a goroutine actively trying to CAS”: An unlock in this case does not wake a waiter.
 prevent severe goroutine starvation “a waiter has been waiting”: If a waiter is resumed but loses the CAS again, it’s queued at the head of the wait queue.
 If a waiter fails to lock for 1ms, switch the mutex to “starvation mode”. other goroutines cannot CAS, they must queue The unlock hands the mutex off to the first waiter.
 i.e. the waiter does not have to compete.
  63. how does it perform? Run on a 12-core x86_64 SMP

    machine.
 Lock & unlock a Go sync.Mutex 10M times in loop
 (lock, increment an integer, unlock).
 Measure average time taken per lock/unlock pair
 (from within the program). uncontended case (1 goroutine): ~13ns contended case (12 goroutines): ~0.8us
  64. how does it perform? Contended case performance of C vs.

    Go:
 Go initially performs better than C
 but they ~converge as concurrency gets high enough.
  65. how does it perform? Contended case performance of C vs.

    Go:
 Go initially performs better than C
 but they ~converge as concurrency gets high enough.
  66. &flag G1 G3 G4 &other the Go runtime semaphore’s hash

    table for waiting goroutines: each hash bucket needs a lock. …and it’s a futex!
  67. &flag G1 G3 G4 &other the Go runtime semaphore’s hash

    table for waiting goroutines: each hash bucket needs a lock. …it’s a futex!
  68. &flag G1 G3 G4 &other &flag G1 the Linux kernel’s

    futex hash table for waiting threads: each hash bucket needs a lock. …it’s a spin lock! each hash bucket needs a lock. …it’s a futex! the Go runtime semaphore’s hash table for waiting goroutines:
  69. &flag G1 G3 G4 &other &flag G1 each hash bucket

    needs a lock. …it’s a spinlock! each hash bucket needs a lock. …it’s a futex! the Go runtime semaphore’s hash table for waiting goroutines: the Linux kernel’s futex hash table for waiting threads:
  70. uncontended case
 Cost of the atomic CAS. contended case In

    the worst-case, cost of failed atomic operations + spinning + goroutine context switch + 
 thread context switch. ….But really, depends on degree of contention.
  71. how many threads do we need to support a target

    throughput? 
 while keeping response time the same. how does response time change with the number of threads? assuming a constant workload. “How does application performance change with concurrency?”
  72. Amdahl’s Law Speed-up depends on the fraction of the workload

    that can be parallelized (p). speed-up with N threads = 1 (1 — p) + p N
  73. a simple experiment Measure time taken to complete a fixed

    workload.
 serial fraction holds a lock (sync.Mutex). scale parallel fraction (p) from 0.25 to 0.75 measure time taken for number of goroutines (N) = 1 —> 12.
  74. p = 0.75 p = 0.25 Amdahl’s Law Speed-up depends

    on the fraction of the workload that can be parallelized (p).
  75. Universal Scalability Law (USL) • contention penalty
 due to serialization

    for shared resources.
 examples: lock contention, database contention.
 • crosstalk penalty
 due to coordination for coherence. examples: servers coordinating to synchronize
 mutable state. αN Scalability depends on contention and cross-talk.
  76. Universal Scalability Law (USL) • contention penalty
 due to serialization

    for shared resources.
 examples: lock contention, database contention.
 • crosstalk penalty
 due to coordination for coherence. examples: servers coordinating to synchronize
 mutable state. αN Scalability depends on contention and cross-talk. βN2
  77. Universal Scalability Law (USL) N (αN + βN2 + C)

    N C N (αN + C) contention and crosstalk linear scaling contention throughput concurrency throughput of N threads = N (αN + βN2 + C)
  78. p = 0.75 p = 0.25 USL curves plotted using

    the R usl package p = parallel fraction of workload
  79. but first, profile! • Go mutex contention profiler • perf-lock:


    perf examples by Brendan Gregg
 Brendan Gregg article on off-cpu analysis • Dtrace / systemtap • mutrace • Valgrind-drd
  80. strategy I: don’t use a lock • remove the need

    for synchronization from
 hot-paths • use atomic operations • use lock-free data structures:
 example: Go scheduler’s per-CPU runqueues. scheduler benchmark (CreateGoroutineParallel) uses a global lock
  81. strategy II: granular locks • shard data:
 but ensure no

    false sharing, by padding to cache line size.
 example: the go runtime semaphore’s hash table buckets.
 • per-processor locks:
 the Linux scheduler’s per-CPU runqueues.
  82. strategy III: do less serial work lock contention causes ~10x

    latency latency time time smaller critical section change • move computation out of critical section:
  83. Special thanks to Eben Freeman, Justin Delegard, Austin Duffield for

    reading drafts of this. @kavya719 speakerdeck.com/kavya719/lets-talk-locks References
 Jeff Preshing’s excellent blog series
 LWN.net on futexes
 The Go source code The Universal Scalability Law Manifesto, Neil Gunther