30 #ifndef _GLIBCXX_ATOMIC_TIMED_WAIT_H
31 #define _GLIBCXX_ATOMIC_TIMED_WAIT_H 1
33 #ifdef _GLIBCXX_SYSHDR
34 #pragma GCC system_header
39 #if __glibcxx_atomic_wait
43 #ifdef _GLIBCXX_HAVE_LINUX_FUTEX
47 namespace std _GLIBCXX_VISIBILITY(default)
49 _GLIBCXX_BEGIN_NAMESPACE_VERSION
53 using __wait_clock_t = chrono::steady_clock;
55 template<
typename _Clock,
typename _Dur>
56 __wait_clock_t::time_point
57 __to_wait_clock(
const chrono::time_point<_Clock, _Dur>& __atime) noexcept
59 const typename _Clock::time_point __c_entry = _Clock::now();
60 const __wait_clock_t::time_point __w_entry = __wait_clock_t::now();
61 const auto __delta = __atime - __c_entry;
62 using __w_dur =
typename __wait_clock_t::duration;
63 return __w_entry + chrono::ceil<__w_dur>(__delta);
66 template<
typename _Dur>
67 __wait_clock_t::time_point
68 __to_wait_clock(
const chrono::time_point<__wait_clock_t,
69 _Dur>& __atime) noexcept
71 using __w_dur =
typename __wait_clock_t::duration;
72 if constexpr (is_same_v<__w_dur, _Dur>)
75 return chrono::ceil<__w_dur>(__atime);
84 __wait_until_impl(
const void* __addr, __wait_args_base& __args,
87 template<
typename _Clock,
typename _Dur>
89 __wait_until(
const void* __addr, __wait_args_base& __args,
90 const chrono::time_point<_Clock, _Dur>& __atime) noexcept
92 auto __at = __detail::__to_wait_clock(__atime);
93 auto __res = __detail::__wait_until_impl(__addr, __args,
94 __at.time_since_epoch());
96 if constexpr (!is_same_v<__wait_clock_t, _Clock>)
102 if (_Clock::now() < __atime)
103 __res._M_timeout =
false;
108 template<
typename _Rep,
typename _Period>
110 __wait_for(
const void* __addr, __wait_args_base& __args,
111 const chrono::duration<_Rep, _Period>& __rtime) noexcept
113 if (!__rtime.count())
116 __args._M_flags |= __wait_flags::__do_spin | __wait_flags::__spin_only;
117 return __detail::__wait_impl(__addr, __args);
120 auto const __reltime = chrono::ceil<__wait_clock_t::duration>(__rtime);
121 auto const __atime = chrono::steady_clock::now() + __reltime;
122 return __detail::__wait_until(__addr, __args, __atime);
127 template<
typename _Tp,
128 typename _Pred,
typename _ValFn,
129 typename _Clock,
typename _Dur>
131 __atomic_wait_address_until(
const _Tp* __addr, _Pred&& __pred,
133 const chrono::time_point<_Clock, _Dur>& __atime,
134 bool __bare_wait =
false) noexcept
136 __detail::__wait_args __args{ __addr, __bare_wait };
137 _Tp __val = __args._M_setup_wait(__addr, __vfn);
138 while (!__pred(__val))
140 auto __res = __detail::__wait_until(__addr, __args, __atime);
141 if (__res._M_timeout)
143 __val = __args._M_on_wake(__addr, __vfn, __res);
148 template<
typename _Clock,
typename _Dur>
150 __atomic_wait_address_until_v(
const __detail::__platform_wait_t* __addr,
151 __detail::__platform_wait_t __old,
153 const chrono::time_point<_Clock, _Dur>& __atime,
154 bool __bare_wait =
false) noexcept
157 __glibcxx_assert(__platform_wait_uses_type<__detail::__platform_wait_t>);
159 __detail::__wait_args __args{ __addr, __old, __order, __bare_wait };
160 auto __res = __detail::__wait_until(__addr, __args, __atime);
161 return !__res._M_timeout;
164 template<
typename _Tp,
typename _ValFn,
165 typename _Clock,
typename _Dur>
167 __atomic_wait_address_until_v(
const _Tp* __addr, _Tp&& __old,
169 const chrono::time_point<_Clock, _Dur>& __atime,
170 bool __bare_wait =
false) noexcept
172 auto __pfn = [&](
const _Tp& __val) {
173 return !__detail::__atomic_eq(__old, __val);
175 return std::__atomic_wait_address_until(__addr, __pfn, __vfn, __atime,
179 template<
typename _Tp,
180 typename _Pred,
typename _ValFn,
181 typename _Rep,
typename _Period>
183 __atomic_wait_address_for(
const _Tp* __addr, _Pred&& __pred,
185 const chrono::duration<_Rep, _Period>& __rtime,
186 bool __bare_wait =
false) noexcept
188 __detail::__wait_args __args{ __addr, __bare_wait };
189 _Tp __val = __args._M_setup_wait(__addr, __vfn);
190 while (!__pred(__val))
192 auto __res = __detail::__wait_for(__addr, __args, __rtime);
193 if (__res._M_timeout)
195 __val = __args._M_on_wake(__addr, __vfn, __res);
200 template<
typename _Rep,
typename _Period>
202 __atomic_wait_address_for_v(
const __detail::__platform_wait_t* __addr,
203 __detail::__platform_wait_t __old,
205 const chrono::duration<_Rep, _Period>& __rtime,
206 bool __bare_wait =
false) noexcept
209 __glibcxx_assert(__platform_wait_uses_type<__detail::__platform_wait_t>);
211 __detail::__wait_args __args{ __addr, __old, __order, __bare_wait };
212 auto __res = __detail::__wait_for(__addr, __args, __rtime);
213 return !__res._M_timeout;
216 template<
typename _Tp,
typename _ValFn,
217 typename _Rep,
typename _Period>
219 __atomic_wait_address_for_v(
const _Tp* __addr, _Tp&& __old, _ValFn&& __vfn,
220 const chrono::duration<_Rep, _Period>& __rtime,
221 bool __bare_wait =
false) noexcept
223 auto __pfn = [&](
const _Tp& __val) {
224 return !__detail::__atomic_eq(__old, __val);
226 return __atomic_wait_address_for(__addr, __pfn, forward<_ValFn>(__vfn),
227 __rtime, __bare_wait);
229 _GLIBCXX_END_NAMESPACE_VERSION
231 #endif // __cpp_lib_atomic_wait
232 #endif // _GLIBCXX_ATOMIC_TIMED_WAIT_H