// <condition_variable> -*- C++ -*- // Copyright (C) 2008-2017 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // <http://www.gnu.org/licenses/>. /** @file include/condition_variable * This is a Standard C++ Library header. */ #ifndef _GLIBCXX_CONDITION_VARIABLE #define _GLIBCXX_CONDITION_VARIABLE 1 #pragma GCC system_header #if __cplusplus < 201103L # include <bits/c++0x_warning.h> #else #include <chrono> #include <bits/std_mutex.h> #include <ext/concurrence.h> #include <bits/alloc_traits.h> #include <bits/allocator.h> #include <bits/unique_ptr.h> #include <bits/shared_ptr.h> #include <bits/cxxabi_forced.h> #if defined(_GLIBCXX_HAS_GTHREADS) && defined(_GLIBCXX_USE_C99_STDINT_TR1) namespace std _GLIBCXX_VISIBILITY(default) { _GLIBCXX_BEGIN_NAMESPACE_VERSION /** * @defgroup condition_variables Condition Variables * @ingroup concurrency * * Classes for condition_variable support. * @{ */ /// cv_status enum class cv_status { no_timeout, timeout }; /// condition_variable class condition_variable { typedef chrono::system_clock __clock_t; typedef __gthread_cond_t __native_type; #ifdef __GTHREAD_COND_INIT __native_type _M_cond = __GTHREAD_COND_INIT; #else __native_type _M_cond; #endif public: typedef __native_type* native_handle_type; condition_variable() noexcept; ~condition_variable() noexcept; condition_variable(const condition_variable&) = delete; condition_variable& operator=(const condition_variable&) = delete; void notify_one() noexcept; void notify_all() noexcept; void wait(unique_lock<mutex>& __lock) noexcept; template<typename _Predicate> void wait(unique_lock<mutex>& __lock, _Predicate __p) { while (!__p()) wait(__lock); } template<typename _Duration> cv_status wait_until(unique_lock<mutex>& __lock, const chrono::time_point<__clock_t, _Duration>& __atime) { return __wait_until_impl(__lock, __atime); } template<typename _Clock, typename _Duration> cv_status wait_until(unique_lock<mutex>& __lock, const chrono::time_point<_Clock, _Duration>& __atime) { // DR 887 - Sync unknown clock to known clock. const typename _Clock::time_point __c_entry = _Clock::now(); const __clock_t::time_point __s_entry = __clock_t::now(); const auto __delta = __atime - __c_entry; const auto __s_atime = __s_entry + __delta; return __wait_until_impl(__lock, __s_atime); } template<typename _Clock, typename _Duration, typename _Predicate> bool wait_until(unique_lock<mutex>& __lock, const chrono::time_point<_Clock, _Duration>& __atime, _Predicate __p) { while (!__p()) if (wait_until(__lock, __atime) == cv_status::timeout) return __p(); return true; } template<typename _Rep, typename _Period> cv_status wait_for(unique_lock<mutex>& __lock, const chrono::duration<_Rep, _Period>& __rtime) { using __dur = typename __clock_t::duration; auto __reltime = chrono::duration_cast<__dur>(__rtime); if (__reltime < __rtime) ++__reltime; return wait_until(__lock, __clock_t::now() + __reltime); } template<typename _Rep, typename _Period, typename _Predicate> bool wait_for(unique_lock<mutex>& __lock, const chrono::duration<_Rep, _Period>& __rtime, _Predicate __p) { using __dur = typename __clock_t::duration; auto __reltime = chrono::duration_cast<__dur>(__rtime); if (__reltime < __rtime) ++__reltime; return wait_until(__lock, __clock_t::now() + __reltime, std::move(__p)); } native_handle_type native_handle() { return &_M_cond; } private: template<typename _Dur> cv_status __wait_until_impl(unique_lock<mutex>& __lock, const chrono::time_point<__clock_t, _Dur>& __atime) { auto __s = chrono::time_point_cast<chrono::seconds>(__atime); auto __ns = chrono::duration_cast<chrono::nanoseconds>(__atime - __s); __gthread_time_t __ts = { static_cast<std::time_t>(__s.time_since_epoch().count()), static_cast<long>(__ns.count()) }; __gthread_cond_timedwait(&_M_cond, __lock.mutex()->native_handle(), &__ts); return (__clock_t::now() < __atime ? cv_status::no_timeout : cv_status::timeout); } }; void notify_all_at_thread_exit(condition_variable&, unique_lock<mutex>); struct __at_thread_exit_elt { __at_thread_exit_elt* _M_next; void (*_M_cb)(void*); }; inline namespace _V2 { /// condition_variable_any // Like above, but mutex is not required to have try_lock. class condition_variable_any { typedef chrono::system_clock __clock_t; condition_variable _M_cond; shared_ptr<mutex> _M_mutex; // scoped unlock - unlocks in ctor, re-locks in dtor template<typename _Lock> struct _Unlock { explicit _Unlock(_Lock& __lk) : _M_lock(__lk) { __lk.unlock(); } ~_Unlock() noexcept(false) { if (uncaught_exception()) { __try { _M_lock.lock(); } __catch(const __cxxabiv1::__forced_unwind&) { __throw_exception_again; } __catch(...) { } } else _M_lock.lock(); } _Unlock(const _Unlock&) = delete; _Unlock& operator=(const _Unlock&) = delete; _Lock& _M_lock; }; public: condition_variable_any() : _M_mutex(std::make_shared<mutex>()) { } ~condition_variable_any() = default; condition_variable_any(const condition_variable_any&) = delete; condition_variable_any& operator=(const condition_variable_any&) = delete; void notify_one() noexcept { lock_guard<mutex> __lock(*_M_mutex); _M_cond.notify_one(); } void notify_all() noexcept { lock_guard<mutex> __lock(*_M_mutex); _M_cond.notify_all(); } template<typename _Lock> void wait(_Lock& __lock) { shared_ptr<mutex> __mutex = _M_mutex; unique_lock<mutex> __my_lock(*__mutex); _Unlock<_Lock> __unlock(__lock); // *__mutex must be unlocked before re-locking __lock so move // ownership of *__mutex lock to an object with shorter lifetime. unique_lock<mutex> __my_lock2(std::move(__my_lock)); _M_cond.wait(__my_lock2); } template<typename _Lock, typename _Predicate> void wait(_Lock& __lock, _Predicate __p) { while (!__p()) wait(__lock); } template<typename _Lock, typename _Clock, typename _Duration> cv_status wait_until(_Lock& __lock, const chrono::time_point<_Clock, _Duration>& __atime) { shared_ptr<mutex> __mutex = _M_mutex; unique_lock<mutex> __my_lock(*__mutex); _Unlock<_Lock> __unlock(__lock); // *__mutex must be unlocked before re-locking __lock so move // ownership of *__mutex lock to an object with shorter lifetime. unique_lock<mutex> __my_lock2(std::move(__my_lock)); return _M_cond.wait_until(__my_lock2, __atime); } template<typename _Lock, typename _Clock, typename _Duration, typename _Predicate> bool wait_until(_Lock& __lock, const chrono::time_point<_Clock, _Duration>& __atime, _Predicate __p) { while (!__p()) if (wait_until(__lock, __atime) == cv_status::timeout) return __p(); return true; } template<typename _Lock, typename _Rep, typename _Period> cv_status wait_for(_Lock& __lock, const chrono::duration<_Rep, _Period>& __rtime) { return wait_until(__lock, __clock_t::now() + __rtime); } template<typename _Lock, typename _Rep, typename _Period, typename _Predicate> bool wait_for(_Lock& __lock, const chrono::duration<_Rep, _Period>& __rtime, _Predicate __p) { return wait_until(__lock, __clock_t::now() + __rtime, std::move(__p)); } }; } // end inline namespace // @} group condition_variables _GLIBCXX_END_NAMESPACE_VERSION } // namespace #endif // _GLIBCXX_HAS_GTHREADS && _GLIBCXX_USE_C99_STDINT_TR1 #endif // C++11 #endif // _GLIBCXX_CONDITION_VARIABLE
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
backward | Folder | 0755 |
|
|
bits | Folder | 0755 |
|
|
debug | Folder | 0755 |
|
|
decimal | Folder | 0755 |
|
|
experimental | Folder | 0755 |
|
|
ext | Folder | 0755 |
|
|
parallel | Folder | 0755 |
|
|
profile | Folder | 0755 |
|
|
tr1 | Folder | 0755 |
|
|
tr2 | Folder | 0755 |
|
|
algorithm | File | 2.46 KB | 0644 |
|
any | File | 18.48 KB | 0644 |
|
array | File | 11.12 KB | 0644 |
|
atomic | File | 40.01 KB | 0644 |
|
bitset | File | 44.81 KB | 0644 |
|
cassert | File | 1.61 KB | 0644 |
|
ccomplex | File | 1.3 KB | 0644 |
|
cctype | File | 2.35 KB | 0644 |
|
cerrno | File | 1.73 KB | 0644 |
|
cfenv | File | 2 KB | 0644 |
|
cfloat | File | 1.84 KB | 0644 |
|
chrono | File | 29.38 KB | 0644 |
|
cinttypes | File | 2.11 KB | 0644 |
|
ciso646 | File | 1.43 KB | 0644 |
|
climits | File | 1.87 KB | 0644 |
|
clocale | File | 1.86 KB | 0644 |
|
cmath | File | 47.35 KB | 0644 |
|
codecvt | File | 5.22 KB | 0644 |
|
complex | File | 52.93 KB | 0644 |
|
complex.h | File | 1.43 KB | 0644 |
|
condition_variable | File | 8.74 KB | 0644 |
|
csetjmp | File | 1.9 KB | 0644 |
|
csignal | File | 1.81 KB | 0644 |
|
cstdalign | File | 1.37 KB | 0644 |
|
cstdarg | File | 1.82 KB | 0644 |
|
cstdbool | File | 1.37 KB | 0644 |
|
cstddef | File | 6.29 KB | 0644 |
|
cstdint | File | 2.12 KB | 0644 |
|
cstdio | File | 4.33 KB | 0644 |
|
cstdlib | File | 6.18 KB | 0644 |
|
cstring | File | 3.05 KB | 0644 |
|
ctgmath | File | 1.33 KB | 0644 |
|
ctime | File | 2.08 KB | 0644 |
|
cuchar | File | 2.16 KB | 0644 |
|
cwchar | File | 6.36 KB | 0644 |
|
cwctype | File | 2.73 KB | 0644 |
|
cxxabi.h | File | 21.46 KB | 0644 |
|
deque | File | 2.6 KB | 0644 |
|
exception | File | 4.67 KB | 0644 |
|
fenv.h | File | 1.97 KB | 0644 |
|
forward_list | File | 1.54 KB | 0644 |
|
fstream | File | 33.05 KB | 0644 |
|
functional | File | 38.64 KB | 0644 |
|
future | File | 49.49 KB | 0644 |
|
initializer_list | File | 2.9 KB | 0644 |
|
iomanip | File | 15.75 KB | 0644 |
|
ios | File | 1.56 KB | 0644 |
|
iosfwd | File | 6.76 KB | 0644 |
|
iostream | File | 2.63 KB | 0644 |
|
istream | File | 32.07 KB | 0644 |
|
iterator | File | 2.58 KB | 0644 |
|
limits | File | 67.56 KB | 0644 |
|
list | File | 2.53 KB | 0644 |
|
locale | File | 1.45 KB | 0644 |
|
map | File | 2.5 KB | 0644 |
|
math.h | File | 4.28 KB | 0644 |
|
memory | File | 4.6 KB | 0644 |
|
mutex | File | 17.88 KB | 0644 |
|
new | File | 7.33 KB | 0644 |
|
numeric | File | 5.11 KB | 0644 |
|
optional | File | 30.39 KB | 0644 |
|
ostream | File | 21.46 KB | 0644 |
|
queue | File | 2.41 KB | 0644 |
|
random | File | 1.65 KB | 0644 |
|
ratio | File | 19.42 KB | 0644 |
|
regex | File | 1.85 KB | 0644 |
|
scoped_allocator | File | 15.84 KB | 0644 |
|
set | File | 2.5 KB | 0644 |
|
shared_mutex | File | 18.95 KB | 0644 |
|
sstream | File | 26.21 KB | 0644 |
|
stack | File | 2.33 KB | 0644 |
|
stdexcept | File | 7.79 KB | 0644 |
|
stdlib.h | File | 2.2 KB | 0644 |
|
streambuf | File | 28.87 KB | 0644 |
|
string | File | 1.89 KB | 0644 |
|
string_view | File | 20.8 KB | 0644 |
|
system_error | File | 11.25 KB | 0644 |
|
tgmath.h | File | 1.33 KB | 0644 |
|
thread | File | 10.03 KB | 0644 |
|
tuple | File | 58.22 KB | 0644 |
|
type_traits | File | 90.28 KB | 0644 |
|
typeindex | File | 3.02 KB | 0644 |
|
typeinfo | File | 7.51 KB | 0644 |
|
unordered_map | File | 1.81 KB | 0644 |
|
unordered_set | File | 1.81 KB | 0644 |
|
utility | File | 12.16 KB | 0644 |
|
valarray | File | 39.33 KB | 0644 |
|
variant | File | 42.09 KB | 0644 |
|
vector | File | 2.68 KB | 0644 |
|