xref: /freebsd/contrib/llvm-project/libcxx/src/include/atomic_support.h (revision b2d2a78ad80ec68d4a17f5aef97d21686cb1e29b)
1 //===----------------------------------------------------------------------===////
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===////
8 
9 #ifndef ATOMIC_SUPPORT_H
10 #define ATOMIC_SUPPORT_H
11 
12 #include <__config>
13 #include <memory> // for __libcpp_relaxed_load
14 
15 #if defined(__clang__) && __has_builtin(__atomic_load_n) && __has_builtin(__atomic_store_n) &&                         \
16     __has_builtin(__atomic_add_fetch) && __has_builtin(__atomic_exchange_n) &&                                         \
17     __has_builtin(__atomic_compare_exchange_n) && defined(__ATOMIC_RELAXED) && defined(__ATOMIC_CONSUME) &&            \
18     defined(__ATOMIC_ACQUIRE) && defined(__ATOMIC_RELEASE) && defined(__ATOMIC_ACQ_REL) && defined(__ATOMIC_SEQ_CST)
19 #  define _LIBCPP_HAS_ATOMIC_BUILTINS
20 #elif defined(_LIBCPP_COMPILER_GCC)
21 #  define _LIBCPP_HAS_ATOMIC_BUILTINS
22 #endif
23 
24 #if !defined(_LIBCPP_HAS_ATOMIC_BUILTINS) && !defined(_LIBCPP_HAS_NO_THREADS)
25 #  if defined(_LIBCPP_WARNING)
26 _LIBCPP_WARNING("Building libc++ without __atomic builtins is unsupported")
27 #  else
28 #    warning Building libc++ without __atomic builtins is unsupported
29 #  endif
30 #endif
31 
32 _LIBCPP_BEGIN_NAMESPACE_STD
33 
34 namespace {
35 
36 #if defined(_LIBCPP_HAS_ATOMIC_BUILTINS) && !defined(_LIBCPP_HAS_NO_THREADS)
37 
38 enum __libcpp_atomic_order {
39   _AO_Relaxed = __ATOMIC_RELAXED,
40   _AO_Consume = __ATOMIC_CONSUME,
41   _AO_Acquire = __ATOMIC_ACQUIRE,
42   _AO_Release = __ATOMIC_RELEASE,
43   _AO_Acq_Rel = __ATOMIC_ACQ_REL,
44   _AO_Seq     = __ATOMIC_SEQ_CST
45 };
46 
47 template <class _ValueType, class _FromType>
48 inline _LIBCPP_HIDE_FROM_ABI void __libcpp_atomic_store(_ValueType* __dest, _FromType __val, int __order = _AO_Seq) {
49   __atomic_store_n(__dest, __val, __order);
50 }
51 
52 template <class _ValueType, class _FromType>
53 inline _LIBCPP_HIDE_FROM_ABI void __libcpp_relaxed_store(_ValueType* __dest, _FromType __val) {
54   __atomic_store_n(__dest, __val, _AO_Relaxed);
55 }
56 
57 template <class _ValueType>
58 inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_load(_ValueType const* __val, int __order = _AO_Seq) {
59   return __atomic_load_n(__val, __order);
60 }
61 
62 template <class _ValueType, class _AddType>
63 inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_add(_ValueType* __val, _AddType __a, int __order = _AO_Seq) {
64   return __atomic_add_fetch(__val, __a, __order);
65 }
66 
67 template <class _ValueType>
68 inline _LIBCPP_HIDE_FROM_ABI _ValueType
69 __libcpp_atomic_exchange(_ValueType* __target, _ValueType __value, int __order = _AO_Seq) {
70   return __atomic_exchange_n(__target, __value, __order);
71 }
72 
73 template <class _ValueType>
74 inline _LIBCPP_HIDE_FROM_ABI bool __libcpp_atomic_compare_exchange(
75     _ValueType* __val,
76     _ValueType* __expected,
77     _ValueType __after,
78     int __success_order = _AO_Seq,
79     int __fail_order    = _AO_Seq) {
80   return __atomic_compare_exchange_n(__val, __expected, __after, true, __success_order, __fail_order);
81 }
82 
83 #else // _LIBCPP_HAS_NO_THREADS
84 
85 enum __libcpp_atomic_order { _AO_Relaxed, _AO_Consume, _AO_Acquire, _AO_Release, _AO_Acq_Rel, _AO_Seq };
86 
87 template <class _ValueType, class _FromType>
88 inline _LIBCPP_HIDE_FROM_ABI void __libcpp_atomic_store(_ValueType* __dest, _FromType __val, int = 0) {
89   *__dest = __val;
90 }
91 
92 template <class _ValueType, class _FromType>
93 inline _LIBCPP_HIDE_FROM_ABI void __libcpp_relaxed_store(_ValueType* __dest, _FromType __val) {
94   *__dest = __val;
95 }
96 
97 template <class _ValueType>
98 inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_load(_ValueType const* __val, int = 0) {
99   return *__val;
100 }
101 
102 template <class _ValueType, class _AddType>
103 inline _LIBCPP_HIDE_FROM_ABI _ValueType __libcpp_atomic_add(_ValueType* __val, _AddType __a, int = 0) {
104   return *__val += __a;
105 }
106 
107 template <class _ValueType>
108 inline _LIBCPP_HIDE_FROM_ABI _ValueType
109 __libcpp_atomic_exchange(_ValueType* __target, _ValueType __value, int = _AO_Seq) {
110   _ValueType old = *__target;
111   *__target      = __value;
112   return old;
113 }
114 
115 template <class _ValueType>
116 inline _LIBCPP_HIDE_FROM_ABI bool
117 __libcpp_atomic_compare_exchange(_ValueType* __val, _ValueType* __expected, _ValueType __after, int = 0, int = 0) {
118   if (*__val == *__expected) {
119     *__val = __after;
120     return true;
121   }
122   *__expected = *__val;
123   return false;
124 }
125 
126 #endif // _LIBCPP_HAS_NO_THREADS
127 
128 } // end namespace
129 
130 _LIBCPP_END_NAMESPACE_STD
131 
132 #endif // ATOMIC_SUPPORT_H
133