#ifndef BOOST_THREAD_PTHREAD_CONDITION_VARIABLE_FWD_HPP #define BOOST_THREAD_PTHREAD_CONDITION_VARIABLE_FWD_HPP // Distributed under the Boost Software License, Version 1.0. (See // accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // (C) Copyright 2007-8 Anthony Williams // (C) Copyright 2011-2012 Vicente J. Botet Escriba #include #include #include #include #include #include #include #include #include #if defined BOOST_THREAD_USES_DATETIME #include #endif #ifdef BOOST_THREAD_USES_CHRONO #include #include #endif #include #include #include #include namespace lslboost { class condition_variable { private: //#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS pthread_mutex_t internal_mutex; //#endif pthread_cond_t cond; public: //private: // used by lslboost::thread::try_join_until bool do_wait_until( unique_lock& lock, detail::internal_platform_timepoint const &timeout); public: BOOST_THREAD_NO_COPYABLE(condition_variable) condition_variable() { int res; //#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS // Even if it is not used, the internal_mutex exists (see // above) and must be initialized (etc) in case some // compilation units provide interruptions and others // don't. res=pthread_mutex_init(&internal_mutex,NULL); if(res) { lslboost::throw_exception(thread_resource_error(res, "lslboost::condition_variable::condition_variable() constructor failed in pthread_mutex_init")); } //#endif res = pthread::cond_init(cond); if (res) { //#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS // ditto BOOST_VERIFY(!pthread_mutex_destroy(&internal_mutex)); //#endif lslboost::throw_exception(thread_resource_error(res, "lslboost::condition_variable::condition_variable() constructor failed in pthread::cond_init")); } } ~condition_variable() { int ret; //#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS // ditto do { ret = pthread_mutex_destroy(&internal_mutex); } while (ret == EINTR); BOOST_ASSERT(!ret); //#endif do { ret = pthread_cond_destroy(&cond); } while (ret == EINTR); BOOST_ASSERT(!ret); } void wait(unique_lock& m); template void wait(unique_lock& m,predicate_type pred) { while (!pred()) { wait(m); } } #if defined BOOST_THREAD_USES_DATETIME bool timed_wait( unique_lock& m, lslboost::system_time const& abs_time) { #if defined BOOST_THREAD_WAIT_BUG const detail::real_platform_timepoint ts(abs_time + BOOST_THREAD_WAIT_BUG); #else const detail::real_platform_timepoint ts(abs_time); #endif #if defined BOOST_THREAD_INTERNAL_CLOCK_IS_MONO // The system time may jump while this function is waiting. To compensate for this and time // out near the correct time, we could call do_wait_until() in a loop with a short timeout // and recheck the time remaining each time through the loop. However, because we can't // check the predicate each time do_wait_until() completes, this introduces the possibility // of not exiting the function when a notification occurs, since do_wait_until() may report // that it timed out even though a notification was received. The best this function can do // is report correctly whether or not it reached the timeout time. const detail::platform_duration d(ts - detail::real_platform_clock::now()); do_wait_until(m, detail::internal_platform_clock::now() + d); return ts > detail::real_platform_clock::now(); #else return do_wait_until(m, ts); #endif } bool timed_wait( unique_lock& m, ::lslboost::xtime const& abs_time) { return timed_wait(m,system_time(abs_time)); } template bool timed_wait( unique_lock& m, duration_type const& wait_duration) { if (wait_duration.is_pos_infinity()) { wait(m); return true; } if (wait_duration.is_special()) { return true; } detail::platform_duration d(wait_duration); #if defined(BOOST_THREAD_HAS_MONO_CLOCK) && !defined(BOOST_THREAD_INTERNAL_CLOCK_IS_MONO) // The system time may jump while this function is waiting. To compensate for this and time // out near the correct time, we could call do_wait_until() in a loop with a short timeout // and recheck the time remaining each time through the loop. However, because we can't // check the predicate each time do_wait_until() completes, this introduces the possibility // of not exiting the function when a notification occurs, since do_wait_until() may report // that it timed out even though a notification was received. The best this function can do // is report correctly whether or not it reached the timeout time. const detail::mono_platform_timepoint ts(detail::mono_platform_clock::now() + d); do_wait_until(m, detail::internal_platform_clock::now() + d); return ts > detail::mono_platform_clock::now(); #else return do_wait_until(m, detail::internal_platform_clock::now() + d); #endif } template bool timed_wait( unique_lock& m, lslboost::system_time const& abs_time,predicate_type pred) { #if defined BOOST_THREAD_WAIT_BUG const detail::real_platform_timepoint ts(abs_time + BOOST_THREAD_WAIT_BUG); #else const detail::real_platform_timepoint ts(abs_time); #endif while (!pred()) { #if defined BOOST_THREAD_INTERNAL_CLOCK_IS_MONO // The system time may jump while this function is waiting. To compensate for this // and time out near the correct time, we call do_wait_until() in a loop with a // short timeout and recheck the time remaining each time through the loop. detail::platform_duration d(ts - detail::real_platform_clock::now()); if (d <= detail::platform_duration::zero()) break; // timeout occurred d = (std::min)(d, detail::platform_milliseconds(BOOST_THREAD_POLL_INTERVAL_MILLISECONDS)); do_wait_until(m, detail::internal_platform_clock::now() + d); #else if (!do_wait_until(m, ts)) break; // timeout occurred #endif } return pred(); } template bool timed_wait( unique_lock& m, ::lslboost::xtime const& abs_time,predicate_type pred) { return timed_wait(m,system_time(abs_time),pred); } template bool timed_wait( unique_lock& m, duration_type const& wait_duration,predicate_type pred) { if (wait_duration.is_pos_infinity()) { while (!pred()) { wait(m); } return true; } if (wait_duration.is_special()) { return pred(); } detail::platform_duration d(wait_duration); #if defined(BOOST_THREAD_HAS_MONO_CLOCK) && !defined(BOOST_THREAD_INTERNAL_CLOCK_IS_MONO) // The system time may jump while this function is waiting. To compensate for this // and time out near the correct time, we call do_wait_until() in a loop with a // short timeout and recheck the time remaining each time through the loop. const detail::mono_platform_timepoint ts(detail::mono_platform_clock::now() + d); while (!pred()) { if (d <= detail::platform_duration::zero()) break; // timeout occurred d = (std::min)(d, detail::platform_milliseconds(BOOST_THREAD_POLL_INTERVAL_MILLISECONDS)); do_wait_until(m, detail::internal_platform_clock::now() + d); d = ts - detail::mono_platform_clock::now(); } #else const detail::internal_platform_timepoint ts(detail::internal_platform_clock::now() + d); while (!pred()) { if (!do_wait_until(m, ts)) break; // timeout occurred } #endif return pred(); } #endif #ifdef BOOST_THREAD_USES_CHRONO template cv_status wait_until( unique_lock& lock, const chrono::time_point& t) { const detail::internal_platform_timepoint ts(t); if (do_wait_until(lock, ts)) return cv_status::no_timeout; else return cv_status::timeout; } template cv_status wait_until( unique_lock& lock, const chrono::time_point& t) { // The system time may jump while this function is waiting. To compensate for this and time // out near the correct time, we could call do_wait_until() in a loop with a short timeout // and recheck the time remaining each time through the loop. However, because we can't // check the predicate each time do_wait_until() completes, this introduces the possibility // of not exiting the function when a notification occurs, since do_wait_until() may report // that it timed out even though a notification was received. The best this function can do // is report correctly whether or not it reached the timeout time. typedef typename common_type::type common_duration; common_duration d(t - Clock::now()); do_wait_until(lock, detail::internal_chrono_clock::now() + d); if (t > Clock::now()) return cv_status::no_timeout; else return cv_status::timeout; } template cv_status wait_for( unique_lock& lock, const chrono::duration& d) { return wait_until(lock, chrono::steady_clock::now() + d); } template bool wait_until( unique_lock& lock, const chrono::time_point& t, Predicate pred) { const detail::internal_platform_timepoint ts(t); while (!pred()) { if (!do_wait_until(lock, ts)) break; // timeout occurred } return pred(); } template bool wait_until( unique_lock& lock, const chrono::time_point& t, Predicate pred) { // The system time may jump while this function is waiting. To compensate for this // and time out near the correct time, we call do_wait_until() in a loop with a // short timeout and recheck the time remaining each time through the loop. typedef typename common_type::type common_duration; while (!pred()) { common_duration d(t - Clock::now()); if (d <= common_duration::zero()) break; // timeout occurred d = (std::min)(d, common_duration(chrono::milliseconds(BOOST_THREAD_POLL_INTERVAL_MILLISECONDS))); do_wait_until(lock, detail::internal_platform_clock::now() + detail::platform_duration(d)); } return pred(); } template bool wait_for( unique_lock& lock, const chrono::duration& d, Predicate pred) { return wait_until(lock, chrono::steady_clock::now() + d, lslboost::move(pred)); } #endif #define BOOST_THREAD_DEFINES_CONDITION_VARIABLE_NATIVE_HANDLE typedef pthread_cond_t* native_handle_type; native_handle_type native_handle() { return &cond; } void notify_one() BOOST_NOEXCEPT; void notify_all() BOOST_NOEXCEPT; }; BOOST_THREAD_DECL void notify_all_at_thread_exit(condition_variable& cond, unique_lock lk); } #include #endif