xref: /freebsd/contrib/llvm-project/libcxx/include/atomic (revision 6132212808e8dccedc9e5d85fea4390c2f38059a)
1// -*- C++ -*-
2//===--------------------------- atomic -----------------------------------===//
3//
4// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5// See https://llvm.org/LICENSE.txt for license information.
6// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7//
8//===----------------------------------------------------------------------===//
9
10#ifndef _LIBCPP_ATOMIC
11#define _LIBCPP_ATOMIC
12
13/*
14    atomic synopsis
15
16namespace std
17{
18
19// feature test macro
20
21#define __cpp_lib_atomic_is_always_lock_free // as specified by SG10
22
23 // order and consistency
24
25 enum memory_order: unspecified // enum class in C++20
26 {
27    relaxed,
28    consume, // load-consume
29    acquire, // load-acquire
30    release, // store-release
31    acq_rel, // store-release load-acquire
32    seq_cst // store-release load-acquire
33 };
34
35 inline constexpr auto memory_order_relaxed = memory_order::relaxed;
36 inline constexpr auto memory_order_consume = memory_order::consume;
37 inline constexpr auto memory_order_acquire = memory_order::acquire;
38 inline constexpr auto memory_order_release = memory_order::release;
39 inline constexpr auto memory_order_acq_rel = memory_order::acq_rel;
40 inline constexpr auto memory_order_seq_cst = memory_order::seq_cst;
41
42template <class T> T kill_dependency(T y) noexcept;
43
44// lock-free property
45
46#define ATOMIC_BOOL_LOCK_FREE unspecified
47#define ATOMIC_CHAR_LOCK_FREE unspecified
48#define ATOMIC_CHAR16_T_LOCK_FREE unspecified
49#define ATOMIC_CHAR32_T_LOCK_FREE unspecified
50#define ATOMIC_WCHAR_T_LOCK_FREE unspecified
51#define ATOMIC_SHORT_LOCK_FREE unspecified
52#define ATOMIC_INT_LOCK_FREE unspecified
53#define ATOMIC_LONG_LOCK_FREE unspecified
54#define ATOMIC_LLONG_LOCK_FREE unspecified
55#define ATOMIC_POINTER_LOCK_FREE unspecified
56
57template <class T>
58struct atomic
59{
60    using value_type = T;
61
62    static constexpr bool is_always_lock_free;
63    bool is_lock_free() const volatile noexcept;
64    bool is_lock_free() const noexcept;
65
66    atomic() noexcept = default;
67    constexpr atomic(T desr) noexcept;
68    atomic(const atomic&) = delete;
69    atomic& operator=(const atomic&) = delete;
70    atomic& operator=(const atomic&) volatile = delete;
71
72    T load(memory_order m = memory_order_seq_cst) const volatile noexcept;
73    T load(memory_order m = memory_order_seq_cst) const noexcept;
74    operator T() const volatile noexcept;
75    operator T() const noexcept;
76    void store(T desr, memory_order m = memory_order_seq_cst) volatile noexcept;
77    void store(T desr, memory_order m = memory_order_seq_cst) noexcept;
78    T operator=(T) volatile noexcept;
79    T operator=(T) noexcept;
80
81    T exchange(T desr, memory_order m = memory_order_seq_cst) volatile noexcept;
82    T exchange(T desr, memory_order m = memory_order_seq_cst) noexcept;
83    bool compare_exchange_weak(T& expc, T desr,
84                               memory_order s, memory_order f) volatile noexcept;
85    bool compare_exchange_weak(T& expc, T desr, memory_order s, memory_order f) noexcept;
86    bool compare_exchange_strong(T& expc, T desr,
87                                 memory_order s, memory_order f) volatile noexcept;
88    bool compare_exchange_strong(T& expc, T desr,
89                                 memory_order s, memory_order f) noexcept;
90    bool compare_exchange_weak(T& expc, T desr,
91                               memory_order m = memory_order_seq_cst) volatile noexcept;
92    bool compare_exchange_weak(T& expc, T desr,
93                               memory_order m = memory_order_seq_cst) noexcept;
94    bool compare_exchange_strong(T& expc, T desr,
95                                memory_order m = memory_order_seq_cst) volatile noexcept;
96    bool compare_exchange_strong(T& expc, T desr,
97                                 memory_order m = memory_order_seq_cst) noexcept;
98
99    void wait(T, memory_order = memory_order::seq_cst) const volatile noexcept;
100    void wait(T, memory_order = memory_order::seq_cst) const noexcept;
101    void notify_one() volatile noexcept;
102    void notify_one() noexcept;
103    void notify_all() volatile noexcept;
104    void notify_all() noexcept;
105};
106
107template <>
108struct atomic<integral>
109{
110    using value_type = integral;
111
112    static constexpr bool is_always_lock_free;
113    bool is_lock_free() const volatile noexcept;
114    bool is_lock_free() const noexcept;
115
116    atomic() noexcept = default;
117    constexpr atomic(integral desr) noexcept;
118    atomic(const atomic&) = delete;
119    atomic& operator=(const atomic&) = delete;
120    atomic& operator=(const atomic&) volatile = delete;
121
122    integral load(memory_order m = memory_order_seq_cst) const volatile noexcept;
123    integral load(memory_order m = memory_order_seq_cst) const noexcept;
124    operator integral() const volatile noexcept;
125    operator integral() const noexcept;
126    void store(integral desr, memory_order m = memory_order_seq_cst) volatile noexcept;
127    void store(integral desr, memory_order m = memory_order_seq_cst) noexcept;
128    integral operator=(integral desr) volatile noexcept;
129    integral operator=(integral desr) noexcept;
130
131    integral exchange(integral desr,
132                      memory_order m = memory_order_seq_cst) volatile noexcept;
133    integral exchange(integral desr, memory_order m = memory_order_seq_cst) noexcept;
134    bool compare_exchange_weak(integral& expc, integral desr,
135                               memory_order s, memory_order f) volatile noexcept;
136    bool compare_exchange_weak(integral& expc, integral desr,
137                               memory_order s, memory_order f) noexcept;
138    bool compare_exchange_strong(integral& expc, integral desr,
139                                 memory_order s, memory_order f) volatile noexcept;
140    bool compare_exchange_strong(integral& expc, integral desr,
141                                 memory_order s, memory_order f) noexcept;
142    bool compare_exchange_weak(integral& expc, integral desr,
143                               memory_order m = memory_order_seq_cst) volatile noexcept;
144    bool compare_exchange_weak(integral& expc, integral desr,
145                               memory_order m = memory_order_seq_cst) noexcept;
146    bool compare_exchange_strong(integral& expc, integral desr,
147                                memory_order m = memory_order_seq_cst) volatile noexcept;
148    bool compare_exchange_strong(integral& expc, integral desr,
149                                 memory_order m = memory_order_seq_cst) noexcept;
150
151    integral fetch_add(integral op, memory_order m = memory_order_seq_cst) volatile noexcept;
152    integral fetch_add(integral op, memory_order m = memory_order_seq_cst) noexcept;
153    integral fetch_sub(integral op, memory_order m = memory_order_seq_cst) volatile noexcept;
154    integral fetch_sub(integral op, memory_order m = memory_order_seq_cst) noexcept;
155    integral fetch_and(integral op, memory_order m = memory_order_seq_cst) volatile noexcept;
156    integral fetch_and(integral op, memory_order m = memory_order_seq_cst) noexcept;
157    integral fetch_or(integral op, memory_order m = memory_order_seq_cst) volatile noexcept;
158    integral fetch_or(integral op, memory_order m = memory_order_seq_cst) noexcept;
159    integral fetch_xor(integral op, memory_order m = memory_order_seq_cst) volatile noexcept;
160    integral fetch_xor(integral op, memory_order m = memory_order_seq_cst) noexcept;
161
162    integral operator++(int) volatile noexcept;
163    integral operator++(int) noexcept;
164    integral operator--(int) volatile noexcept;
165    integral operator--(int) noexcept;
166    integral operator++() volatile noexcept;
167    integral operator++() noexcept;
168    integral operator--() volatile noexcept;
169    integral operator--() noexcept;
170    integral operator+=(integral op) volatile noexcept;
171    integral operator+=(integral op) noexcept;
172    integral operator-=(integral op) volatile noexcept;
173    integral operator-=(integral op) noexcept;
174    integral operator&=(integral op) volatile noexcept;
175    integral operator&=(integral op) noexcept;
176    integral operator|=(integral op) volatile noexcept;
177    integral operator|=(integral op) noexcept;
178    integral operator^=(integral op) volatile noexcept;
179    integral operator^=(integral op) noexcept;
180
181    void wait(integral, memory_order = memory_order::seq_cst) const volatile noexcept;
182    void wait(integral, memory_order = memory_order::seq_cst) const noexcept;
183    void notify_one() volatile noexcept;
184    void notify_one() noexcept;
185    void notify_all() volatile noexcept;
186    void notify_all() noexcept;
187};
188
189template <class T>
190struct atomic<T*>
191{
192    using value_type = T*;
193
194    static constexpr bool is_always_lock_free;
195    bool is_lock_free() const volatile noexcept;
196    bool is_lock_free() const noexcept;
197
198    atomic() noexcept = default;
199    constexpr atomic(T* desr) noexcept;
200    atomic(const atomic&) = delete;
201    atomic& operator=(const atomic&) = delete;
202    atomic& operator=(const atomic&) volatile = delete;
203
204    T* load(memory_order m = memory_order_seq_cst) const volatile noexcept;
205    T* load(memory_order m = memory_order_seq_cst) const noexcept;
206    operator T*() const volatile noexcept;
207    operator T*() const noexcept;
208    void store(T* desr, memory_order m = memory_order_seq_cst) volatile noexcept;
209    void store(T* desr, memory_order m = memory_order_seq_cst) noexcept;
210    T* operator=(T*) volatile noexcept;
211    T* operator=(T*) noexcept;
212
213    T* exchange(T* desr, memory_order m = memory_order_seq_cst) volatile noexcept;
214    T* exchange(T* desr, memory_order m = memory_order_seq_cst) noexcept;
215    bool compare_exchange_weak(T*& expc, T* desr,
216                               memory_order s, memory_order f) volatile noexcept;
217    bool compare_exchange_weak(T*& expc, T* desr,
218                               memory_order s, memory_order f) noexcept;
219    bool compare_exchange_strong(T*& expc, T* desr,
220                                 memory_order s, memory_order f) volatile noexcept;
221    bool compare_exchange_strong(T*& expc, T* desr,
222                                 memory_order s, memory_order f) noexcept;
223    bool compare_exchange_weak(T*& expc, T* desr,
224                               memory_order m = memory_order_seq_cst) volatile noexcept;
225    bool compare_exchange_weak(T*& expc, T* desr,
226                               memory_order m = memory_order_seq_cst) noexcept;
227    bool compare_exchange_strong(T*& expc, T* desr,
228                                memory_order m = memory_order_seq_cst) volatile noexcept;
229    bool compare_exchange_strong(T*& expc, T* desr,
230                                 memory_order m = memory_order_seq_cst) noexcept;
231    T* fetch_add(ptrdiff_t op, memory_order m = memory_order_seq_cst) volatile noexcept;
232    T* fetch_add(ptrdiff_t op, memory_order m = memory_order_seq_cst) noexcept;
233    T* fetch_sub(ptrdiff_t op, memory_order m = memory_order_seq_cst) volatile noexcept;
234    T* fetch_sub(ptrdiff_t op, memory_order m = memory_order_seq_cst) noexcept;
235
236    T* operator++(int) volatile noexcept;
237    T* operator++(int) noexcept;
238    T* operator--(int) volatile noexcept;
239    T* operator--(int) noexcept;
240    T* operator++() volatile noexcept;
241    T* operator++() noexcept;
242    T* operator--() volatile noexcept;
243    T* operator--() noexcept;
244    T* operator+=(ptrdiff_t op) volatile noexcept;
245    T* operator+=(ptrdiff_t op) noexcept;
246    T* operator-=(ptrdiff_t op) volatile noexcept;
247    T* operator-=(ptrdiff_t op) noexcept;
248
249    void wait(T*, memory_order = memory_order::seq_cst) const volatile noexcept;
250    void wait(T*, memory_order = memory_order::seq_cst) const noexcept;
251    void notify_one() volatile noexcept;
252    void notify_one() noexcept;
253    void notify_all() volatile noexcept;
254    void notify_all() noexcept;
255};
256
257
258template <class T>
259  bool atomic_is_lock_free(const volatile atomic<T>* obj) noexcept;
260
261template <class T>
262  bool atomic_is_lock_free(const atomic<T>* obj) noexcept;
263
264template <class T>
265  void atomic_store(volatile atomic<T>* obj, T desr) noexcept;
266
267template <class T>
268  void atomic_store(atomic<T>* obj, T desr) noexcept;
269
270template <class T>
271  void atomic_store_explicit(volatile atomic<T>* obj, T desr, memory_order m) noexcept;
272
273template <class T>
274  void atomic_store_explicit(atomic<T>* obj, T desr, memory_order m) noexcept;
275
276template <class T>
277  T atomic_load(const volatile atomic<T>* obj) noexcept;
278
279template <class T>
280  T atomic_load(const atomic<T>* obj) noexcept;
281
282template <class T>
283  T atomic_load_explicit(const volatile atomic<T>* obj, memory_order m) noexcept;
284
285template <class T>
286  T atomic_load_explicit(const atomic<T>* obj, memory_order m) noexcept;
287
288template <class T>
289  T atomic_exchange(volatile atomic<T>* obj, T desr) noexcept;
290
291template <class T>
292  T atomic_exchange(atomic<T>* obj, T desr) noexcept;
293
294template <class T>
295  T atomic_exchange_explicit(volatile atomic<T>* obj, T desr, memory_order m) noexcept;
296
297template <class T>
298  T atomic_exchange_explicit(atomic<T>* obj, T desr, memory_order m) noexcept;
299
300template <class T>
301  bool atomic_compare_exchange_weak(volatile atomic<T>* obj, T* expc, T desr) noexcept;
302
303template <class T>
304  bool atomic_compare_exchange_weak(atomic<T>* obj, T* expc, T desr) noexcept;
305
306template <class T>
307  bool atomic_compare_exchange_strong(volatile atomic<T>* obj, T* expc, T desr) noexcept;
308
309template <class T>
310  bool atomic_compare_exchange_strong(atomic<T>* obj, T* expc, T desr) noexcept;
311
312template <class T>
313  bool atomic_compare_exchange_weak_explicit(volatile atomic<T>* obj, T* expc,
314                                             T desr,
315                                             memory_order s, memory_order f) noexcept;
316
317template <class T>
318  bool atomic_compare_exchange_weak_explicit(atomic<T>* obj, T* expc, T desr,
319                                             memory_order s, memory_order f) noexcept;
320
321template <class T>
322  bool atomic_compare_exchange_strong_explicit(volatile atomic<T>* obj,
323                                               T* expc, T desr,
324                                               memory_order s, memory_order f) noexcept;
325
326template <class T>
327  bool atomic_compare_exchange_strong_explicit(atomic<T>* obj, T* expc,
328                                               T desr,
329                                               memory_order s, memory_order f) noexcept;
330
331template <class T>
332  void atomic_wait(const volatile atomic<T>* obj, T old) noexcept;
333
334template <class T>
335  void atomic_wait(const atomic<T>* obj, T old) noexcept;
336
337template <class T>
338  void atomic_wait_explicit(const volatile atomic<T>* obj, T old, memory_order m) noexcept;
339
340template <class T>
341  void atomic_wait_explicit(const atomic<T>* obj, T old, memory_order m) noexcept;
342
343template <class T>
344  void atomic_one(volatile atomic<T>* obj) noexcept;
345
346template <class T>
347  void atomic_one(atomic<T>* obj) noexcept;
348
349template <class T>
350  void atomic_all(volatile atomic<T>* obj) noexcept;
351
352template <class T>
353  void atomic_all(atomic<T>* obj) noexcept;
354
355template <class Integral>
356  Integral atomic_fetch_add(volatile atomic<Integral>* obj, Integral op) noexcept;
357
358template <class Integral>
359  Integral atomic_fetch_add(atomic<Integral>* obj, Integral op) noexcept;
360
361template <class Integral>
362  Integral atomic_fetch_add_explicit(volatile atomic<Integral>* obj, Integral op,
363                              memory_order m) noexcept;
364template <class Integral>
365  Integral atomic_fetch_add_explicit(atomic<Integral>* obj, Integral op,
366                              memory_order m) noexcept;
367template <class Integral>
368  Integral atomic_fetch_sub(volatile atomic<Integral>* obj, Integral op) noexcept;
369
370template <class Integral>
371  Integral atomic_fetch_sub(atomic<Integral>* obj, Integral op) noexcept;
372
373template <class Integral>
374  Integral atomic_fetch_sub_explicit(volatile atomic<Integral>* obj, Integral op,
375                                     memory_order m) noexcept;
376
377template <class Integral>
378  Integral atomic_fetch_sub_explicit(atomic<Integral>* obj, Integral op,
379                                     memory_order m) noexcept;
380
381template <class Integral>
382  Integral atomic_fetch_and(volatile atomic<Integral>* obj, Integral op) noexcept;
383
384template <class Integral>
385  Integral atomic_fetch_and(atomic<Integral>* obj, Integral op) noexcept;
386
387template <class Integral>
388  Integral atomic_fetch_and_explicit(volatile atomic<Integral>* obj, Integral op,
389                                     memory_order m) noexcept;
390
391template <class Integral>
392  Integral atomic_fetch_and_explicit(atomic<Integral>* obj, Integral op,
393                                     memory_order m) noexcept;
394
395template <class Integral>
396  Integral atomic_fetch_or(volatile atomic<Integral>* obj, Integral op) noexcept;
397
398template <class Integral>
399  Integral atomic_fetch_or(atomic<Integral>* obj, Integral op) noexcept;
400
401template <class Integral>
402  Integral atomic_fetch_or_explicit(volatile atomic<Integral>* obj, Integral op,
403                             memory_order m) noexcept;
404
405template <class Integral>
406  Integral atomic_fetch_or_explicit(atomic<Integral>* obj, Integral op,
407                             memory_order m) noexcept;
408
409template <class Integral>
410  Integral atomic_fetch_xor(volatile atomic<Integral>* obj, Integral op) noexcept;
411
412template <class Integral>
413  Integral atomic_fetch_xor(atomic<Integral>* obj, Integral op) noexcept;
414
415template <class Integral>
416  Integral atomic_fetch_xor_explicit(volatile atomic<Integral>* obj, Integral op,
417                                     memory_order m) noexcept;
418
419template <class Integral>
420  Integral atomic_fetch_xor_explicit(atomic<Integral>* obj, Integral op,
421                                     memory_order m) noexcept;
422
423template <class T>
424  T* atomic_fetch_add(volatile atomic<T*>* obj, ptrdiff_t op) noexcept;
425
426template <class T>
427  T* atomic_fetch_add(atomic<T*>* obj, ptrdiff_t op) noexcept;
428
429template <class T>
430  T* atomic_fetch_add_explicit(volatile atomic<T*>* obj, ptrdiff_t op,
431                               memory_order m) noexcept;
432
433template <class T>
434  T* atomic_fetch_add_explicit(atomic<T*>* obj, ptrdiff_t op, memory_order m) noexcept;
435
436template <class T>
437  T* atomic_fetch_sub(volatile atomic<T*>* obj, ptrdiff_t op) noexcept;
438
439template <class T>
440  T* atomic_fetch_sub(atomic<T*>* obj, ptrdiff_t op) noexcept;
441
442template <class T>
443  T* atomic_fetch_sub_explicit(volatile atomic<T*>* obj, ptrdiff_t op,
444                               memory_order m) noexcept;
445
446template <class T>
447  T* atomic_fetch_sub_explicit(atomic<T*>* obj, ptrdiff_t op, memory_order m) noexcept;
448
449// Atomics for standard typedef types
450
451typedef atomic<bool>               atomic_bool;
452typedef atomic<char>               atomic_char;
453typedef atomic<signed char>        atomic_schar;
454typedef atomic<unsigned char>      atomic_uchar;
455typedef atomic<short>              atomic_short;
456typedef atomic<unsigned short>     atomic_ushort;
457typedef atomic<int>                atomic_int;
458typedef atomic<unsigned int>       atomic_uint;
459typedef atomic<long>               atomic_long;
460typedef atomic<unsigned long>      atomic_ulong;
461typedef atomic<long long>          atomic_llong;
462typedef atomic<unsigned long long> atomic_ullong;
463typedef atomic<char16_t>           atomic_char16_t;
464typedef atomic<char32_t>           atomic_char32_t;
465typedef atomic<wchar_t>            atomic_wchar_t;
466
467typedef atomic<int_least8_t>   atomic_int_least8_t;
468typedef atomic<uint_least8_t>  atomic_uint_least8_t;
469typedef atomic<int_least16_t>  atomic_int_least16_t;
470typedef atomic<uint_least16_t> atomic_uint_least16_t;
471typedef atomic<int_least32_t>  atomic_int_least32_t;
472typedef atomic<uint_least32_t> atomic_uint_least32_t;
473typedef atomic<int_least64_t>  atomic_int_least64_t;
474typedef atomic<uint_least64_t> atomic_uint_least64_t;
475
476typedef atomic<int_fast8_t>   atomic_int_fast8_t;
477typedef atomic<uint_fast8_t>  atomic_uint_fast8_t;
478typedef atomic<int_fast16_t>  atomic_int_fast16_t;
479typedef atomic<uint_fast16_t> atomic_uint_fast16_t;
480typedef atomic<int_fast32_t>  atomic_int_fast32_t;
481typedef atomic<uint_fast32_t> atomic_uint_fast32_t;
482typedef atomic<int_fast64_t>  atomic_int_fast64_t;
483typedef atomic<uint_fast64_t> atomic_uint_fast64_t;
484
485typedef atomic<int8_t>   atomic_int8_t;
486typedef atomic<uint8_t>  atomic_uint8_t;
487typedef atomic<int16_t>  atomic_int16_t;
488typedef atomic<uint16_t> atomic_uint16_t;
489typedef atomic<int32_t>  atomic_int32_t;
490typedef atomic<uint32_t> atomic_uint32_t;
491typedef atomic<int64_t>  atomic_int64_t;
492typedef atomic<uint64_t> atomic_uint64_t;
493
494typedef atomic<intptr_t>  atomic_intptr_t;
495typedef atomic<uintptr_t> atomic_uintptr_t;
496typedef atomic<size_t>    atomic_size_t;
497typedef atomic<ptrdiff_t> atomic_ptrdiff_t;
498typedef atomic<intmax_t>  atomic_intmax_t;
499typedef atomic<uintmax_t> atomic_uintmax_t;
500
501// flag type and operations
502
503typedef struct atomic_flag
504{
505    atomic_flag() noexcept = default;
506    atomic_flag(const atomic_flag&) = delete;
507    atomic_flag& operator=(const atomic_flag&) = delete;
508    atomic_flag& operator=(const atomic_flag&) volatile = delete;
509
510    bool test(memory_order m = memory_order_seq_cst) volatile noexcept;
511    bool test(memory_order m = memory_order_seq_cst) noexcept;
512    bool test_and_set(memory_order m = memory_order_seq_cst) volatile noexcept;
513    bool test_and_set(memory_order m = memory_order_seq_cst) noexcept;
514    void clear(memory_order m = memory_order_seq_cst) volatile noexcept;
515    void clear(memory_order m = memory_order_seq_cst) noexcept;
516
517    void wait(bool, memory_order = memory_order::seq_cst) const volatile noexcept;
518    void wait(bool, memory_order = memory_order::seq_cst) const noexcept;
519    void notify_one() volatile noexcept;
520    void notify_one() noexcept;
521    void notify_all() volatile noexcept;
522    void notify_all() noexcept;
523} atomic_flag;
524
525bool atomic_flag_test(volatile atomic_flag* obj) noexcept;
526bool atomic_flag_test(atomic_flag* obj) noexcept;
527bool atomic_flag_test_explicit(volatile atomic_flag* obj,
528                               memory_order m) noexcept;
529bool atomic_flag_test_explicit(atomic_flag* obj, memory_order m) noexcept;
530bool atomic_flag_test_and_set(volatile atomic_flag* obj) noexcept;
531bool atomic_flag_test_and_set(atomic_flag* obj) noexcept;
532bool atomic_flag_test_and_set_explicit(volatile atomic_flag* obj,
533                                       memory_order m) noexcept;
534bool atomic_flag_test_and_set_explicit(atomic_flag* obj, memory_order m) noexcept;
535void atomic_flag_clear(volatile atomic_flag* obj) noexcept;
536void atomic_flag_clear(atomic_flag* obj) noexcept;
537void atomic_flag_clear_explicit(volatile atomic_flag* obj, memory_order m) noexcept;
538void atomic_flag_clear_explicit(atomic_flag* obj, memory_order m) noexcept;
539
540void atomic_wait(const volatile atomic_flag* obj, T old) noexcept;
541void atomic_wait(const atomic_flag* obj, T old) noexcept;
542void atomic_wait_explicit(const volatile atomic_flag* obj, T old, memory_order m) noexcept;
543void atomic_wait_explicit(const atomic_flag* obj, T old, memory_order m) noexcept;
544void atomic_one(volatile atomic_flag* obj) noexcept;
545void atomic_one(atomic_flag* obj) noexcept;
546void atomic_all(volatile atomic_flag* obj) noexcept;
547void atomic_all(atomic_flag* obj) noexcept;
548
549// fences
550
551void atomic_thread_fence(memory_order m) noexcept;
552void atomic_signal_fence(memory_order m) noexcept;
553
554// deprecated
555
556template <class T>
557  void atomic_init(volatile atomic<T>* obj, typename atomic<T>::value_type desr) noexcept;
558
559template <class T>
560  void atomic_init(atomic<T>* obj, typename atomic<T>::value_type desr) noexcept;
561
562#define ATOMIC_VAR_INIT(value) see below
563
564#define ATOMIC_FLAG_INIT see below
565
566}  // std
567
568*/
569
570#include <__config>
571#include <__threading_support>
572#include <cstddef>
573#include <cstdint>
574#include <cstring>
575#include <type_traits>
576#include <version>
577
578#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
579#pragma GCC system_header
580#endif
581
582#ifdef _LIBCPP_HAS_NO_THREADS
583# error <atomic> is not supported on this single threaded system
584#endif
585#ifdef _LIBCPP_HAS_NO_ATOMIC_HEADER
586# error <atomic> is not implemented
587#endif
588#ifdef kill_dependency
589# error C++ standard library is incompatible with <stdatomic.h>
590#endif
591
592#define _LIBCPP_CHECK_STORE_MEMORY_ORDER(__m) \
593  _LIBCPP_DIAGNOSE_WARNING(__m == memory_order_consume || \
594                           __m == memory_order_acquire || \
595                           __m == memory_order_acq_rel,   \
596                        "memory order argument to atomic operation is invalid")
597
598#define _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m) \
599  _LIBCPP_DIAGNOSE_WARNING(__m == memory_order_release || \
600                           __m == memory_order_acq_rel,   \
601                        "memory order argument to atomic operation is invalid")
602
603#define _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__m, __f) \
604  _LIBCPP_DIAGNOSE_WARNING(__f == memory_order_release || \
605                           __f == memory_order_acq_rel,   \
606                        "memory order argument to atomic operation is invalid")
607
608_LIBCPP_BEGIN_NAMESPACE_STD
609
610// Figure out what the underlying type for `memory_order` would be if it were
611// declared as an unscoped enum (accounting for -fshort-enums). Use this result
612// to pin the underlying type in C++20.
613enum __legacy_memory_order {
614    __mo_relaxed,
615    __mo_consume,
616    __mo_acquire,
617    __mo_release,
618    __mo_acq_rel,
619    __mo_seq_cst
620};
621
622typedef underlying_type<__legacy_memory_order>::type __memory_order_underlying_t;
623
624#if _LIBCPP_STD_VER > 17
625
626enum class memory_order : __memory_order_underlying_t {
627  relaxed = __mo_relaxed,
628  consume = __mo_consume,
629  acquire = __mo_acquire,
630  release = __mo_release,
631  acq_rel = __mo_acq_rel,
632  seq_cst = __mo_seq_cst
633};
634
635inline constexpr auto memory_order_relaxed = memory_order::relaxed;
636inline constexpr auto memory_order_consume = memory_order::consume;
637inline constexpr auto memory_order_acquire = memory_order::acquire;
638inline constexpr auto memory_order_release = memory_order::release;
639inline constexpr auto memory_order_acq_rel = memory_order::acq_rel;
640inline constexpr auto memory_order_seq_cst = memory_order::seq_cst;
641
642#else
643
644typedef enum memory_order {
645  memory_order_relaxed = __mo_relaxed,
646  memory_order_consume = __mo_consume,
647  memory_order_acquire = __mo_acquire,
648  memory_order_release = __mo_release,
649  memory_order_acq_rel = __mo_acq_rel,
650  memory_order_seq_cst = __mo_seq_cst,
651} memory_order;
652
653#endif // _LIBCPP_STD_VER > 17
654
655template <typename _Tp> _LIBCPP_INLINE_VISIBILITY
656bool __cxx_nonatomic_compare_equal(_Tp const& __lhs, _Tp const& __rhs) {
657    return memcmp(&__lhs, &__rhs, sizeof(_Tp)) == 0;
658}
659
660static_assert((is_same<underlying_type<memory_order>::type, __memory_order_underlying_t>::value),
661  "unexpected underlying type for std::memory_order");
662
663#if defined(_LIBCPP_HAS_GCC_ATOMIC_IMP) || \
664	defined(_LIBCPP_ATOMIC_ONLY_USE_BUILTINS)
665
666// [atomics.types.generic]p1 guarantees _Tp is trivially copyable. Because
667// the default operator= in an object is not volatile, a byte-by-byte copy
668// is required.
669template <typename _Tp, typename _Tv> _LIBCPP_INLINE_VISIBILITY
670typename enable_if<is_assignable<_Tp&, _Tv>::value>::type
671__cxx_atomic_assign_volatile(_Tp& __a_value, _Tv const& __val) {
672  __a_value = __val;
673}
674template <typename _Tp, typename _Tv> _LIBCPP_INLINE_VISIBILITY
675typename enable_if<is_assignable<_Tp&, _Tv>::value>::type
676__cxx_atomic_assign_volatile(_Tp volatile& __a_value, _Tv volatile const& __val) {
677  volatile char* __to = reinterpret_cast<volatile char*>(&__a_value);
678  volatile char* __end = __to + sizeof(_Tp);
679  volatile const char* __from = reinterpret_cast<volatile const char*>(&__val);
680  while (__to != __end)
681    *__to++ = *__from++;
682}
683
684#endif
685
686#if defined(_LIBCPP_HAS_GCC_ATOMIC_IMP)
687
688template <typename _Tp>
689struct __cxx_atomic_base_impl {
690
691  _LIBCPP_INLINE_VISIBILITY
692#ifndef _LIBCPP_CXX03_LANG
693    __cxx_atomic_base_impl() _NOEXCEPT = default;
694#else
695    __cxx_atomic_base_impl() _NOEXCEPT : __a_value() {}
696#endif // _LIBCPP_CXX03_LANG
697  _LIBCPP_CONSTEXPR explicit __cxx_atomic_base_impl(_Tp value) _NOEXCEPT
698    : __a_value(value) {}
699  _Tp __a_value;
700};
701
702_LIBCPP_INLINE_VISIBILITY inline _LIBCPP_CONSTEXPR int __to_gcc_order(memory_order __order) {
703  // Avoid switch statement to make this a constexpr.
704  return __order == memory_order_relaxed ? __ATOMIC_RELAXED:
705         (__order == memory_order_acquire ? __ATOMIC_ACQUIRE:
706          (__order == memory_order_release ? __ATOMIC_RELEASE:
707           (__order == memory_order_seq_cst ? __ATOMIC_SEQ_CST:
708            (__order == memory_order_acq_rel ? __ATOMIC_ACQ_REL:
709              __ATOMIC_CONSUME))));
710}
711
712_LIBCPP_INLINE_VISIBILITY inline _LIBCPP_CONSTEXPR int __to_gcc_failure_order(memory_order __order) {
713  // Avoid switch statement to make this a constexpr.
714  return __order == memory_order_relaxed ? __ATOMIC_RELAXED:
715         (__order == memory_order_acquire ? __ATOMIC_ACQUIRE:
716          (__order == memory_order_release ? __ATOMIC_RELAXED:
717           (__order == memory_order_seq_cst ? __ATOMIC_SEQ_CST:
718            (__order == memory_order_acq_rel ? __ATOMIC_ACQUIRE:
719              __ATOMIC_CONSUME))));
720}
721
722template <typename _Tp>
723_LIBCPP_INLINE_VISIBILITY
724void __cxx_atomic_init(volatile __cxx_atomic_base_impl<_Tp>* __a,  _Tp __val) {
725  __cxx_atomic_assign_volatile(__a->__a_value, __val);
726}
727
728template <typename _Tp>
729_LIBCPP_INLINE_VISIBILITY
730void __cxx_atomic_init(__cxx_atomic_base_impl<_Tp>* __a,  _Tp __val) {
731  __a->__a_value = __val;
732}
733
734_LIBCPP_INLINE_VISIBILITY inline
735void __cxx_atomic_thread_fence(memory_order __order) {
736  __atomic_thread_fence(__to_gcc_order(__order));
737}
738
739_LIBCPP_INLINE_VISIBILITY inline
740void __cxx_atomic_signal_fence(memory_order __order) {
741  __atomic_signal_fence(__to_gcc_order(__order));
742}
743
744template <typename _Tp>
745_LIBCPP_INLINE_VISIBILITY
746void __cxx_atomic_store(volatile __cxx_atomic_base_impl<_Tp>* __a,  _Tp __val,
747                        memory_order __order) {
748  __atomic_store(&__a->__a_value, &__val,
749                 __to_gcc_order(__order));
750}
751
752template <typename _Tp>
753_LIBCPP_INLINE_VISIBILITY
754void __cxx_atomic_store(__cxx_atomic_base_impl<_Tp>* __a,  _Tp __val,
755                        memory_order __order) {
756  __atomic_store(&__a->__a_value, &__val,
757                 __to_gcc_order(__order));
758}
759
760template <typename _Tp>
761_LIBCPP_INLINE_VISIBILITY
762_Tp __cxx_atomic_load(const volatile __cxx_atomic_base_impl<_Tp>* __a,
763                      memory_order __order) {
764  _Tp __ret;
765  __atomic_load(&__a->__a_value, &__ret,
766                __to_gcc_order(__order));
767  return __ret;
768}
769
770template <typename _Tp>
771_LIBCPP_INLINE_VISIBILITY
772_Tp __cxx_atomic_load(const __cxx_atomic_base_impl<_Tp>* __a, memory_order __order) {
773  _Tp __ret;
774  __atomic_load(&__a->__a_value, &__ret,
775                __to_gcc_order(__order));
776  return __ret;
777}
778
779template <typename _Tp>
780_LIBCPP_INLINE_VISIBILITY
781_Tp __cxx_atomic_exchange(volatile __cxx_atomic_base_impl<_Tp>* __a,
782                          _Tp __value, memory_order __order) {
783  _Tp __ret;
784  __atomic_exchange(&__a->__a_value, &__value, &__ret,
785                    __to_gcc_order(__order));
786  return __ret;
787}
788
789template <typename _Tp>
790_LIBCPP_INLINE_VISIBILITY
791_Tp __cxx_atomic_exchange(__cxx_atomic_base_impl<_Tp>* __a, _Tp __value,
792                          memory_order __order) {
793  _Tp __ret;
794  __atomic_exchange(&__a->__a_value, &__value, &__ret,
795                    __to_gcc_order(__order));
796  return __ret;
797}
798
799template <typename _Tp>
800_LIBCPP_INLINE_VISIBILITY
801bool __cxx_atomic_compare_exchange_strong(
802    volatile __cxx_atomic_base_impl<_Tp>* __a, _Tp* __expected, _Tp __value,
803    memory_order __success, memory_order __failure) {
804  return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
805                                   false,
806                                   __to_gcc_order(__success),
807                                   __to_gcc_failure_order(__failure));
808}
809
810template <typename _Tp>
811_LIBCPP_INLINE_VISIBILITY
812bool __cxx_atomic_compare_exchange_strong(
813    __cxx_atomic_base_impl<_Tp>* __a, _Tp* __expected, _Tp __value, memory_order __success,
814    memory_order __failure) {
815  return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
816                                   false,
817                                   __to_gcc_order(__success),
818                                   __to_gcc_failure_order(__failure));
819}
820
821template <typename _Tp>
822_LIBCPP_INLINE_VISIBILITY
823bool __cxx_atomic_compare_exchange_weak(
824    volatile __cxx_atomic_base_impl<_Tp>* __a, _Tp* __expected, _Tp __value,
825    memory_order __success, memory_order __failure) {
826  return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
827                                   true,
828                                   __to_gcc_order(__success),
829                                   __to_gcc_failure_order(__failure));
830}
831
832template <typename _Tp>
833_LIBCPP_INLINE_VISIBILITY
834bool __cxx_atomic_compare_exchange_weak(
835    __cxx_atomic_base_impl<_Tp>* __a, _Tp* __expected, _Tp __value, memory_order __success,
836    memory_order __failure) {
837  return __atomic_compare_exchange(&__a->__a_value, __expected, &__value,
838                                   true,
839                                   __to_gcc_order(__success),
840                                   __to_gcc_failure_order(__failure));
841}
842
843template <typename _Tp>
844struct __skip_amt { enum {value = 1}; };
845
846template <typename _Tp>
847struct __skip_amt<_Tp*> { enum {value = sizeof(_Tp)}; };
848
849// FIXME: Haven't figured out what the spec says about using arrays with
850// atomic_fetch_add. Force a failure rather than creating bad behavior.
851template <typename _Tp>
852struct __skip_amt<_Tp[]> { };
853template <typename _Tp, int n>
854struct __skip_amt<_Tp[n]> { };
855
856template <typename _Tp, typename _Td>
857_LIBCPP_INLINE_VISIBILITY
858_Tp __cxx_atomic_fetch_add(volatile __cxx_atomic_base_impl<_Tp>* __a,
859                           _Td __delta, memory_order __order) {
860  return __atomic_fetch_add(&__a->__a_value, __delta * __skip_amt<_Tp>::value,
861                            __to_gcc_order(__order));
862}
863
864template <typename _Tp, typename _Td>
865_LIBCPP_INLINE_VISIBILITY
866_Tp __cxx_atomic_fetch_add(__cxx_atomic_base_impl<_Tp>* __a, _Td __delta,
867                           memory_order __order) {
868  return __atomic_fetch_add(&__a->__a_value, __delta * __skip_amt<_Tp>::value,
869                            __to_gcc_order(__order));
870}
871
872template <typename _Tp, typename _Td>
873_LIBCPP_INLINE_VISIBILITY
874_Tp __cxx_atomic_fetch_sub(volatile __cxx_atomic_base_impl<_Tp>* __a,
875                           _Td __delta, memory_order __order) {
876  return __atomic_fetch_sub(&__a->__a_value, __delta * __skip_amt<_Tp>::value,
877                            __to_gcc_order(__order));
878}
879
880template <typename _Tp, typename _Td>
881_LIBCPP_INLINE_VISIBILITY
882_Tp __cxx_atomic_fetch_sub(__cxx_atomic_base_impl<_Tp>* __a, _Td __delta,
883                           memory_order __order) {
884  return __atomic_fetch_sub(&__a->__a_value, __delta * __skip_amt<_Tp>::value,
885                            __to_gcc_order(__order));
886}
887
888template <typename _Tp>
889_LIBCPP_INLINE_VISIBILITY
890_Tp __cxx_atomic_fetch_and(volatile __cxx_atomic_base_impl<_Tp>* __a,
891                           _Tp __pattern, memory_order __order) {
892  return __atomic_fetch_and(&__a->__a_value, __pattern,
893                            __to_gcc_order(__order));
894}
895
896template <typename _Tp>
897_LIBCPP_INLINE_VISIBILITY
898_Tp __cxx_atomic_fetch_and(__cxx_atomic_base_impl<_Tp>* __a,
899                           _Tp __pattern, memory_order __order) {
900  return __atomic_fetch_and(&__a->__a_value, __pattern,
901                            __to_gcc_order(__order));
902}
903
904template <typename _Tp>
905_LIBCPP_INLINE_VISIBILITY
906_Tp __cxx_atomic_fetch_or(volatile __cxx_atomic_base_impl<_Tp>* __a,
907                          _Tp __pattern, memory_order __order) {
908  return __atomic_fetch_or(&__a->__a_value, __pattern,
909                           __to_gcc_order(__order));
910}
911
912template <typename _Tp>
913_LIBCPP_INLINE_VISIBILITY
914_Tp __cxx_atomic_fetch_or(__cxx_atomic_base_impl<_Tp>* __a, _Tp __pattern,
915                          memory_order __order) {
916  return __atomic_fetch_or(&__a->__a_value, __pattern,
917                           __to_gcc_order(__order));
918}
919
920template <typename _Tp>
921_LIBCPP_INLINE_VISIBILITY
922_Tp __cxx_atomic_fetch_xor(volatile __cxx_atomic_base_impl<_Tp>* __a,
923                           _Tp __pattern, memory_order __order) {
924  return __atomic_fetch_xor(&__a->__a_value, __pattern,
925                            __to_gcc_order(__order));
926}
927
928template <typename _Tp>
929_LIBCPP_INLINE_VISIBILITY
930_Tp __cxx_atomic_fetch_xor(__cxx_atomic_base_impl<_Tp>* __a, _Tp __pattern,
931                           memory_order __order) {
932  return __atomic_fetch_xor(&__a->__a_value, __pattern,
933                            __to_gcc_order(__order));
934}
935
936#define __cxx_atomic_is_lock_free(__s) __atomic_is_lock_free(__s, 0)
937
938#elif defined(_LIBCPP_HAS_C_ATOMIC_IMP)
939
940template <typename _Tp>
941struct __cxx_atomic_base_impl {
942
943  _LIBCPP_INLINE_VISIBILITY
944#ifndef _LIBCPP_CXX03_LANG
945    __cxx_atomic_base_impl() _NOEXCEPT = default;
946#else
947    __cxx_atomic_base_impl() _NOEXCEPT : __a_value() {}
948#endif // _LIBCPP_CXX03_LANG
949  _LIBCPP_CONSTEXPR explicit __cxx_atomic_base_impl(_Tp value) _NOEXCEPT
950    : __a_value(value) {}
951  _LIBCPP_DISABLE_EXTENSION_WARNING _Atomic(_Tp) __a_value;
952};
953
954#define __cxx_atomic_is_lock_free(__s) __c11_atomic_is_lock_free(__s)
955
956_LIBCPP_INLINE_VISIBILITY inline
957void __cxx_atomic_thread_fence(memory_order __order) _NOEXCEPT {
958    __c11_atomic_thread_fence(static_cast<__memory_order_underlying_t>(__order));
959}
960
961_LIBCPP_INLINE_VISIBILITY inline
962void __cxx_atomic_signal_fence(memory_order __order) _NOEXCEPT {
963    __c11_atomic_signal_fence(static_cast<__memory_order_underlying_t>(__order));
964}
965
966template<class _Tp>
967_LIBCPP_INLINE_VISIBILITY
968void __cxx_atomic_init(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __val) _NOEXCEPT {
969    __c11_atomic_init(&__a->__a_value, __val);
970}
971template<class _Tp>
972_LIBCPP_INLINE_VISIBILITY
973void __cxx_atomic_init(__cxx_atomic_base_impl<_Tp> * __a, _Tp __val) _NOEXCEPT {
974    __c11_atomic_init(&__a->__a_value, __val);
975}
976
977template<class _Tp>
978_LIBCPP_INLINE_VISIBILITY
979void __cxx_atomic_store(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __val, memory_order __order) _NOEXCEPT {
980    __c11_atomic_store(&__a->__a_value, __val, static_cast<__memory_order_underlying_t>(__order));
981}
982template<class _Tp>
983_LIBCPP_INLINE_VISIBILITY
984void __cxx_atomic_store(__cxx_atomic_base_impl<_Tp> * __a, _Tp __val, memory_order __order) _NOEXCEPT {
985    __c11_atomic_store(&__a->__a_value, __val, static_cast<__memory_order_underlying_t>(__order));
986}
987
988template<class _Tp>
989_LIBCPP_INLINE_VISIBILITY
990_Tp __cxx_atomic_load(__cxx_atomic_base_impl<_Tp> const volatile* __a, memory_order __order) _NOEXCEPT {
991    using __ptr_type = typename remove_const<decltype(__a->__a_value)>::type*;
992    return __c11_atomic_load(const_cast<__ptr_type>(&__a->__a_value), static_cast<__memory_order_underlying_t>(__order));
993}
994template<class _Tp>
995_LIBCPP_INLINE_VISIBILITY
996_Tp __cxx_atomic_load(__cxx_atomic_base_impl<_Tp> const* __a, memory_order __order) _NOEXCEPT {
997    using __ptr_type = typename remove_const<decltype(__a->__a_value)>::type*;
998    return __c11_atomic_load(const_cast<__ptr_type>(&__a->__a_value), static_cast<__memory_order_underlying_t>(__order));
999}
1000
1001template<class _Tp>
1002_LIBCPP_INLINE_VISIBILITY
1003_Tp __cxx_atomic_exchange(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __value, memory_order __order) _NOEXCEPT {
1004    return __c11_atomic_exchange(&__a->__a_value, __value, static_cast<__memory_order_underlying_t>(__order));
1005}
1006template<class _Tp>
1007_LIBCPP_INLINE_VISIBILITY
1008_Tp __cxx_atomic_exchange(__cxx_atomic_base_impl<_Tp> * __a, _Tp __value, memory_order __order) _NOEXCEPT {
1009    return __c11_atomic_exchange(&__a->__a_value, __value, static_cast<__memory_order_underlying_t>(__order));
1010}
1011
1012template<class _Tp>
1013_LIBCPP_INLINE_VISIBILITY
1014bool __cxx_atomic_compare_exchange_strong(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp* __expected, _Tp __value, memory_order __success, memory_order __failure) _NOEXCEPT {
1015    return __c11_atomic_compare_exchange_strong(&__a->__a_value, __expected, __value, static_cast<__memory_order_underlying_t>(__success), static_cast<__memory_order_underlying_t>(__failure));
1016}
1017template<class _Tp>
1018_LIBCPP_INLINE_VISIBILITY
1019bool __cxx_atomic_compare_exchange_strong(__cxx_atomic_base_impl<_Tp> * __a, _Tp* __expected, _Tp __value, memory_order __success, memory_order __failure) _NOEXCEPT {
1020    return __c11_atomic_compare_exchange_strong(&__a->__a_value, __expected, __value, static_cast<__memory_order_underlying_t>(__success), static_cast<__memory_order_underlying_t>(__failure));
1021}
1022
1023template<class _Tp>
1024_LIBCPP_INLINE_VISIBILITY
1025bool __cxx_atomic_compare_exchange_weak(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp* __expected, _Tp __value, memory_order __success, memory_order __failure) _NOEXCEPT {
1026    return __c11_atomic_compare_exchange_weak(&__a->__a_value, __expected, __value, static_cast<__memory_order_underlying_t>(__success), static_cast<__memory_order_underlying_t>(__failure));
1027}
1028template<class _Tp>
1029_LIBCPP_INLINE_VISIBILITY
1030bool __cxx_atomic_compare_exchange_weak(__cxx_atomic_base_impl<_Tp> * __a, _Tp* __expected, _Tp __value, memory_order __success, memory_order __failure) _NOEXCEPT {
1031    return __c11_atomic_compare_exchange_weak(&__a->__a_value, __expected, __value,  static_cast<__memory_order_underlying_t>(__success), static_cast<__memory_order_underlying_t>(__failure));
1032}
1033
1034template<class _Tp>
1035_LIBCPP_INLINE_VISIBILITY
1036_Tp __cxx_atomic_fetch_add(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __delta, memory_order __order) _NOEXCEPT {
1037    return __c11_atomic_fetch_add(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1038}
1039template<class _Tp>
1040_LIBCPP_INLINE_VISIBILITY
1041_Tp __cxx_atomic_fetch_add(__cxx_atomic_base_impl<_Tp> * __a, _Tp __delta, memory_order __order) _NOEXCEPT {
1042    return __c11_atomic_fetch_add(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1043}
1044
1045template<class _Tp>
1046_LIBCPP_INLINE_VISIBILITY
1047_Tp* __cxx_atomic_fetch_add(__cxx_atomic_base_impl<_Tp*> volatile* __a, ptrdiff_t __delta, memory_order __order) _NOEXCEPT {
1048    return __c11_atomic_fetch_add(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1049}
1050template<class _Tp>
1051_LIBCPP_INLINE_VISIBILITY
1052_Tp* __cxx_atomic_fetch_add(__cxx_atomic_base_impl<_Tp*> * __a, ptrdiff_t __delta, memory_order __order) _NOEXCEPT {
1053    return __c11_atomic_fetch_add(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1054}
1055
1056template<class _Tp>
1057_LIBCPP_INLINE_VISIBILITY
1058_Tp __cxx_atomic_fetch_sub(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __delta, memory_order __order) _NOEXCEPT {
1059    return __c11_atomic_fetch_sub(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1060}
1061template<class _Tp>
1062_LIBCPP_INLINE_VISIBILITY
1063_Tp __cxx_atomic_fetch_sub(__cxx_atomic_base_impl<_Tp> * __a, _Tp __delta, memory_order __order) _NOEXCEPT {
1064    return __c11_atomic_fetch_sub(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1065}
1066template<class _Tp>
1067_LIBCPP_INLINE_VISIBILITY
1068_Tp* __cxx_atomic_fetch_sub(__cxx_atomic_base_impl<_Tp*> volatile* __a, ptrdiff_t __delta, memory_order __order) _NOEXCEPT {
1069    return __c11_atomic_fetch_sub(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1070}
1071template<class _Tp>
1072_LIBCPP_INLINE_VISIBILITY
1073_Tp* __cxx_atomic_fetch_sub(__cxx_atomic_base_impl<_Tp*> * __a, ptrdiff_t __delta, memory_order __order) _NOEXCEPT {
1074    return __c11_atomic_fetch_sub(&__a->__a_value, __delta, static_cast<__memory_order_underlying_t>(__order));
1075}
1076
1077template<class _Tp>
1078_LIBCPP_INLINE_VISIBILITY
1079_Tp __cxx_atomic_fetch_and(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1080    return __c11_atomic_fetch_and(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1081}
1082template<class _Tp>
1083_LIBCPP_INLINE_VISIBILITY
1084_Tp __cxx_atomic_fetch_and(__cxx_atomic_base_impl<_Tp> * __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1085    return __c11_atomic_fetch_and(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1086}
1087
1088template<class _Tp>
1089_LIBCPP_INLINE_VISIBILITY
1090_Tp __cxx_atomic_fetch_or(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1091    return __c11_atomic_fetch_or(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1092}
1093template<class _Tp>
1094_LIBCPP_INLINE_VISIBILITY
1095_Tp __cxx_atomic_fetch_or(__cxx_atomic_base_impl<_Tp> * __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1096    return __c11_atomic_fetch_or(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1097}
1098
1099template<class _Tp>
1100_LIBCPP_INLINE_VISIBILITY
1101_Tp __cxx_atomic_fetch_xor(__cxx_atomic_base_impl<_Tp> volatile* __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1102    return __c11_atomic_fetch_xor(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1103}
1104template<class _Tp>
1105_LIBCPP_INLINE_VISIBILITY
1106_Tp __cxx_atomic_fetch_xor(__cxx_atomic_base_impl<_Tp> * __a, _Tp __pattern, memory_order __order) _NOEXCEPT {
1107    return __c11_atomic_fetch_xor(&__a->__a_value, __pattern, static_cast<__memory_order_underlying_t>(__order));
1108}
1109
1110#endif // _LIBCPP_HAS_GCC_ATOMIC_IMP, _LIBCPP_HAS_C_ATOMIC_IMP
1111
1112template <class _Tp>
1113_LIBCPP_INLINE_VISIBILITY
1114_Tp kill_dependency(_Tp __y) _NOEXCEPT
1115{
1116    return __y;
1117}
1118
1119#if defined(__CLANG_ATOMIC_BOOL_LOCK_FREE)
1120# define ATOMIC_BOOL_LOCK_FREE      __CLANG_ATOMIC_BOOL_LOCK_FREE
1121# define ATOMIC_CHAR_LOCK_FREE      __CLANG_ATOMIC_CHAR_LOCK_FREE
1122# define ATOMIC_CHAR16_T_LOCK_FREE  __CLANG_ATOMIC_CHAR16_T_LOCK_FREE
1123# define ATOMIC_CHAR32_T_LOCK_FREE  __CLANG_ATOMIC_CHAR32_T_LOCK_FREE
1124# define ATOMIC_WCHAR_T_LOCK_FREE   __CLANG_ATOMIC_WCHAR_T_LOCK_FREE
1125# define ATOMIC_SHORT_LOCK_FREE     __CLANG_ATOMIC_SHORT_LOCK_FREE
1126# define ATOMIC_INT_LOCK_FREE       __CLANG_ATOMIC_INT_LOCK_FREE
1127# define ATOMIC_LONG_LOCK_FREE      __CLANG_ATOMIC_LONG_LOCK_FREE
1128# define ATOMIC_LLONG_LOCK_FREE     __CLANG_ATOMIC_LLONG_LOCK_FREE
1129# define ATOMIC_POINTER_LOCK_FREE   __CLANG_ATOMIC_POINTER_LOCK_FREE
1130#elif defined(__GCC_ATOMIC_BOOL_LOCK_FREE)
1131# define ATOMIC_BOOL_LOCK_FREE      __GCC_ATOMIC_BOOL_LOCK_FREE
1132# define ATOMIC_CHAR_LOCK_FREE      __GCC_ATOMIC_CHAR_LOCK_FREE
1133# define ATOMIC_CHAR16_T_LOCK_FREE  __GCC_ATOMIC_CHAR16_T_LOCK_FREE
1134# define ATOMIC_CHAR32_T_LOCK_FREE  __GCC_ATOMIC_CHAR32_T_LOCK_FREE
1135# define ATOMIC_WCHAR_T_LOCK_FREE   __GCC_ATOMIC_WCHAR_T_LOCK_FREE
1136# define ATOMIC_SHORT_LOCK_FREE     __GCC_ATOMIC_SHORT_LOCK_FREE
1137# define ATOMIC_INT_LOCK_FREE       __GCC_ATOMIC_INT_LOCK_FREE
1138# define ATOMIC_LONG_LOCK_FREE      __GCC_ATOMIC_LONG_LOCK_FREE
1139# define ATOMIC_LLONG_LOCK_FREE     __GCC_ATOMIC_LLONG_LOCK_FREE
1140# define ATOMIC_POINTER_LOCK_FREE   __GCC_ATOMIC_POINTER_LOCK_FREE
1141#endif
1142
1143#ifdef _LIBCPP_ATOMIC_ONLY_USE_BUILTINS
1144
1145template<typename _Tp>
1146struct __cxx_atomic_lock_impl {
1147
1148  _LIBCPP_INLINE_VISIBILITY
1149  __cxx_atomic_lock_impl() _NOEXCEPT
1150    : __a_value(), __a_lock(0) {}
1151  _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR explicit
1152  __cxx_atomic_lock_impl(_Tp value) _NOEXCEPT
1153    : __a_value(value), __a_lock(0) {}
1154
1155  _Tp __a_value;
1156  mutable __cxx_atomic_base_impl<_LIBCPP_ATOMIC_FLAG_TYPE> __a_lock;
1157
1158  _LIBCPP_INLINE_VISIBILITY void __lock() const volatile {
1159    while(1 == __cxx_atomic_exchange(&__a_lock, _LIBCPP_ATOMIC_FLAG_TYPE(true), memory_order_acquire))
1160        /*spin*/;
1161  }
1162  _LIBCPP_INLINE_VISIBILITY void __lock() const {
1163    while(1 == __cxx_atomic_exchange(&__a_lock, _LIBCPP_ATOMIC_FLAG_TYPE(true), memory_order_acquire))
1164        /*spin*/;
1165  }
1166  _LIBCPP_INLINE_VISIBILITY void __unlock() const volatile {
1167    __cxx_atomic_store(&__a_lock, _LIBCPP_ATOMIC_FLAG_TYPE(false), memory_order_release);
1168  }
1169  _LIBCPP_INLINE_VISIBILITY void __unlock() const {
1170    __cxx_atomic_store(&__a_lock, _LIBCPP_ATOMIC_FLAG_TYPE(false), memory_order_release);
1171  }
1172  _LIBCPP_INLINE_VISIBILITY _Tp __read() const volatile {
1173    __lock();
1174    _Tp __old;
1175    __cxx_atomic_assign_volatile(__old, __a_value);
1176    __unlock();
1177    return __old;
1178  }
1179  _LIBCPP_INLINE_VISIBILITY _Tp __read() const {
1180    __lock();
1181    _Tp __old = __a_value;
1182    __unlock();
1183    return __old;
1184  }
1185};
1186
1187template <typename _Tp>
1188_LIBCPP_INLINE_VISIBILITY
1189void __cxx_atomic_init(volatile __cxx_atomic_lock_impl<_Tp>* __a,  _Tp __val) {
1190  __cxx_atomic_assign_volatile(__a->__a_value, __val);
1191}
1192template <typename _Tp>
1193_LIBCPP_INLINE_VISIBILITY
1194void __cxx_atomic_init(__cxx_atomic_lock_impl<_Tp>* __a,  _Tp __val) {
1195  __a->__a_value = __val;
1196}
1197
1198template <typename _Tp>
1199_LIBCPP_INLINE_VISIBILITY
1200void __cxx_atomic_store(volatile __cxx_atomic_lock_impl<_Tp>* __a,  _Tp __val, memory_order) {
1201  __a->__lock();
1202  __cxx_atomic_assign_volatile(__a->__a_value, __val);
1203  __a->__unlock();
1204}
1205template <typename _Tp>
1206_LIBCPP_INLINE_VISIBILITY
1207void __cxx_atomic_store(__cxx_atomic_lock_impl<_Tp>* __a,  _Tp __val, memory_order) {
1208  __a->__lock();
1209  __a->__a_value = __val;
1210  __a->__unlock();
1211}
1212
1213template <typename _Tp>
1214_LIBCPP_INLINE_VISIBILITY
1215_Tp __cxx_atomic_load(const volatile __cxx_atomic_lock_impl<_Tp>* __a, memory_order) {
1216  return __a->__read();
1217}
1218template <typename _Tp>
1219_LIBCPP_INLINE_VISIBILITY
1220_Tp __cxx_atomic_load(const __cxx_atomic_lock_impl<_Tp>* __a, memory_order) {
1221  return __a->__read();
1222}
1223
1224template <typename _Tp>
1225_LIBCPP_INLINE_VISIBILITY
1226_Tp __cxx_atomic_exchange(volatile __cxx_atomic_lock_impl<_Tp>* __a, _Tp __value, memory_order) {
1227  __a->__lock();
1228  _Tp __old;
1229  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1230  __cxx_atomic_assign_volatile(__a->__a_value, __value);
1231  __a->__unlock();
1232  return __old;
1233}
1234template <typename _Tp>
1235_LIBCPP_INLINE_VISIBILITY
1236_Tp __cxx_atomic_exchange(__cxx_atomic_lock_impl<_Tp>* __a, _Tp __value, memory_order) {
1237  __a->__lock();
1238  _Tp __old = __a->__a_value;
1239  __a->__a_value = __value;
1240  __a->__unlock();
1241  return __old;
1242}
1243
1244template <typename _Tp>
1245_LIBCPP_INLINE_VISIBILITY
1246bool __cxx_atomic_compare_exchange_strong(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1247                                          _Tp* __expected, _Tp __value, memory_order, memory_order) {
1248  __a->__lock();
1249  _Tp __temp;
1250  __cxx_atomic_assign_volatile(__temp, __a->__a_value);
1251  bool __ret = __temp == *__expected;
1252  if(__ret)
1253    __cxx_atomic_assign_volatile(__a->__a_value, __value);
1254  else
1255    __cxx_atomic_assign_volatile(*__expected, __a->__a_value);
1256  __a->__unlock();
1257  return __ret;
1258}
1259template <typename _Tp>
1260_LIBCPP_INLINE_VISIBILITY
1261bool __cxx_atomic_compare_exchange_strong(__cxx_atomic_lock_impl<_Tp>* __a,
1262                                          _Tp* __expected, _Tp __value, memory_order, memory_order) {
1263  __a->__lock();
1264  bool __ret = __a->__a_value == *__expected;
1265  if(__ret)
1266    __a->__a_value = __value;
1267  else
1268    *__expected = __a->__a_value;
1269  __a->__unlock();
1270  return __ret;
1271}
1272
1273template <typename _Tp>
1274_LIBCPP_INLINE_VISIBILITY
1275bool __cxx_atomic_compare_exchange_weak(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1276                                        _Tp* __expected, _Tp __value, memory_order, memory_order) {
1277  __a->__lock();
1278  _Tp __temp;
1279  __cxx_atomic_assign_volatile(__temp, __a->__a_value);
1280  bool __ret = __temp == *__expected;
1281  if(__ret)
1282    __cxx_atomic_assign_volatile(__a->__a_value, __value);
1283  else
1284    __cxx_atomic_assign_volatile(*__expected, __a->__a_value);
1285  __a->__unlock();
1286  return __ret;
1287}
1288template <typename _Tp>
1289_LIBCPP_INLINE_VISIBILITY
1290bool __cxx_atomic_compare_exchange_weak(__cxx_atomic_lock_impl<_Tp>* __a,
1291                                        _Tp* __expected, _Tp __value, memory_order, memory_order) {
1292  __a->__lock();
1293  bool __ret = __a->__a_value == *__expected;
1294  if(__ret)
1295    __a->__a_value = __value;
1296  else
1297    *__expected = __a->__a_value;
1298  __a->__unlock();
1299  return __ret;
1300}
1301
1302template <typename _Tp, typename _Td>
1303_LIBCPP_INLINE_VISIBILITY
1304_Tp __cxx_atomic_fetch_add(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1305                           _Td __delta, memory_order) {
1306  __a->__lock();
1307  _Tp __old;
1308  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1309  __cxx_atomic_assign_volatile(__a->__a_value, _Tp(__old + __delta));
1310  __a->__unlock();
1311  return __old;
1312}
1313template <typename _Tp, typename _Td>
1314_LIBCPP_INLINE_VISIBILITY
1315_Tp __cxx_atomic_fetch_add(__cxx_atomic_lock_impl<_Tp>* __a,
1316                           _Td __delta, memory_order) {
1317  __a->__lock();
1318  _Tp __old = __a->__a_value;
1319  __a->__a_value += __delta;
1320  __a->__unlock();
1321  return __old;
1322}
1323
1324template <typename _Tp, typename _Td>
1325_LIBCPP_INLINE_VISIBILITY
1326_Tp* __cxx_atomic_fetch_add(volatile __cxx_atomic_lock_impl<_Tp*>* __a,
1327                           ptrdiff_t __delta, memory_order) {
1328  __a->__lock();
1329  _Tp* __old;
1330  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1331  __cxx_atomic_assign_volatile(__a->__a_value, __old + __delta);
1332  __a->__unlock();
1333  return __old;
1334}
1335template <typename _Tp, typename _Td>
1336_LIBCPP_INLINE_VISIBILITY
1337_Tp* __cxx_atomic_fetch_add(__cxx_atomic_lock_impl<_Tp*>* __a,
1338                           ptrdiff_t __delta, memory_order) {
1339  __a->__lock();
1340  _Tp* __old = __a->__a_value;
1341  __a->__a_value += __delta;
1342  __a->__unlock();
1343  return __old;
1344}
1345
1346template <typename _Tp, typename _Td>
1347_LIBCPP_INLINE_VISIBILITY
1348_Tp __cxx_atomic_fetch_sub(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1349                           _Td __delta, memory_order) {
1350  __a->__lock();
1351  _Tp __old;
1352  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1353  __cxx_atomic_assign_volatile(__a->__a_value, _Tp(__old - __delta));
1354  __a->__unlock();
1355  return __old;
1356}
1357template <typename _Tp, typename _Td>
1358_LIBCPP_INLINE_VISIBILITY
1359_Tp __cxx_atomic_fetch_sub(__cxx_atomic_lock_impl<_Tp>* __a,
1360                           _Td __delta, memory_order) {
1361  __a->__lock();
1362  _Tp __old = __a->__a_value;
1363  __a->__a_value -= __delta;
1364  __a->__unlock();
1365  return __old;
1366}
1367
1368template <typename _Tp>
1369_LIBCPP_INLINE_VISIBILITY
1370_Tp __cxx_atomic_fetch_and(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1371                           _Tp __pattern, memory_order) {
1372  __a->__lock();
1373  _Tp __old;
1374  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1375  __cxx_atomic_assign_volatile(__a->__a_value, _Tp(__old & __pattern));
1376  __a->__unlock();
1377  return __old;
1378}
1379template <typename _Tp>
1380_LIBCPP_INLINE_VISIBILITY
1381_Tp __cxx_atomic_fetch_and(__cxx_atomic_lock_impl<_Tp>* __a,
1382                           _Tp __pattern, memory_order) {
1383  __a->__lock();
1384  _Tp __old = __a->__a_value;
1385  __a->__a_value &= __pattern;
1386  __a->__unlock();
1387  return __old;
1388}
1389
1390template <typename _Tp>
1391_LIBCPP_INLINE_VISIBILITY
1392_Tp __cxx_atomic_fetch_or(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1393                          _Tp __pattern, memory_order) {
1394  __a->__lock();
1395  _Tp __old;
1396  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1397  __cxx_atomic_assign_volatile(__a->__a_value, _Tp(__old | __pattern));
1398  __a->__unlock();
1399  return __old;
1400}
1401template <typename _Tp>
1402_LIBCPP_INLINE_VISIBILITY
1403_Tp __cxx_atomic_fetch_or(__cxx_atomic_lock_impl<_Tp>* __a,
1404                          _Tp __pattern, memory_order) {
1405  __a->__lock();
1406  _Tp __old = __a->__a_value;
1407  __a->__a_value |= __pattern;
1408  __a->__unlock();
1409  return __old;
1410}
1411
1412template <typename _Tp>
1413_LIBCPP_INLINE_VISIBILITY
1414_Tp __cxx_atomic_fetch_xor(volatile __cxx_atomic_lock_impl<_Tp>* __a,
1415                           _Tp __pattern, memory_order) {
1416  __a->__lock();
1417  _Tp __old;
1418  __cxx_atomic_assign_volatile(__old, __a->__a_value);
1419  __cxx_atomic_assign_volatile(__a->__a_value, _Tp(__old ^ __pattern));
1420  __a->__unlock();
1421  return __old;
1422}
1423template <typename _Tp>
1424_LIBCPP_INLINE_VISIBILITY
1425_Tp __cxx_atomic_fetch_xor(__cxx_atomic_lock_impl<_Tp>* __a,
1426                           _Tp __pattern, memory_order) {
1427  __a->__lock();
1428  _Tp __old = __a->__a_value;
1429  __a->__a_value ^= __pattern;
1430  __a->__unlock();
1431  return __old;
1432}
1433
1434#ifdef __cpp_lib_atomic_is_always_lock_free
1435
1436template<typename _Tp> struct __cxx_is_always_lock_free {
1437    enum { __value = __atomic_always_lock_free(sizeof(_Tp), 0) }; };
1438
1439#else
1440
1441template<typename _Tp> struct __cxx_is_always_lock_free { enum { __value = false }; };
1442// Implementations must match the C ATOMIC_*_LOCK_FREE macro values.
1443template<> struct __cxx_is_always_lock_free<bool> { enum { __value = 2 == ATOMIC_BOOL_LOCK_FREE }; };
1444template<> struct __cxx_is_always_lock_free<char> { enum { __value = 2 == ATOMIC_CHAR_LOCK_FREE }; };
1445template<> struct __cxx_is_always_lock_free<signed char> { enum { __value = 2 == ATOMIC_CHAR_LOCK_FREE }; };
1446template<> struct __cxx_is_always_lock_free<unsigned char> { enum { __value = 2 == ATOMIC_CHAR_LOCK_FREE }; };
1447template<> struct __cxx_is_always_lock_free<char16_t> { enum { __value = 2 == ATOMIC_CHAR16_T_LOCK_FREE }; };
1448template<> struct __cxx_is_always_lock_free<char32_t> { enum { __value = 2 == ATOMIC_CHAR32_T_LOCK_FREE }; };
1449template<> struct __cxx_is_always_lock_free<wchar_t> { enum { __value = 2 == ATOMIC_WCHAR_T_LOCK_FREE }; };
1450template<> struct __cxx_is_always_lock_free<short> { enum { __value = 2 == ATOMIC_SHORT_LOCK_FREE }; };
1451template<> struct __cxx_is_always_lock_free<unsigned short> { enum { __value = 2 == ATOMIC_SHORT_LOCK_FREE }; };
1452template<> struct __cxx_is_always_lock_free<int> { enum { __value = 2 == ATOMIC_INT_LOCK_FREE }; };
1453template<> struct __cxx_is_always_lock_free<unsigned int> { enum { __value = 2 == ATOMIC_INT_LOCK_FREE }; };
1454template<> struct __cxx_is_always_lock_free<long> { enum { __value = 2 == ATOMIC_LONG_LOCK_FREE }; };
1455template<> struct __cxx_is_always_lock_free<unsigned long> { enum { __value = 2 == ATOMIC_LONG_LOCK_FREE }; };
1456template<> struct __cxx_is_always_lock_free<long long> { enum { __value = 2 == ATOMIC_LLONG_LOCK_FREE }; };
1457template<> struct __cxx_is_always_lock_free<unsigned long long> { enum { __value = 2 == ATOMIC_LLONG_LOCK_FREE }; };
1458template<typename _Tp> struct __cxx_is_always_lock_free<_Tp*> { enum { __value = 2 == ATOMIC_POINTER_LOCK_FREE }; };
1459template<> struct __cxx_is_always_lock_free<std::nullptr_t> { enum { __value = 2 == ATOMIC_POINTER_LOCK_FREE }; };
1460
1461#endif //__cpp_lib_atomic_is_always_lock_free
1462
1463template <typename _Tp,
1464          typename _Base = typename conditional<__cxx_is_always_lock_free<_Tp>::__value,
1465                                                __cxx_atomic_base_impl<_Tp>,
1466                                                __cxx_atomic_lock_impl<_Tp> >::type>
1467#else
1468template <typename _Tp,
1469          typename _Base = __cxx_atomic_base_impl<_Tp> >
1470#endif //_LIBCPP_ATOMIC_ONLY_USE_BUILTINS
1471struct __cxx_atomic_impl : public _Base {
1472
1473#if _GNUC_VER >= 501
1474    static_assert(is_trivially_copyable<_Tp>::value,
1475      "std::atomic<Tp> requires that 'Tp' be a trivially copyable type");
1476#endif
1477
1478  _LIBCPP_INLINE_VISIBILITY __cxx_atomic_impl() _NOEXCEPT _LIBCPP_DEFAULT
1479  _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR explicit __cxx_atomic_impl(_Tp value) _NOEXCEPT
1480    : _Base(value) {}
1481};
1482
1483#ifdef __linux__
1484    using __cxx_contention_t = int32_t;
1485#else
1486    using __cxx_contention_t = int64_t;
1487#endif //__linux__
1488
1489#if _LIBCPP_STD_VER >= 11
1490
1491using __cxx_atomic_contention_t = __cxx_atomic_impl<__cxx_contention_t>;
1492
1493#ifndef _LIBCPP_HAS_NO_PLATFORM_WAIT
1494
1495_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __cxx_atomic_notify_one(void const volatile*);
1496_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __cxx_atomic_notify_all(void const volatile*);
1497_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI __cxx_contention_t __libcpp_atomic_monitor(void const volatile*);
1498_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __libcpp_atomic_wait(void const volatile*, __cxx_contention_t);
1499
1500_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __cxx_atomic_notify_one(__cxx_atomic_contention_t const volatile*);
1501_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __cxx_atomic_notify_all(__cxx_atomic_contention_t const volatile*);
1502_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI __cxx_contention_t __libcpp_atomic_monitor(__cxx_atomic_contention_t const volatile*);
1503_LIBCPP_AVAILABILITY_SYNC _LIBCPP_EXPORTED_FROM_ABI void __libcpp_atomic_wait(__cxx_atomic_contention_t const volatile*, __cxx_contention_t);
1504
1505template <class _Atp, class _Fn>
1506struct __libcpp_atomic_wait_backoff_impl {
1507    _Atp* __a;
1508    _Fn __test_fn;
1509    _LIBCPP_AVAILABILITY_SYNC
1510    _LIBCPP_INLINE_VISIBILITY bool operator()(chrono::nanoseconds __elapsed) const
1511    {
1512        if(__elapsed > chrono::microseconds(64))
1513        {
1514            auto const __monitor = __libcpp_atomic_monitor(__a);
1515            if(__test_fn())
1516                return true;
1517            __libcpp_atomic_wait(__a, __monitor);
1518        }
1519        else if(__elapsed > chrono::microseconds(4))
1520            __libcpp_thread_yield();
1521        else
1522            ; // poll
1523        return false;
1524    }
1525};
1526
1527template <class _Atp, class _Fn>
1528_LIBCPP_AVAILABILITY_SYNC
1529_LIBCPP_INLINE_VISIBILITY bool __cxx_atomic_wait(_Atp* __a, _Fn && __test_fn)
1530{
1531    __libcpp_atomic_wait_backoff_impl<_Atp, typename decay<_Fn>::type> __backoff_fn = {__a, __test_fn};
1532    return __libcpp_thread_poll_with_backoff(__test_fn, __backoff_fn);
1533}
1534
1535#else // _LIBCPP_HAS_NO_PLATFORM_WAIT
1536
1537template <class _Tp>
1538_LIBCPP_INLINE_VISIBILITY void __cxx_atomic_notify_all(__cxx_atomic_impl<_Tp> const volatile*) { }
1539template <class _Tp>
1540_LIBCPP_INLINE_VISIBILITY void __cxx_atomic_notify_one(__cxx_atomic_impl<_Tp> const volatile*) { }
1541template <class _Atp, class _Fn>
1542_LIBCPP_INLINE_VISIBILITY bool __cxx_atomic_wait(_Atp*, _Fn && __test_fn)
1543{
1544    return __libcpp_thread_poll_with_backoff(__test_fn, __libcpp_timed_backoff_policy());
1545}
1546
1547#endif // _LIBCPP_HAS_NO_PLATFORM_WAIT
1548
1549template <class _Atp, class _Tp>
1550struct __cxx_atomic_wait_test_fn_impl {
1551    _Atp* __a;
1552    _Tp __val;
1553    memory_order __order;
1554    _LIBCPP_INLINE_VISIBILITY bool operator()() const
1555    {
1556        return !__cxx_nonatomic_compare_equal(__cxx_atomic_load(__a, __order), __val);
1557    }
1558};
1559
1560template <class _Atp, class _Tp>
1561_LIBCPP_AVAILABILITY_SYNC
1562_LIBCPP_INLINE_VISIBILITY bool __cxx_atomic_wait(_Atp* __a, _Tp const __val, memory_order __order)
1563{
1564    __cxx_atomic_wait_test_fn_impl<_Atp, _Tp> __test_fn = {__a, __val, __order};
1565    return __cxx_atomic_wait(__a, __test_fn);
1566}
1567
1568#endif //_LIBCPP_STD_VER >= 11
1569
1570// general atomic<T>
1571
1572template <class _Tp, bool = is_integral<_Tp>::value && !is_same<_Tp, bool>::value>
1573struct __atomic_base  // false
1574{
1575    mutable __cxx_atomic_impl<_Tp> __a_;
1576
1577#if defined(__cpp_lib_atomic_is_always_lock_free)
1578  static _LIBCPP_CONSTEXPR bool is_always_lock_free = __atomic_always_lock_free(sizeof(__a_), 0);
1579#endif
1580
1581    _LIBCPP_INLINE_VISIBILITY
1582    bool is_lock_free() const volatile _NOEXCEPT
1583        {return __cxx_atomic_is_lock_free(sizeof(_Tp));}
1584    _LIBCPP_INLINE_VISIBILITY
1585    bool is_lock_free() const _NOEXCEPT
1586        {return static_cast<__atomic_base const volatile*>(this)->is_lock_free();}
1587    _LIBCPP_INLINE_VISIBILITY
1588    void store(_Tp __d, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1589      _LIBCPP_CHECK_STORE_MEMORY_ORDER(__m)
1590        {__cxx_atomic_store(&__a_, __d, __m);}
1591    _LIBCPP_INLINE_VISIBILITY
1592    void store(_Tp __d, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1593      _LIBCPP_CHECK_STORE_MEMORY_ORDER(__m)
1594        {__cxx_atomic_store(&__a_, __d, __m);}
1595    _LIBCPP_INLINE_VISIBILITY
1596    _Tp load(memory_order __m = memory_order_seq_cst) const volatile _NOEXCEPT
1597      _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
1598        {return __cxx_atomic_load(&__a_, __m);}
1599    _LIBCPP_INLINE_VISIBILITY
1600    _Tp load(memory_order __m = memory_order_seq_cst) const _NOEXCEPT
1601      _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
1602        {return __cxx_atomic_load(&__a_, __m);}
1603    _LIBCPP_INLINE_VISIBILITY
1604    operator _Tp() const volatile _NOEXCEPT {return load();}
1605    _LIBCPP_INLINE_VISIBILITY
1606    operator _Tp() const _NOEXCEPT          {return load();}
1607    _LIBCPP_INLINE_VISIBILITY
1608    _Tp exchange(_Tp __d, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1609        {return __cxx_atomic_exchange(&__a_, __d, __m);}
1610    _LIBCPP_INLINE_VISIBILITY
1611    _Tp exchange(_Tp __d, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1612        {return __cxx_atomic_exchange(&__a_, __d, __m);}
1613    _LIBCPP_INLINE_VISIBILITY
1614    bool compare_exchange_weak(_Tp& __e, _Tp __d,
1615                               memory_order __s, memory_order __f) volatile _NOEXCEPT
1616      _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
1617        {return __cxx_atomic_compare_exchange_weak(&__a_, &__e, __d, __s, __f);}
1618    _LIBCPP_INLINE_VISIBILITY
1619    bool compare_exchange_weak(_Tp& __e, _Tp __d,
1620                               memory_order __s, memory_order __f) _NOEXCEPT
1621      _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
1622        {return __cxx_atomic_compare_exchange_weak(&__a_, &__e, __d, __s, __f);}
1623    _LIBCPP_INLINE_VISIBILITY
1624    bool compare_exchange_strong(_Tp& __e, _Tp __d,
1625                                 memory_order __s, memory_order __f) volatile _NOEXCEPT
1626      _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
1627        {return __cxx_atomic_compare_exchange_strong(&__a_, &__e, __d, __s, __f);}
1628    _LIBCPP_INLINE_VISIBILITY
1629    bool compare_exchange_strong(_Tp& __e, _Tp __d,
1630                                 memory_order __s, memory_order __f) _NOEXCEPT
1631      _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
1632        {return __cxx_atomic_compare_exchange_strong(&__a_, &__e, __d, __s, __f);}
1633    _LIBCPP_INLINE_VISIBILITY
1634    bool compare_exchange_weak(_Tp& __e, _Tp __d,
1635                              memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1636        {return __cxx_atomic_compare_exchange_weak(&__a_, &__e, __d, __m, __m);}
1637    _LIBCPP_INLINE_VISIBILITY
1638    bool compare_exchange_weak(_Tp& __e, _Tp __d,
1639                               memory_order __m = memory_order_seq_cst) _NOEXCEPT
1640        {return __cxx_atomic_compare_exchange_weak(&__a_, &__e, __d, __m, __m);}
1641    _LIBCPP_INLINE_VISIBILITY
1642    bool compare_exchange_strong(_Tp& __e, _Tp __d,
1643                              memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1644        {return __cxx_atomic_compare_exchange_strong(&__a_, &__e, __d, __m, __m);}
1645    _LIBCPP_INLINE_VISIBILITY
1646    bool compare_exchange_strong(_Tp& __e, _Tp __d,
1647                                 memory_order __m = memory_order_seq_cst) _NOEXCEPT
1648        {return __cxx_atomic_compare_exchange_strong(&__a_, &__e, __d, __m, __m);}
1649
1650    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void wait(_Tp __v, memory_order __m = memory_order_seq_cst) const volatile _NOEXCEPT
1651        {__cxx_atomic_wait(&__a_, __v, __m);}
1652    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void wait(_Tp __v, memory_order __m = memory_order_seq_cst) const _NOEXCEPT
1653        {__cxx_atomic_wait(&__a_, __v, __m);}
1654    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void notify_one() volatile _NOEXCEPT
1655        {__cxx_atomic_notify_one(&__a_);}
1656    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void notify_one() _NOEXCEPT
1657        {__cxx_atomic_notify_one(&__a_);}
1658    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void notify_all() volatile _NOEXCEPT
1659        {__cxx_atomic_notify_all(&__a_);}
1660    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY void notify_all() _NOEXCEPT
1661        {__cxx_atomic_notify_all(&__a_);}
1662
1663    _LIBCPP_INLINE_VISIBILITY
1664    __atomic_base() _NOEXCEPT _LIBCPP_DEFAULT
1665
1666    _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR
1667    __atomic_base(_Tp __d) _NOEXCEPT : __a_(__d) {}
1668
1669#ifndef _LIBCPP_CXX03_LANG
1670    __atomic_base(const __atomic_base&) = delete;
1671    __atomic_base& operator=(const __atomic_base&) = delete;
1672    __atomic_base& operator=(const __atomic_base&) volatile = delete;
1673#else
1674private:
1675    _LIBCPP_INLINE_VISIBILITY
1676    __atomic_base(const __atomic_base&);
1677    _LIBCPP_INLINE_VISIBILITY
1678    __atomic_base& operator=(const __atomic_base&);
1679    _LIBCPP_INLINE_VISIBILITY
1680    __atomic_base& operator=(const __atomic_base&) volatile;
1681#endif
1682};
1683
1684#if defined(__cpp_lib_atomic_is_always_lock_free)
1685template <class _Tp, bool __b>
1686_LIBCPP_CONSTEXPR bool __atomic_base<_Tp, __b>::is_always_lock_free;
1687#endif
1688
1689// atomic<Integral>
1690
1691template <class _Tp>
1692struct __atomic_base<_Tp, true>
1693    : public __atomic_base<_Tp, false>
1694{
1695    typedef __atomic_base<_Tp, false> __base;
1696    _LIBCPP_INLINE_VISIBILITY
1697    __atomic_base() _NOEXCEPT _LIBCPP_DEFAULT
1698    _LIBCPP_INLINE_VISIBILITY
1699    _LIBCPP_CONSTEXPR __atomic_base(_Tp __d) _NOEXCEPT : __base(__d) {}
1700
1701    _LIBCPP_INLINE_VISIBILITY
1702    _Tp fetch_add(_Tp __op, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1703        {return __cxx_atomic_fetch_add(&this->__a_, __op, __m);}
1704    _LIBCPP_INLINE_VISIBILITY
1705    _Tp fetch_add(_Tp __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1706        {return __cxx_atomic_fetch_add(&this->__a_, __op, __m);}
1707    _LIBCPP_INLINE_VISIBILITY
1708    _Tp fetch_sub(_Tp __op, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1709        {return __cxx_atomic_fetch_sub(&this->__a_, __op, __m);}
1710    _LIBCPP_INLINE_VISIBILITY
1711    _Tp fetch_sub(_Tp __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1712        {return __cxx_atomic_fetch_sub(&this->__a_, __op, __m);}
1713    _LIBCPP_INLINE_VISIBILITY
1714    _Tp fetch_and(_Tp __op, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1715        {return __cxx_atomic_fetch_and(&this->__a_, __op, __m);}
1716    _LIBCPP_INLINE_VISIBILITY
1717    _Tp fetch_and(_Tp __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1718        {return __cxx_atomic_fetch_and(&this->__a_, __op, __m);}
1719    _LIBCPP_INLINE_VISIBILITY
1720    _Tp fetch_or(_Tp __op, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1721        {return __cxx_atomic_fetch_or(&this->__a_, __op, __m);}
1722    _LIBCPP_INLINE_VISIBILITY
1723    _Tp fetch_or(_Tp __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1724        {return __cxx_atomic_fetch_or(&this->__a_, __op, __m);}
1725    _LIBCPP_INLINE_VISIBILITY
1726    _Tp fetch_xor(_Tp __op, memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
1727        {return __cxx_atomic_fetch_xor(&this->__a_, __op, __m);}
1728    _LIBCPP_INLINE_VISIBILITY
1729    _Tp fetch_xor(_Tp __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1730        {return __cxx_atomic_fetch_xor(&this->__a_, __op, __m);}
1731
1732    _LIBCPP_INLINE_VISIBILITY
1733    _Tp operator++(int) volatile _NOEXCEPT      {return fetch_add(_Tp(1));}
1734    _LIBCPP_INLINE_VISIBILITY
1735    _Tp operator++(int) _NOEXCEPT               {return fetch_add(_Tp(1));}
1736    _LIBCPP_INLINE_VISIBILITY
1737    _Tp operator--(int) volatile _NOEXCEPT      {return fetch_sub(_Tp(1));}
1738    _LIBCPP_INLINE_VISIBILITY
1739    _Tp operator--(int) _NOEXCEPT               {return fetch_sub(_Tp(1));}
1740    _LIBCPP_INLINE_VISIBILITY
1741    _Tp operator++() volatile _NOEXCEPT         {return fetch_add(_Tp(1)) + _Tp(1);}
1742    _LIBCPP_INLINE_VISIBILITY
1743    _Tp operator++() _NOEXCEPT                  {return fetch_add(_Tp(1)) + _Tp(1);}
1744    _LIBCPP_INLINE_VISIBILITY
1745    _Tp operator--() volatile _NOEXCEPT         {return fetch_sub(_Tp(1)) - _Tp(1);}
1746    _LIBCPP_INLINE_VISIBILITY
1747    _Tp operator--() _NOEXCEPT                  {return fetch_sub(_Tp(1)) - _Tp(1);}
1748    _LIBCPP_INLINE_VISIBILITY
1749    _Tp operator+=(_Tp __op) volatile _NOEXCEPT {return fetch_add(__op) + __op;}
1750    _LIBCPP_INLINE_VISIBILITY
1751    _Tp operator+=(_Tp __op) _NOEXCEPT          {return fetch_add(__op) + __op;}
1752    _LIBCPP_INLINE_VISIBILITY
1753    _Tp operator-=(_Tp __op) volatile _NOEXCEPT {return fetch_sub(__op) - __op;}
1754    _LIBCPP_INLINE_VISIBILITY
1755    _Tp operator-=(_Tp __op) _NOEXCEPT          {return fetch_sub(__op) - __op;}
1756    _LIBCPP_INLINE_VISIBILITY
1757    _Tp operator&=(_Tp __op) volatile _NOEXCEPT {return fetch_and(__op) & __op;}
1758    _LIBCPP_INLINE_VISIBILITY
1759    _Tp operator&=(_Tp __op) _NOEXCEPT          {return fetch_and(__op) & __op;}
1760    _LIBCPP_INLINE_VISIBILITY
1761    _Tp operator|=(_Tp __op) volatile _NOEXCEPT {return fetch_or(__op) | __op;}
1762    _LIBCPP_INLINE_VISIBILITY
1763    _Tp operator|=(_Tp __op) _NOEXCEPT          {return fetch_or(__op) | __op;}
1764    _LIBCPP_INLINE_VISIBILITY
1765    _Tp operator^=(_Tp __op) volatile _NOEXCEPT {return fetch_xor(__op) ^ __op;}
1766    _LIBCPP_INLINE_VISIBILITY
1767    _Tp operator^=(_Tp __op) _NOEXCEPT          {return fetch_xor(__op) ^ __op;}
1768};
1769
1770// atomic<T>
1771
1772template <class _Tp>
1773struct atomic
1774    : public __atomic_base<_Tp>
1775{
1776    typedef __atomic_base<_Tp> __base;
1777    typedef _Tp value_type;
1778    _LIBCPP_INLINE_VISIBILITY
1779    atomic() _NOEXCEPT _LIBCPP_DEFAULT
1780    _LIBCPP_INLINE_VISIBILITY
1781    _LIBCPP_CONSTEXPR atomic(_Tp __d) _NOEXCEPT : __base(__d) {}
1782
1783    _LIBCPP_INLINE_VISIBILITY
1784    _Tp operator=(_Tp __d) volatile _NOEXCEPT
1785        {__base::store(__d); return __d;}
1786    _LIBCPP_INLINE_VISIBILITY
1787    _Tp operator=(_Tp __d) _NOEXCEPT
1788        {__base::store(__d); return __d;}
1789};
1790
1791// atomic<T*>
1792
1793template <class _Tp>
1794struct atomic<_Tp*>
1795    : public __atomic_base<_Tp*>
1796{
1797    typedef __atomic_base<_Tp*> __base;
1798    typedef _Tp* value_type;
1799    _LIBCPP_INLINE_VISIBILITY
1800    atomic() _NOEXCEPT _LIBCPP_DEFAULT
1801    _LIBCPP_INLINE_VISIBILITY
1802    _LIBCPP_CONSTEXPR atomic(_Tp* __d) _NOEXCEPT : __base(__d) {}
1803
1804    _LIBCPP_INLINE_VISIBILITY
1805    _Tp* operator=(_Tp* __d) volatile _NOEXCEPT
1806        {__base::store(__d); return __d;}
1807    _LIBCPP_INLINE_VISIBILITY
1808    _Tp* operator=(_Tp* __d) _NOEXCEPT
1809        {__base::store(__d); return __d;}
1810
1811    _LIBCPP_INLINE_VISIBILITY
1812    _Tp* fetch_add(ptrdiff_t __op, memory_order __m = memory_order_seq_cst)
1813                                                                        volatile _NOEXCEPT
1814        {return __cxx_atomic_fetch_add(&this->__a_, __op, __m);}
1815    _LIBCPP_INLINE_VISIBILITY
1816    _Tp* fetch_add(ptrdiff_t __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1817        {return __cxx_atomic_fetch_add(&this->__a_, __op, __m);}
1818    _LIBCPP_INLINE_VISIBILITY
1819    _Tp* fetch_sub(ptrdiff_t __op, memory_order __m = memory_order_seq_cst)
1820                                                                        volatile _NOEXCEPT
1821        {return __cxx_atomic_fetch_sub(&this->__a_, __op, __m);}
1822    _LIBCPP_INLINE_VISIBILITY
1823    _Tp* fetch_sub(ptrdiff_t __op, memory_order __m = memory_order_seq_cst) _NOEXCEPT
1824        {return __cxx_atomic_fetch_sub(&this->__a_, __op, __m);}
1825
1826    _LIBCPP_INLINE_VISIBILITY
1827    _Tp* operator++(int) volatile _NOEXCEPT            {return fetch_add(1);}
1828    _LIBCPP_INLINE_VISIBILITY
1829    _Tp* operator++(int) _NOEXCEPT                     {return fetch_add(1);}
1830    _LIBCPP_INLINE_VISIBILITY
1831    _Tp* operator--(int) volatile _NOEXCEPT            {return fetch_sub(1);}
1832    _LIBCPP_INLINE_VISIBILITY
1833    _Tp* operator--(int) _NOEXCEPT                     {return fetch_sub(1);}
1834    _LIBCPP_INLINE_VISIBILITY
1835    _Tp* operator++() volatile _NOEXCEPT               {return fetch_add(1) + 1;}
1836    _LIBCPP_INLINE_VISIBILITY
1837    _Tp* operator++() _NOEXCEPT                        {return fetch_add(1) + 1;}
1838    _LIBCPP_INLINE_VISIBILITY
1839    _Tp* operator--() volatile _NOEXCEPT               {return fetch_sub(1) - 1;}
1840    _LIBCPP_INLINE_VISIBILITY
1841    _Tp* operator--() _NOEXCEPT                        {return fetch_sub(1) - 1;}
1842    _LIBCPP_INLINE_VISIBILITY
1843    _Tp* operator+=(ptrdiff_t __op) volatile _NOEXCEPT {return fetch_add(__op) + __op;}
1844    _LIBCPP_INLINE_VISIBILITY
1845    _Tp* operator+=(ptrdiff_t __op) _NOEXCEPT          {return fetch_add(__op) + __op;}
1846    _LIBCPP_INLINE_VISIBILITY
1847    _Tp* operator-=(ptrdiff_t __op) volatile _NOEXCEPT {return fetch_sub(__op) - __op;}
1848    _LIBCPP_INLINE_VISIBILITY
1849    _Tp* operator-=(ptrdiff_t __op) _NOEXCEPT          {return fetch_sub(__op) - __op;}
1850};
1851
1852// atomic_is_lock_free
1853
1854template <class _Tp>
1855_LIBCPP_INLINE_VISIBILITY
1856bool
1857atomic_is_lock_free(const volatile atomic<_Tp>* __o) _NOEXCEPT
1858{
1859    return __o->is_lock_free();
1860}
1861
1862template <class _Tp>
1863_LIBCPP_INLINE_VISIBILITY
1864bool
1865atomic_is_lock_free(const atomic<_Tp>* __o) _NOEXCEPT
1866{
1867    return __o->is_lock_free();
1868}
1869
1870// atomic_init
1871
1872template <class _Tp>
1873_LIBCPP_INLINE_VISIBILITY
1874void
1875atomic_init(volatile atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1876{
1877    __cxx_atomic_init(&__o->__a_, __d);
1878}
1879
1880template <class _Tp>
1881_LIBCPP_INLINE_VISIBILITY
1882void
1883atomic_init(atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1884{
1885    __cxx_atomic_init(&__o->__a_, __d);
1886}
1887
1888// atomic_store
1889
1890template <class _Tp>
1891_LIBCPP_INLINE_VISIBILITY
1892void
1893atomic_store(volatile atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1894{
1895    __o->store(__d);
1896}
1897
1898template <class _Tp>
1899_LIBCPP_INLINE_VISIBILITY
1900void
1901atomic_store(atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1902{
1903    __o->store(__d);
1904}
1905
1906// atomic_store_explicit
1907
1908template <class _Tp>
1909_LIBCPP_INLINE_VISIBILITY
1910void
1911atomic_store_explicit(volatile atomic<_Tp>* __o, _Tp __d, memory_order __m) _NOEXCEPT
1912  _LIBCPP_CHECK_STORE_MEMORY_ORDER(__m)
1913{
1914    __o->store(__d, __m);
1915}
1916
1917template <class _Tp>
1918_LIBCPP_INLINE_VISIBILITY
1919void
1920atomic_store_explicit(atomic<_Tp>* __o, _Tp __d, memory_order __m) _NOEXCEPT
1921  _LIBCPP_CHECK_STORE_MEMORY_ORDER(__m)
1922{
1923    __o->store(__d, __m);
1924}
1925
1926// atomic_load
1927
1928template <class _Tp>
1929_LIBCPP_INLINE_VISIBILITY
1930_Tp
1931atomic_load(const volatile atomic<_Tp>* __o) _NOEXCEPT
1932{
1933    return __o->load();
1934}
1935
1936template <class _Tp>
1937_LIBCPP_INLINE_VISIBILITY
1938_Tp
1939atomic_load(const atomic<_Tp>* __o) _NOEXCEPT
1940{
1941    return __o->load();
1942}
1943
1944// atomic_load_explicit
1945
1946template <class _Tp>
1947_LIBCPP_INLINE_VISIBILITY
1948_Tp
1949atomic_load_explicit(const volatile atomic<_Tp>* __o, memory_order __m) _NOEXCEPT
1950  _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
1951{
1952    return __o->load(__m);
1953}
1954
1955template <class _Tp>
1956_LIBCPP_INLINE_VISIBILITY
1957_Tp
1958atomic_load_explicit(const atomic<_Tp>* __o, memory_order __m) _NOEXCEPT
1959  _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
1960{
1961    return __o->load(__m);
1962}
1963
1964// atomic_exchange
1965
1966template <class _Tp>
1967_LIBCPP_INLINE_VISIBILITY
1968_Tp
1969atomic_exchange(volatile atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1970{
1971    return __o->exchange(__d);
1972}
1973
1974template <class _Tp>
1975_LIBCPP_INLINE_VISIBILITY
1976_Tp
1977atomic_exchange(atomic<_Tp>* __o, _Tp __d) _NOEXCEPT
1978{
1979    return __o->exchange(__d);
1980}
1981
1982// atomic_exchange_explicit
1983
1984template <class _Tp>
1985_LIBCPP_INLINE_VISIBILITY
1986_Tp
1987atomic_exchange_explicit(volatile atomic<_Tp>* __o, _Tp __d, memory_order __m) _NOEXCEPT
1988{
1989    return __o->exchange(__d, __m);
1990}
1991
1992template <class _Tp>
1993_LIBCPP_INLINE_VISIBILITY
1994_Tp
1995atomic_exchange_explicit(atomic<_Tp>* __o, _Tp __d, memory_order __m) _NOEXCEPT
1996{
1997    return __o->exchange(__d, __m);
1998}
1999
2000// atomic_compare_exchange_weak
2001
2002template <class _Tp>
2003_LIBCPP_INLINE_VISIBILITY
2004bool
2005atomic_compare_exchange_weak(volatile atomic<_Tp>* __o, _Tp* __e, _Tp __d) _NOEXCEPT
2006{
2007    return __o->compare_exchange_weak(*__e, __d);
2008}
2009
2010template <class _Tp>
2011_LIBCPP_INLINE_VISIBILITY
2012bool
2013atomic_compare_exchange_weak(atomic<_Tp>* __o, _Tp* __e, _Tp __d) _NOEXCEPT
2014{
2015    return __o->compare_exchange_weak(*__e, __d);
2016}
2017
2018// atomic_compare_exchange_strong
2019
2020template <class _Tp>
2021_LIBCPP_INLINE_VISIBILITY
2022bool
2023atomic_compare_exchange_strong(volatile atomic<_Tp>* __o, _Tp* __e, _Tp __d) _NOEXCEPT
2024{
2025    return __o->compare_exchange_strong(*__e, __d);
2026}
2027
2028template <class _Tp>
2029_LIBCPP_INLINE_VISIBILITY
2030bool
2031atomic_compare_exchange_strong(atomic<_Tp>* __o, _Tp* __e, _Tp __d) _NOEXCEPT
2032{
2033    return __o->compare_exchange_strong(*__e, __d);
2034}
2035
2036// atomic_compare_exchange_weak_explicit
2037
2038template <class _Tp>
2039_LIBCPP_INLINE_VISIBILITY
2040bool
2041atomic_compare_exchange_weak_explicit(volatile atomic<_Tp>* __o, _Tp* __e,
2042                                      _Tp __d,
2043                                      memory_order __s, memory_order __f) _NOEXCEPT
2044  _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
2045{
2046    return __o->compare_exchange_weak(*__e, __d, __s, __f);
2047}
2048
2049template <class _Tp>
2050_LIBCPP_INLINE_VISIBILITY
2051bool
2052atomic_compare_exchange_weak_explicit(atomic<_Tp>* __o, _Tp* __e, _Tp __d,
2053                                      memory_order __s, memory_order __f) _NOEXCEPT
2054  _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
2055{
2056    return __o->compare_exchange_weak(*__e, __d, __s, __f);
2057}
2058
2059// atomic_compare_exchange_strong_explicit
2060
2061template <class _Tp>
2062_LIBCPP_INLINE_VISIBILITY
2063bool
2064atomic_compare_exchange_strong_explicit(volatile atomic<_Tp>* __o,
2065                                        _Tp* __e, _Tp __d,
2066                                        memory_order __s, memory_order __f) _NOEXCEPT
2067  _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
2068{
2069    return __o->compare_exchange_strong(*__e, __d, __s, __f);
2070}
2071
2072template <class _Tp>
2073_LIBCPP_INLINE_VISIBILITY
2074bool
2075atomic_compare_exchange_strong_explicit(atomic<_Tp>* __o, _Tp* __e,
2076                                        _Tp __d,
2077                                        memory_order __s, memory_order __f) _NOEXCEPT
2078  _LIBCPP_CHECK_EXCHANGE_MEMORY_ORDER(__s, __f)
2079{
2080    return __o->compare_exchange_strong(*__e, __d, __s, __f);
2081}
2082
2083// atomic_wait
2084
2085template <class _Tp>
2086_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2087void atomic_wait(const volatile atomic<_Tp>* __o,
2088                 typename atomic<_Tp>::value_type __v) _NOEXCEPT
2089{
2090    return __o->wait(__v);
2091}
2092
2093template <class _Tp>
2094_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2095void atomic_wait(const atomic<_Tp>* __o,
2096                 typename atomic<_Tp>::value_type __v) _NOEXCEPT
2097{
2098    return __o->wait(__v);
2099}
2100
2101// atomic_wait_explicit
2102
2103template <class _Tp>
2104_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2105void atomic_wait_explicit(const volatile atomic<_Tp>* __o,
2106                          typename atomic<_Tp>::value_type __v,
2107                          memory_order __m) _NOEXCEPT
2108  _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
2109{
2110    return __o->wait(__v, __m);
2111}
2112
2113template <class _Tp>
2114_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2115void atomic_wait_explicit(const atomic<_Tp>* __o,
2116                          typename atomic<_Tp>::value_type __v,
2117                          memory_order __m) _NOEXCEPT
2118  _LIBCPP_CHECK_LOAD_MEMORY_ORDER(__m)
2119{
2120    return __o->wait(__v, __m);
2121}
2122
2123// atomic_notify_one
2124
2125template <class _Tp>
2126_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2127void atomic_notify_one(volatile atomic<_Tp>* __o) _NOEXCEPT
2128{
2129    __o->notify_one();
2130}
2131template <class _Tp>
2132_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2133void atomic_notify_one(atomic<_Tp>* __o) _NOEXCEPT
2134{
2135    __o->notify_one();
2136}
2137
2138// atomic_notify_one
2139
2140template <class _Tp>
2141_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2142void atomic_notify_all(volatile atomic<_Tp>* __o) _NOEXCEPT
2143{
2144    __o->notify_all();
2145}
2146template <class _Tp>
2147_LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2148void atomic_notify_all(atomic<_Tp>* __o) _NOEXCEPT
2149{
2150    __o->notify_all();
2151}
2152
2153// atomic_fetch_add
2154
2155template <class _Tp>
2156_LIBCPP_INLINE_VISIBILITY
2157typename enable_if
2158<
2159    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2160    _Tp
2161>::type
2162atomic_fetch_add(volatile atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2163{
2164    return __o->fetch_add(__op);
2165}
2166
2167template <class _Tp>
2168_LIBCPP_INLINE_VISIBILITY
2169typename enable_if
2170<
2171    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2172    _Tp
2173>::type
2174atomic_fetch_add(atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2175{
2176    return __o->fetch_add(__op);
2177}
2178
2179template <class _Tp>
2180_LIBCPP_INLINE_VISIBILITY
2181_Tp*
2182atomic_fetch_add(volatile atomic<_Tp*>* __o, ptrdiff_t __op) _NOEXCEPT
2183{
2184    return __o->fetch_add(__op);
2185}
2186
2187template <class _Tp>
2188_LIBCPP_INLINE_VISIBILITY
2189_Tp*
2190atomic_fetch_add(atomic<_Tp*>* __o, ptrdiff_t __op) _NOEXCEPT
2191{
2192    return __o->fetch_add(__op);
2193}
2194
2195// atomic_fetch_add_explicit
2196
2197template <class _Tp>
2198_LIBCPP_INLINE_VISIBILITY
2199typename enable_if
2200<
2201    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2202    _Tp
2203>::type
2204atomic_fetch_add_explicit(volatile atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2205{
2206    return __o->fetch_add(__op, __m);
2207}
2208
2209template <class _Tp>
2210_LIBCPP_INLINE_VISIBILITY
2211typename enable_if
2212<
2213    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2214    _Tp
2215>::type
2216atomic_fetch_add_explicit(atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2217{
2218    return __o->fetch_add(__op, __m);
2219}
2220
2221template <class _Tp>
2222_LIBCPP_INLINE_VISIBILITY
2223_Tp*
2224atomic_fetch_add_explicit(volatile atomic<_Tp*>* __o, ptrdiff_t __op,
2225                          memory_order __m) _NOEXCEPT
2226{
2227    return __o->fetch_add(__op, __m);
2228}
2229
2230template <class _Tp>
2231_LIBCPP_INLINE_VISIBILITY
2232_Tp*
2233atomic_fetch_add_explicit(atomic<_Tp*>* __o, ptrdiff_t __op, memory_order __m) _NOEXCEPT
2234{
2235    return __o->fetch_add(__op, __m);
2236}
2237
2238// atomic_fetch_sub
2239
2240template <class _Tp>
2241_LIBCPP_INLINE_VISIBILITY
2242typename enable_if
2243<
2244    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2245    _Tp
2246>::type
2247atomic_fetch_sub(volatile atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2248{
2249    return __o->fetch_sub(__op);
2250}
2251
2252template <class _Tp>
2253_LIBCPP_INLINE_VISIBILITY
2254typename enable_if
2255<
2256    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2257    _Tp
2258>::type
2259atomic_fetch_sub(atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2260{
2261    return __o->fetch_sub(__op);
2262}
2263
2264template <class _Tp>
2265_LIBCPP_INLINE_VISIBILITY
2266_Tp*
2267atomic_fetch_sub(volatile atomic<_Tp*>* __o, ptrdiff_t __op) _NOEXCEPT
2268{
2269    return __o->fetch_sub(__op);
2270}
2271
2272template <class _Tp>
2273_LIBCPP_INLINE_VISIBILITY
2274_Tp*
2275atomic_fetch_sub(atomic<_Tp*>* __o, ptrdiff_t __op) _NOEXCEPT
2276{
2277    return __o->fetch_sub(__op);
2278}
2279
2280// atomic_fetch_sub_explicit
2281
2282template <class _Tp>
2283_LIBCPP_INLINE_VISIBILITY
2284typename enable_if
2285<
2286    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2287    _Tp
2288>::type
2289atomic_fetch_sub_explicit(volatile atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2290{
2291    return __o->fetch_sub(__op, __m);
2292}
2293
2294template <class _Tp>
2295_LIBCPP_INLINE_VISIBILITY
2296typename enable_if
2297<
2298    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2299    _Tp
2300>::type
2301atomic_fetch_sub_explicit(atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2302{
2303    return __o->fetch_sub(__op, __m);
2304}
2305
2306template <class _Tp>
2307_LIBCPP_INLINE_VISIBILITY
2308_Tp*
2309atomic_fetch_sub_explicit(volatile atomic<_Tp*>* __o, ptrdiff_t __op,
2310                          memory_order __m) _NOEXCEPT
2311{
2312    return __o->fetch_sub(__op, __m);
2313}
2314
2315template <class _Tp>
2316_LIBCPP_INLINE_VISIBILITY
2317_Tp*
2318atomic_fetch_sub_explicit(atomic<_Tp*>* __o, ptrdiff_t __op, memory_order __m) _NOEXCEPT
2319{
2320    return __o->fetch_sub(__op, __m);
2321}
2322
2323// atomic_fetch_and
2324
2325template <class _Tp>
2326_LIBCPP_INLINE_VISIBILITY
2327typename enable_if
2328<
2329    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2330    _Tp
2331>::type
2332atomic_fetch_and(volatile atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2333{
2334    return __o->fetch_and(__op);
2335}
2336
2337template <class _Tp>
2338_LIBCPP_INLINE_VISIBILITY
2339typename enable_if
2340<
2341    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2342    _Tp
2343>::type
2344atomic_fetch_and(atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2345{
2346    return __o->fetch_and(__op);
2347}
2348
2349// atomic_fetch_and_explicit
2350
2351template <class _Tp>
2352_LIBCPP_INLINE_VISIBILITY
2353typename enable_if
2354<
2355    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2356    _Tp
2357>::type
2358atomic_fetch_and_explicit(volatile atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2359{
2360    return __o->fetch_and(__op, __m);
2361}
2362
2363template <class _Tp>
2364_LIBCPP_INLINE_VISIBILITY
2365typename enable_if
2366<
2367    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2368    _Tp
2369>::type
2370atomic_fetch_and_explicit(atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2371{
2372    return __o->fetch_and(__op, __m);
2373}
2374
2375// atomic_fetch_or
2376
2377template <class _Tp>
2378_LIBCPP_INLINE_VISIBILITY
2379typename enable_if
2380<
2381    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2382    _Tp
2383>::type
2384atomic_fetch_or(volatile atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2385{
2386    return __o->fetch_or(__op);
2387}
2388
2389template <class _Tp>
2390_LIBCPP_INLINE_VISIBILITY
2391typename enable_if
2392<
2393    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2394    _Tp
2395>::type
2396atomic_fetch_or(atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2397{
2398    return __o->fetch_or(__op);
2399}
2400
2401// atomic_fetch_or_explicit
2402
2403template <class _Tp>
2404_LIBCPP_INLINE_VISIBILITY
2405typename enable_if
2406<
2407    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2408    _Tp
2409>::type
2410atomic_fetch_or_explicit(volatile atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2411{
2412    return __o->fetch_or(__op, __m);
2413}
2414
2415template <class _Tp>
2416_LIBCPP_INLINE_VISIBILITY
2417typename enable_if
2418<
2419    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2420    _Tp
2421>::type
2422atomic_fetch_or_explicit(atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2423{
2424    return __o->fetch_or(__op, __m);
2425}
2426
2427// atomic_fetch_xor
2428
2429template <class _Tp>
2430_LIBCPP_INLINE_VISIBILITY
2431typename enable_if
2432<
2433    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2434    _Tp
2435>::type
2436atomic_fetch_xor(volatile atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2437{
2438    return __o->fetch_xor(__op);
2439}
2440
2441template <class _Tp>
2442_LIBCPP_INLINE_VISIBILITY
2443typename enable_if
2444<
2445    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2446    _Tp
2447>::type
2448atomic_fetch_xor(atomic<_Tp>* __o, _Tp __op) _NOEXCEPT
2449{
2450    return __o->fetch_xor(__op);
2451}
2452
2453// atomic_fetch_xor_explicit
2454
2455template <class _Tp>
2456_LIBCPP_INLINE_VISIBILITY
2457typename enable_if
2458<
2459    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2460    _Tp
2461>::type
2462atomic_fetch_xor_explicit(volatile atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2463{
2464    return __o->fetch_xor(__op, __m);
2465}
2466
2467template <class _Tp>
2468_LIBCPP_INLINE_VISIBILITY
2469typename enable_if
2470<
2471    is_integral<_Tp>::value && !is_same<_Tp, bool>::value,
2472    _Tp
2473>::type
2474atomic_fetch_xor_explicit(atomic<_Tp>* __o, _Tp __op, memory_order __m) _NOEXCEPT
2475{
2476    return __o->fetch_xor(__op, __m);
2477}
2478
2479// flag type and operations
2480
2481typedef struct atomic_flag
2482{
2483    __cxx_atomic_impl<_LIBCPP_ATOMIC_FLAG_TYPE> __a_;
2484
2485    _LIBCPP_INLINE_VISIBILITY
2486    bool test(memory_order __m = memory_order_seq_cst) const volatile _NOEXCEPT
2487        {return _LIBCPP_ATOMIC_FLAG_TYPE(true) == __cxx_atomic_load(&__a_, __m);}
2488    _LIBCPP_INLINE_VISIBILITY
2489    bool test(memory_order __m = memory_order_seq_cst) const _NOEXCEPT
2490        {return _LIBCPP_ATOMIC_FLAG_TYPE(true) == __cxx_atomic_load(&__a_, __m);}
2491
2492    _LIBCPP_INLINE_VISIBILITY
2493    bool test_and_set(memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
2494        {return __cxx_atomic_exchange(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(true), __m);}
2495    _LIBCPP_INLINE_VISIBILITY
2496    bool test_and_set(memory_order __m = memory_order_seq_cst) _NOEXCEPT
2497        {return __cxx_atomic_exchange(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(true), __m);}
2498    _LIBCPP_INLINE_VISIBILITY
2499    void clear(memory_order __m = memory_order_seq_cst) volatile _NOEXCEPT
2500        {__cxx_atomic_store(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(false), __m);}
2501    _LIBCPP_INLINE_VISIBILITY
2502    void clear(memory_order __m = memory_order_seq_cst) _NOEXCEPT
2503        {__cxx_atomic_store(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(false), __m);}
2504
2505    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2506    void wait(bool __v, memory_order __m = memory_order_seq_cst) const volatile _NOEXCEPT
2507        {__cxx_atomic_wait(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(__v), __m);}
2508    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2509    void wait(bool __v, memory_order __m = memory_order_seq_cst) const _NOEXCEPT
2510        {__cxx_atomic_wait(&__a_, _LIBCPP_ATOMIC_FLAG_TYPE(__v), __m);}
2511    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2512    void notify_one() volatile _NOEXCEPT
2513        {__cxx_atomic_notify_one(&__a_);}
2514    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2515    void notify_one() _NOEXCEPT
2516        {__cxx_atomic_notify_one(&__a_);}
2517    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2518    void notify_all() volatile _NOEXCEPT
2519        {__cxx_atomic_notify_all(&__a_);}
2520    _LIBCPP_AVAILABILITY_SYNC _LIBCPP_INLINE_VISIBILITY
2521    void notify_all() _NOEXCEPT
2522        {__cxx_atomic_notify_all(&__a_);}
2523
2524    _LIBCPP_INLINE_VISIBILITY
2525    atomic_flag() _NOEXCEPT _LIBCPP_DEFAULT
2526
2527    _LIBCPP_INLINE_VISIBILITY _LIBCPP_CONSTEXPR
2528    atomic_flag(bool __b) _NOEXCEPT : __a_(__b) {} // EXTENSION
2529
2530#ifndef _LIBCPP_CXX03_LANG
2531    atomic_flag(const atomic_flag&) = delete;
2532    atomic_flag& operator=(const atomic_flag&) = delete;
2533    atomic_flag& operator=(const atomic_flag&) volatile = delete;
2534#else
2535private:
2536    _LIBCPP_INLINE_VISIBILITY
2537    atomic_flag(const atomic_flag&);
2538    _LIBCPP_INLINE_VISIBILITY
2539    atomic_flag& operator=(const atomic_flag&);
2540    _LIBCPP_INLINE_VISIBILITY
2541    atomic_flag& operator=(const atomic_flag&) volatile;
2542#endif
2543} atomic_flag;
2544
2545
2546inline _LIBCPP_INLINE_VISIBILITY
2547bool
2548atomic_flag_test(const volatile atomic_flag* __o) _NOEXCEPT
2549{
2550    return __o->test();
2551}
2552
2553inline _LIBCPP_INLINE_VISIBILITY
2554bool
2555atomic_flag_test(const atomic_flag* __o) _NOEXCEPT
2556{
2557    return __o->test();
2558}
2559
2560inline _LIBCPP_INLINE_VISIBILITY
2561bool
2562atomic_flag_test_explicit(const volatile atomic_flag* __o, memory_order __m) _NOEXCEPT
2563{
2564    return __o->test(__m);
2565}
2566
2567inline _LIBCPP_INLINE_VISIBILITY
2568bool
2569atomic_flag_test_explicit(const atomic_flag* __o, memory_order __m) _NOEXCEPT
2570{
2571    return __o->test(__m);
2572}
2573
2574inline _LIBCPP_INLINE_VISIBILITY
2575bool
2576atomic_flag_test_and_set(volatile atomic_flag* __o) _NOEXCEPT
2577{
2578    return __o->test_and_set();
2579}
2580
2581inline _LIBCPP_INLINE_VISIBILITY
2582bool
2583atomic_flag_test_and_set(atomic_flag* __o) _NOEXCEPT
2584{
2585    return __o->test_and_set();
2586}
2587
2588inline _LIBCPP_INLINE_VISIBILITY
2589bool
2590atomic_flag_test_and_set_explicit(volatile atomic_flag* __o, memory_order __m) _NOEXCEPT
2591{
2592    return __o->test_and_set(__m);
2593}
2594
2595inline _LIBCPP_INLINE_VISIBILITY
2596bool
2597atomic_flag_test_and_set_explicit(atomic_flag* __o, memory_order __m) _NOEXCEPT
2598{
2599    return __o->test_and_set(__m);
2600}
2601
2602inline _LIBCPP_INLINE_VISIBILITY
2603void
2604atomic_flag_clear(volatile atomic_flag* __o) _NOEXCEPT
2605{
2606    __o->clear();
2607}
2608
2609inline _LIBCPP_INLINE_VISIBILITY
2610void
2611atomic_flag_clear(atomic_flag* __o) _NOEXCEPT
2612{
2613    __o->clear();
2614}
2615
2616inline _LIBCPP_INLINE_VISIBILITY
2617void
2618atomic_flag_clear_explicit(volatile atomic_flag* __o, memory_order __m) _NOEXCEPT
2619{
2620    __o->clear(__m);
2621}
2622
2623inline _LIBCPP_INLINE_VISIBILITY
2624void
2625atomic_flag_clear_explicit(atomic_flag* __o, memory_order __m) _NOEXCEPT
2626{
2627    __o->clear(__m);
2628}
2629
2630inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2631void
2632atomic_flag_wait(const volatile atomic_flag* __o, bool __v) _NOEXCEPT
2633{
2634    __o->wait(__v);
2635}
2636
2637inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2638void
2639atomic_flag_wait(const atomic_flag* __o, bool __v) _NOEXCEPT
2640{
2641    __o->wait(__v);
2642}
2643
2644inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2645void
2646atomic_flag_wait_explicit(const volatile atomic_flag* __o,
2647                          bool __v, memory_order __m) _NOEXCEPT
2648{
2649    __o->wait(__v, __m);
2650}
2651
2652inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2653void
2654atomic_flag_wait_explicit(const atomic_flag* __o,
2655                          bool __v, memory_order __m) _NOEXCEPT
2656{
2657    __o->wait(__v, __m);
2658}
2659
2660inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2661void
2662atomic_flag_notify_one(volatile atomic_flag* __o) _NOEXCEPT
2663{
2664    __o->notify_one();
2665}
2666
2667inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2668void
2669atomic_flag_notify_one(atomic_flag* __o) _NOEXCEPT
2670{
2671    __o->notify_one();
2672}
2673
2674inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2675void
2676atomic_flag_notify_all(volatile atomic_flag* __o) _NOEXCEPT
2677{
2678    __o->notify_all();
2679}
2680
2681inline _LIBCPP_INLINE_VISIBILITY _LIBCPP_AVAILABILITY_SYNC
2682void
2683atomic_flag_notify_all(atomic_flag* __o) _NOEXCEPT
2684{
2685    __o->notify_all();
2686}
2687
2688// fences
2689
2690inline _LIBCPP_INLINE_VISIBILITY
2691void
2692atomic_thread_fence(memory_order __m) _NOEXCEPT
2693{
2694    __cxx_atomic_thread_fence(__m);
2695}
2696
2697inline _LIBCPP_INLINE_VISIBILITY
2698void
2699atomic_signal_fence(memory_order __m) _NOEXCEPT
2700{
2701    __cxx_atomic_signal_fence(__m);
2702}
2703
2704// Atomics for standard typedef types
2705
2706typedef atomic<bool>               atomic_bool;
2707typedef atomic<char>               atomic_char;
2708typedef atomic<signed char>        atomic_schar;
2709typedef atomic<unsigned char>      atomic_uchar;
2710typedef atomic<short>              atomic_short;
2711typedef atomic<unsigned short>     atomic_ushort;
2712typedef atomic<int>                atomic_int;
2713typedef atomic<unsigned int>       atomic_uint;
2714typedef atomic<long>               atomic_long;
2715typedef atomic<unsigned long>      atomic_ulong;
2716typedef atomic<long long>          atomic_llong;
2717typedef atomic<unsigned long long> atomic_ullong;
2718typedef atomic<char16_t>           atomic_char16_t;
2719typedef atomic<char32_t>           atomic_char32_t;
2720typedef atomic<wchar_t>            atomic_wchar_t;
2721
2722typedef atomic<int_least8_t>   atomic_int_least8_t;
2723typedef atomic<uint_least8_t>  atomic_uint_least8_t;
2724typedef atomic<int_least16_t>  atomic_int_least16_t;
2725typedef atomic<uint_least16_t> atomic_uint_least16_t;
2726typedef atomic<int_least32_t>  atomic_int_least32_t;
2727typedef atomic<uint_least32_t> atomic_uint_least32_t;
2728typedef atomic<int_least64_t>  atomic_int_least64_t;
2729typedef atomic<uint_least64_t> atomic_uint_least64_t;
2730
2731typedef atomic<int_fast8_t>   atomic_int_fast8_t;
2732typedef atomic<uint_fast8_t>  atomic_uint_fast8_t;
2733typedef atomic<int_fast16_t>  atomic_int_fast16_t;
2734typedef atomic<uint_fast16_t> atomic_uint_fast16_t;
2735typedef atomic<int_fast32_t>  atomic_int_fast32_t;
2736typedef atomic<uint_fast32_t> atomic_uint_fast32_t;
2737typedef atomic<int_fast64_t>  atomic_int_fast64_t;
2738typedef atomic<uint_fast64_t> atomic_uint_fast64_t;
2739
2740typedef atomic< int8_t>  atomic_int8_t;
2741typedef atomic<uint8_t>  atomic_uint8_t;
2742typedef atomic< int16_t> atomic_int16_t;
2743typedef atomic<uint16_t> atomic_uint16_t;
2744typedef atomic< int32_t> atomic_int32_t;
2745typedef atomic<uint32_t> atomic_uint32_t;
2746typedef atomic< int64_t> atomic_int64_t;
2747typedef atomic<uint64_t> atomic_uint64_t;
2748
2749typedef atomic<intptr_t>  atomic_intptr_t;
2750typedef atomic<uintptr_t> atomic_uintptr_t;
2751typedef atomic<size_t>    atomic_size_t;
2752typedef atomic<ptrdiff_t> atomic_ptrdiff_t;
2753typedef atomic<intmax_t>  atomic_intmax_t;
2754typedef atomic<uintmax_t> atomic_uintmax_t;
2755
2756// atomic_*_lock_free : prefer the contention type most highly, then the largest lock-free type
2757
2758#ifdef __cpp_lib_atomic_is_always_lock_free
2759# define _LIBCPP_CONTENTION_LOCK_FREE __atomic_always_lock_free(sizeof(__cxx_contention_t), 0)
2760#else
2761# define _LIBCPP_CONTENTION_LOCK_FREE false
2762#endif
2763
2764#if ATOMIC_LLONG_LOCK_FREE == 2
2765typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, long long>::type          __libcpp_signed_lock_free;
2766typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, unsigned long long>::type __libcpp_unsigned_lock_free;
2767#elif ATOMIC_INT_LOCK_FREE == 2
2768typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, int>::type                __libcpp_signed_lock_free;
2769typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, unsigned int>::type       __libcpp_unsigned_lock_free;
2770#elif ATOMIC_SHORT_LOCK_FREE == 2
2771typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, short>::type              __libcpp_signed_lock_free;
2772typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, unsigned short>::type     __libcpp_unsigned_lock_free;
2773#elif ATOMIC_CHAR_LOCK_FREE == 2
2774typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, char>::type               __libcpp_signed_lock_free;
2775typedef conditional<_LIBCPP_CONTENTION_LOCK_FREE, __cxx_contention_t, unsigned char>::type      __libcpp_unsigned_lock_free;
2776#else
2777    // No signed/unsigned lock-free types
2778#endif
2779
2780typedef atomic<__libcpp_signed_lock_free> atomic_signed_lock_free;
2781typedef atomic<__libcpp_unsigned_lock_free> atomic_unsigned_lock_free;
2782
2783#define ATOMIC_FLAG_INIT {false}
2784#define ATOMIC_VAR_INIT(__v) {__v}
2785
2786_LIBCPP_END_NAMESPACE_STD
2787
2788#endif  // _LIBCPP_ATOMIC
2789