Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/contrib/llvm-project/libcxx/src/include/atomic_support.h
35230 views
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
134