38#ifndef _GLIBCXX_BARRIER
39#define _GLIBCXX_BARRIER 1
41#pragma GCC system_header
45#define __glibcxx_want_barrier
48#ifdef __cpp_lib_barrier
55namespace std _GLIBCXX_VISIBILITY(default)
57_GLIBCXX_BEGIN_NAMESPACE_VERSION
59 struct __empty_completion
61 _GLIBCXX_ALWAYS_INLINE
void
80 enum class __barrier_phase_t :
unsigned char { };
82 template<
typename _CompletionF>
85 using __atomic_phase_ref_t = std::__atomic_ref<__barrier_phase_t>;
86 using __atomic_phase_const_ref_t = std::__atomic_ref<const __barrier_phase_t>;
87 static constexpr auto __phase_alignment =
88 __atomic_phase_ref_t::required_alignment;
91 struct alignas(64) __state_t
93 alignas(__phase_alignment) __tickets_t __tickets;
96 ptrdiff_t _M_expected;
97 unique_ptr<__state_t[]> _M_state;
98 __atomic_base<ptrdiff_t> _M_expected_adjustment;
99 _CompletionF _M_completion;
101 alignas(__phase_alignment) __barrier_phase_t _M_phase;
104 _M_arrive(__barrier_phase_t __old_phase,
size_t __current)
106 const auto __old_phase_val =
static_cast<unsigned char>(__old_phase);
107 const auto __half_step =
108 static_cast<__barrier_phase_t
>(__old_phase_val + 1);
109 const auto __full_step =
110 static_cast<__barrier_phase_t
>(__old_phase_val + 2);
112 size_t __current_expected = _M_expected;
113 __current %= ((_M_expected + 1) >> 1);
115 for (
int __round = 0; ; ++__round)
117 if (__current_expected <= 1)
119 size_t const __end_node = ((__current_expected + 1) >> 1),
120 __last_node = __end_node - 1;
121 for ( ; ; ++__current)
123 if (__current == __end_node)
125 auto __expect = __old_phase;
126 __atomic_phase_ref_t __phase(_M_state[__current]
127 .__tickets[__round]);
128 if (__current == __last_node && (__current_expected & 1))
130 if (__phase.compare_exchange_strong(__expect, __full_step,
131 memory_order_acq_rel))
134 else if (__phase.compare_exchange_strong(__expect, __half_step,
135 memory_order_acq_rel))
139 else if (__expect == __half_step)
141 if (__phase.compare_exchange_strong(__expect, __full_step,
142 memory_order_acq_rel))
146 __current_expected = __last_node + 1;
152 using arrival_token = __barrier_phase_t;
154 static constexpr ptrdiff_t
156 {
return __PTRDIFF_MAX__; }
158 __tree_barrier(ptrdiff_t __expected, _CompletionF __completion)
159 : _M_expected(__expected), _M_expected_adjustment(0),
160 _M_completion(
move(__completion)),
161 _M_phase(static_cast<__barrier_phase_t>(0))
163 size_t const __count = (_M_expected + 1) >> 1;
165 _M_state = std::make_unique<__state_t[]>(__count);
168 [[nodiscard]] arrival_token
169 arrive(ptrdiff_t __update)
173 __atomic_phase_ref_t __phase(_M_phase);
174 const auto __old_phase = __phase.load(memory_order_relaxed);
175 const auto __cur =
static_cast<unsigned char>(__old_phase);
176 for(; __update; --__update)
178 if(_M_arrive(__old_phase, __current))
181 _M_expected += _M_expected_adjustment.load(memory_order_relaxed);
182 _M_expected_adjustment.store(0, memory_order_relaxed);
183 auto __new_phase =
static_cast<__barrier_phase_t
>(__cur + 2);
184 __phase.store(__new_phase, memory_order_release);
185 __phase.notify_all();
192 wait(arrival_token&& __old_phase)
const
194 __atomic_phase_const_ref_t __phase(_M_phase);
195 auto const __test_fn = [=]
197 return __phase.load(memory_order_acquire) != __old_phase;
199 std::__atomic_wait_address(&_M_phase, __test_fn);
205 _M_expected_adjustment.fetch_sub(1, memory_order_relaxed);
210 template<
typename _CompletionF = __empty_completion>
215 using __algorithm_t = __tree_barrier<_CompletionF>;
219 class arrival_token final
222 arrival_token(arrival_token&&) =
default;
223 arrival_token& operator=(arrival_token&&) =
default;
224 ~arrival_token() =
default;
227 friend class barrier;
228 using __token =
typename __algorithm_t::arrival_token;
229 explicit arrival_token(__token __tok) noexcept : _M_tok(__tok) { }
233 static constexpr ptrdiff_t
235 {
return __algorithm_t::max(); }
238 barrier(ptrdiff_t __count, _CompletionF __completion = _CompletionF())
239 : _M_b(__count,
std::
move(__completion))
242 barrier(barrier
const&) =
delete;
243 barrier& operator=(barrier
const&) =
delete;
245 [[nodiscard]] arrival_token
246 arrive(ptrdiff_t __update = 1)
247 {
return arrival_token{_M_b.arrive(__update)}; }
250 wait(arrival_token&& __phase)
const
251 { _M_b.wait(
std::move(__phase._M_tok)); }
259 { _M_b.arrive_and_drop(); }
262_GLIBCXX_END_NAMESPACE_VERSION
constexpr std::remove_reference< _Tp >::type && move(_Tp &&__t) noexcept
Convert a value to an rvalue.
ISO C++ entities toplevel namespace is std.
thread::id get_id() noexcept
The unique identifier of the current thread.
A standard container for storing a fixed size sequence of elements.
Primary class template hash.