2 //===----------------------------------------------------------------------===//
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
8 //===----------------------------------------------------------------------===//
10 #ifndef _LIBCPP_CONDITION_VARIABLE
11 #define _LIBCPP_CONDITION_VARIABLE
14 condition_variable synopsis
19 enum class cv_status { no_timeout, timeout };
21 class condition_variable
25 ~condition_variable();
27 condition_variable(const condition_variable&) = delete;
28 condition_variable& operator=(const condition_variable&) = delete;
30 void notify_one() noexcept;
31 void notify_all() noexcept;
33 void wait(unique_lock<mutex>& lock);
34 template <class Predicate>
35 void wait(unique_lock<mutex>& lock, Predicate pred);
37 template <class Clock, class Duration>
39 wait_until(unique_lock<mutex>& lock,
40 const chrono::time_point<Clock, Duration>& abs_time);
42 template <class Clock, class Duration, class Predicate>
44 wait_until(unique_lock<mutex>& lock,
45 const chrono::time_point<Clock, Duration>& abs_time,
48 template <class Rep, class Period>
50 wait_for(unique_lock<mutex>& lock,
51 const chrono::duration<Rep, Period>& rel_time);
53 template <class Rep, class Period, class Predicate>
55 wait_for(unique_lock<mutex>& lock,
56 const chrono::duration<Rep, Period>& rel_time,
59 typedef pthread_cond_t* native_handle_type;
60 native_handle_type native_handle();
63 void notify_all_at_thread_exit(condition_variable& cond, unique_lock<mutex> lk);
65 class condition_variable_any
68 condition_variable_any();
69 ~condition_variable_any();
71 condition_variable_any(const condition_variable_any&) = delete;
72 condition_variable_any& operator=(const condition_variable_any&) = delete;
74 void notify_one() noexcept;
75 void notify_all() noexcept;
78 void wait(Lock& lock);
79 template <class Lock, class Predicate>
80 void wait(Lock& lock, Predicate pred);
82 template <class Lock, class Clock, class Duration>
84 wait_until(Lock& lock,
85 const chrono::time_point<Clock, Duration>& abs_time);
87 template <class Lock, class Clock, class Duration, class Predicate>
89 wait_until(Lock& lock,
90 const chrono::time_point<Clock, Duration>& abs_time,
93 template <class Lock, class Rep, class Period>
96 const chrono::duration<Rep, Period>& rel_time);
98 template <class Lock, class Rep, class Period, class Predicate>
101 const chrono::duration<Rep, Period>& rel_time,
109 #include <__assert> // all public C++ headers provide the assertion handler
110 #include <__chrono/duration.h>
111 #include <__chrono/steady_clock.h>
112 #include <__chrono/time_point.h>
113 #include <__condition_variable/condition_variable.h>
115 #include <__memory/shared_ptr.h>
116 #include <__memory/unique_ptr.h>
117 #include <__mutex/lock_guard.h>
118 #include <__mutex/mutex.h>
119 #include <__mutex/tag_types.h>
120 #include <__mutex/unique_lock.h>
121 #include <__utility/move.h>
124 #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
125 # pragma GCC system_header
128 #ifndef _LIBCPP_HAS_NO_THREADS
130 _LIBCPP_BEGIN_NAMESPACE_STD
132 class _LIBCPP_EXPORTED_FROM_ABI condition_variable_any
134 condition_variable __cv_;
135 shared_ptr<mutex> __mut_;
137 _LIBCPP_INLINE_VISIBILITY
138 condition_variable_any();
140 _LIBCPP_INLINE_VISIBILITY
141 void notify_one() _NOEXCEPT;
142 _LIBCPP_INLINE_VISIBILITY
143 void notify_all() _NOEXCEPT;
145 template <class _Lock>
146 _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS
147 void wait(_Lock& __lock);
148 template <class _Lock, class _Predicate>
149 _LIBCPP_INLINE_VISIBILITY
150 void wait(_Lock& __lock, _Predicate __pred);
152 template <class _Lock, class _Clock, class _Duration>
153 _LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS
155 wait_until(_Lock& __lock,
156 const chrono::time_point<_Clock, _Duration>& __t);
158 template <class _Lock, class _Clock, class _Duration, class _Predicate>
160 _LIBCPP_INLINE_VISIBILITY
161 wait_until(_Lock& __lock,
162 const chrono::time_point<_Clock, _Duration>& __t,
165 template <class _Lock, class _Rep, class _Period>
167 _LIBCPP_INLINE_VISIBILITY
168 wait_for(_Lock& __lock,
169 const chrono::duration<_Rep, _Period>& __d);
171 template <class _Lock, class _Rep, class _Period, class _Predicate>
173 _LIBCPP_INLINE_VISIBILITY
174 wait_for(_Lock& __lock,
175 const chrono::duration<_Rep, _Period>& __d,
180 condition_variable_any::condition_variable_any()
181 : __mut_(make_shared<mutex>()) {}
185 condition_variable_any::notify_one() _NOEXCEPT
187 {lock_guard<mutex> __lx(*__mut_);}
193 condition_variable_any::notify_all() _NOEXCEPT
195 {lock_guard<mutex> __lx(*__mut_);}
199 struct __lock_external
201 template <class _Lock>
202 _LIBCPP_HIDE_FROM_ABI void operator()(_Lock* __m) {__m->lock();}
205 template <class _Lock>
207 condition_variable_any::wait(_Lock& __lock)
209 shared_ptr<mutex> __mut = __mut_;
210 unique_lock<mutex> __lk(*__mut);
212 unique_ptr<_Lock, __lock_external> __lxx(&__lock);
213 lock_guard<unique_lock<mutex> > __lx(__lk, adopt_lock_t());
215 } // __mut_.unlock(), __lock.lock()
217 template <class _Lock, class _Predicate>
220 condition_variable_any::wait(_Lock& __lock, _Predicate __pred)
226 template <class _Lock, class _Clock, class _Duration>
228 condition_variable_any::wait_until(_Lock& __lock,
229 const chrono::time_point<_Clock, _Duration>& __t)
231 shared_ptr<mutex> __mut = __mut_;
232 unique_lock<mutex> __lk(*__mut);
234 unique_ptr<_Lock, __lock_external> __lxx(&__lock);
235 lock_guard<unique_lock<mutex> > __lx(__lk, adopt_lock_t());
236 return __cv_.wait_until(__lk, __t);
237 } // __mut_.unlock(), __lock.lock()
239 template <class _Lock, class _Clock, class _Duration, class _Predicate>
242 condition_variable_any::wait_until(_Lock& __lock,
243 const chrono::time_point<_Clock, _Duration>& __t,
247 if (wait_until(__lock, __t) == cv_status::timeout)
252 template <class _Lock, class _Rep, class _Period>
255 condition_variable_any::wait_for(_Lock& __lock,
256 const chrono::duration<_Rep, _Period>& __d)
258 return wait_until(__lock, chrono::steady_clock::now() + __d);
261 template <class _Lock, class _Rep, class _Period, class _Predicate>
264 condition_variable_any::wait_for(_Lock& __lock,
265 const chrono::duration<_Rep, _Period>& __d,
268 return wait_until(__lock, chrono::steady_clock::now() + __d,
269 _VSTD::move(__pred));
272 _LIBCPP_EXPORTED_FROM_ABI void notify_all_at_thread_exit(condition_variable&, unique_lock<mutex>);
274 _LIBCPP_END_NAMESPACE_STD
276 #endif // !_LIBCPP_HAS_NO_THREADS
278 #if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES) && _LIBCPP_STD_VER <= 20
284 # include <initializer_list>
286 # include <stdexcept>
287 # include <system_error>
288 # include <type_traits>
292 #endif // _LIBCPP_CONDITION_VARIABLE