관리-도구
편집 파일: future.hpp
// (C) Copyright 2008-10 Anthony Williams // (C) Copyright 2011-2015 Vicente J. Botet Escriba // // 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) #ifndef BOOST_THREAD_FUTURE_HPP #define BOOST_THREAD_FUTURE_HPP #include <boost/thread/detail/config.hpp> // boost::thread::future requires exception handling // due to boost::exception::exception_ptr dependency //#define BOOST_THREAD_CONTINUATION_SYNC #ifdef BOOST_NO_EXCEPTIONS namespace boost { namespace detail { struct shared_state_base { void notify_deferred() {} }; } } #else #include <boost/thread/condition_variable.hpp> #include <boost/thread/detail/move.hpp> #include <boost/thread/detail/invoker.hpp> #include <boost/thread/detail/invoke.hpp> #include <boost/thread/detail/is_convertible.hpp> #include <boost/thread/exceptional_ptr.hpp> #include <boost/thread/futures/future_error.hpp> #include <boost/thread/futures/future_error_code.hpp> #include <boost/thread/futures/future_status.hpp> #include <boost/thread/futures/is_future_type.hpp> #include <boost/thread/futures/launch.hpp> #include <boost/thread/futures/wait_for_all.hpp> #include <boost/thread/futures/wait_for_any.hpp> #include <boost/thread/lock_algorithms.hpp> #include <boost/thread/lock_types.hpp> #include <boost/thread/mutex.hpp> #include <boost/thread/thread_only.hpp> #include <boost/thread/thread_time.hpp> #include <boost/thread/executor.hpp> #include <boost/thread/executors/generic_executor_ref.hpp> #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL #include <boost/optional.hpp> #else #include <boost/thread/csbl/memory/unique_ptr.hpp> #endif #include <boost/assert.hpp> #include <boost/bind/bind.hpp> #ifdef BOOST_THREAD_USES_CHRONO #include <boost/chrono/system_clocks.hpp> #endif #include <boost/core/enable_if.hpp> #include <boost/core/ref.hpp> #include <boost/enable_shared_from_this.hpp> #include <boost/exception_ptr.hpp> #include <boost/function.hpp> #include <boost/scoped_array.hpp> #include <boost/shared_ptr.hpp> #include <boost/smart_ptr/make_shared.hpp> #include <boost/throw_exception.hpp> #include <boost/type_traits/conditional.hpp> #include <boost/type_traits/decay.hpp> #include <boost/type_traits/is_copy_constructible.hpp> #include <boost/type_traits/is_fundamental.hpp> #include <boost/type_traits/is_void.hpp> #include <boost/utility/result_of.hpp> #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS #include <boost/thread/detail/memory.hpp> #include <boost/container/scoped_allocator.hpp> #if ! defined BOOST_NO_CXX11_ALLOCATOR #include <memory> #endif #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY #include <boost/thread/csbl/tuple.hpp> #include <boost/thread/csbl/vector.hpp> #endif #include <algorithm> #include <list> #include <vector> #include <utility> #if defined BOOST_THREAD_PROVIDES_FUTURE #define BOOST_THREAD_FUTURE future #else #define BOOST_THREAD_FUTURE unique_future #endif namespace boost { template <class T> shared_ptr<T> static_shared_from_this(T* that) { return static_pointer_cast<T>(that->shared_from_this()); } template <class T> shared_ptr<T const> static_shared_from_this(T const* that) { return static_pointer_cast<T const>(that->shared_from_this()); } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS #else namespace executors { class executor; } using executors::executor; #endif typedef shared_ptr<executor> executor_ptr_type; namespace detail { struct relocker { boost::unique_lock<boost::mutex>& lock_; relocker(boost::unique_lock<boost::mutex>& lk): lock_(lk) { lock_.unlock(); } ~relocker() { if (! lock_.owns_lock()) { lock_.lock(); } } void lock() { if (! lock_.owns_lock()) { lock_.lock(); } } private: relocker& operator=(relocker const&); }; struct shared_state_base : enable_shared_from_this<shared_state_base> { typedef std::list<boost::condition_variable_any*> waiter_list; typedef waiter_list::iterator notify_when_ready_handle; // This type should be only included conditionally if interruptions are allowed, but is included to maintain the same layout. typedef shared_ptr<shared_state_base> continuation_ptr_type; typedef std::vector<continuation_ptr_type> continuations_type; boost::exception_ptr exception; bool done; bool is_valid_; bool is_deferred_; bool is_constructed; launch policy_; mutable boost::mutex mutex; boost::condition_variable waiters; waiter_list external_waiters; boost::function<void()> callback; // This declaration should be only included conditionally, but is included to maintain the same layout. continuations_type continuations; executor_ptr_type ex_; // This declaration should be only included conditionally, but is included to maintain the same layout. virtual void launch_continuation() { } shared_state_base(): done(false), is_valid_(true), is_deferred_(false), is_constructed(false), policy_(launch::none), continuations(), ex_() {} shared_state_base(exceptional_ptr const& ex): exception(ex.ptr_), done(true), is_valid_(true), is_deferred_(false), is_constructed(false), policy_(launch::none), continuations(), ex_() {} virtual ~shared_state_base() { } bool is_done() { return done; } executor_ptr_type get_executor() { return ex_; } void set_executor_policy(executor_ptr_type aex) { set_executor(); ex_ = aex; } void set_executor_policy(executor_ptr_type aex, boost::lock_guard<boost::mutex>&) { set_executor(); ex_ = aex; } void set_executor_policy(executor_ptr_type aex, boost::unique_lock<boost::mutex>&) { set_executor(); ex_ = aex; } bool valid(boost::unique_lock<boost::mutex>&) { return is_valid_; } bool valid() { boost::unique_lock<boost::mutex> lk(this->mutex); return valid(lk); } void invalidate(boost::unique_lock<boost::mutex>&) { is_valid_ = false; } void invalidate() { boost::unique_lock<boost::mutex> lk(this->mutex); invalidate(lk); } void validate(boost::unique_lock<boost::mutex>&) { is_valid_ = true; } void validate() { boost::unique_lock<boost::mutex> lk(this->mutex); validate(lk); } void set_deferred() { is_deferred_ = true; policy_ = launch::deferred; } void set_async() { is_deferred_ = false; policy_ = launch::async; } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS void set_executor() { is_deferred_ = false; policy_ = launch::executor; } #else void set_executor() { } #endif notify_when_ready_handle notify_when_ready(boost::condition_variable_any& cv) { boost::unique_lock<boost::mutex> lock(this->mutex); do_callback(lock); return external_waiters.insert(external_waiters.end(),&cv); } void unnotify_when_ready(notify_when_ready_handle it) { boost::lock_guard<boost::mutex> lock(this->mutex); external_waiters.erase(it); } #if 0 // this inline definition results in ODR. See https://github.com/boostorg/thread/issues/193 // to avoid it, we define the function on the derived templates using the macro BOOST_THREAD_DO_CONTINUATION #define BOOST_THREAD_DO_CONTINUATION #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION void do_continuation(boost::unique_lock<boost::mutex>& lock) { if (! continuations.empty()) { continuations_type the_continuations = continuations; continuations.clear(); relocker rlk(lock); for (continuations_type::iterator it = the_continuations.begin(); it != the_continuations.end(); ++it) { (*it)->launch_continuation(); } } } #else void do_continuation(boost::unique_lock<boost::mutex>&) { } #endif #else #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION #define BOOST_THREAD_DO_CONTINUATION \ void do_continuation(boost::unique_lock<boost::mutex>& lock) \ { \ if (! this->continuations.empty()) { \ continuations_type the_continuations = this->continuations; \ this->continuations.clear(); \ relocker rlk(lock); \ for (continuations_type::iterator it = the_continuations.begin(); it != the_continuations.end(); ++it) { \ (*it)->launch_continuation(); \ } \ } \ } #else #define BOOST_THREAD_DO_CONTINUATION \ void do_continuation(boost::unique_lock<boost::mutex>&) \ { \ } #endif virtual void do_continuation(boost::unique_lock<boost::mutex>&) = 0; #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION virtual void set_continuation_ptr(continuation_ptr_type continuation, boost::unique_lock<boost::mutex>& lock) { continuations.push_back(continuation); if (done) { do_continuation(lock); } } #endif void mark_finished_internal(boost::unique_lock<boost::mutex>& lock) { done=true; waiters.notify_all(); for(waiter_list::const_iterator it=external_waiters.begin(), end=external_waiters.end();it!=end;++it) { (*it)->notify_all(); } do_continuation(lock); } void notify_deferred() { boost::unique_lock<boost::mutex> lock(this->mutex); mark_finished_internal(lock); } void do_callback(boost::unique_lock<boost::mutex>& lock) { if(callback && !done) { boost::function<void()> local_callback=callback; relocker relock(lock); local_callback(); } } virtual bool run_if_is_deferred() { boost::unique_lock<boost::mutex> lk(this->mutex); if (is_deferred_) { is_deferred_=false; execute(lk); return true; } else return false; } virtual bool run_if_is_deferred_or_ready() { boost::unique_lock<boost::mutex> lk(this->mutex); if (is_deferred_) { is_deferred_=false; execute(lk); return true; } else return done; } void wait_internal(boost::unique_lock<boost::mutex> &lk, bool rethrow=true) { do_callback(lk); if (is_deferred_) { is_deferred_=false; execute(lk); } waiters.wait(lk, boost::bind(&shared_state_base::is_done, boost::ref(*this))); if(rethrow && exception) { boost::rethrow_exception(exception); } } virtual void wait(boost::unique_lock<boost::mutex>& lock, bool rethrow=true) { wait_internal(lock, rethrow); } void wait(bool rethrow=true) { boost::unique_lock<boost::mutex> lock(this->mutex); wait(lock, rethrow); } #if defined BOOST_THREAD_USES_DATETIME template<typename Duration> bool timed_wait(Duration const& rel_time) { boost::unique_lock<boost::mutex> lock(this->mutex); if (is_deferred_) return false; do_callback(lock); return waiters.timed_wait(lock, rel_time, boost::bind(&shared_state_base::is_done, boost::ref(*this))); } bool timed_wait_until(boost::system_time const& target_time) { boost::unique_lock<boost::mutex> lock(this->mutex); if (is_deferred_) return false; do_callback(lock); return waiters.timed_wait(lock, target_time, boost::bind(&shared_state_base::is_done, boost::ref(*this))); } #endif #ifdef BOOST_THREAD_USES_CHRONO template <class Clock, class Duration> future_status wait_until(const chrono::time_point<Clock, Duration>& abs_time) { boost::unique_lock<boost::mutex> lock(this->mutex); if (is_deferred_) return future_status::deferred; do_callback(lock); if(!waiters.wait_until(lock, abs_time, boost::bind(&shared_state_base::is_done, boost::ref(*this)))) { return future_status::timeout; } return future_status::ready; } #endif void mark_exceptional_finish_internal(boost::exception_ptr const& e, boost::unique_lock<boost::mutex>& lock) { exception=e; mark_finished_internal(lock); } void mark_exceptional_finish() { boost::unique_lock<boost::mutex> lock(this->mutex); mark_exceptional_finish_internal(boost::current_exception(), lock); } void set_exception_deferred(exception_ptr e) { unique_lock<boost::mutex> lk(this->mutex); if (has_value(lk)) { throw_exception(promise_already_satisfied()); } exception=e; this->is_constructed = true; } void set_exception_at_thread_exit(exception_ptr e) { set_exception_deferred(e); // unique_lock<boost::mutex> lk(this->mutex); // if (has_value(lk)) // { // throw_exception(promise_already_satisfied()); // } // exception=e; // this->is_constructed = true; detail::make_ready_at_thread_exit(shared_from_this()); } bool has_value() const { boost::lock_guard<boost::mutex> lock(this->mutex); return done && ! exception; } bool has_value(unique_lock<boost::mutex>& ) const { return done && ! exception; } bool has_exception() const { boost::lock_guard<boost::mutex> lock(this->mutex); return done && exception; } launch launch_policy(boost::unique_lock<boost::mutex>&) const { return policy_; } future_state::state get_state(boost::unique_lock<boost::mutex>&) const { if(!done) { return future_state::waiting; } else { return future_state::ready; } } future_state::state get_state() const { boost::lock_guard<boost::mutex> guard(this->mutex); if(!done) { return future_state::waiting; } else { return future_state::ready; } } exception_ptr get_exception_ptr() { boost::unique_lock<boost::mutex> lock(this->mutex); wait_internal(lock, false); return exception; } template<typename F,typename U> void set_wait_callback(F f,U* u) { boost::lock_guard<boost::mutex> lock(this->mutex); callback=boost::bind(f,boost::ref(*u)); } virtual void execute(boost::unique_lock<boost::mutex>&) {} private: shared_state_base(shared_state_base const&); shared_state_base& operator=(shared_state_base const&); }; // Used to create stand-alone futures template<typename T> struct shared_state: detail::shared_state_base { #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL typedef boost::optional<T> storage_type; #else typedef boost::csbl::unique_ptr<T> storage_type; #endif #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES typedef T const& source_reference_type; typedef BOOST_THREAD_RV_REF(T) rvalue_source_type; typedef T move_dest_type; #elif defined BOOST_THREAD_USES_MOVE typedef typename conditional<boost::is_fundamental<T>::value,T,T const&>::type source_reference_type; typedef BOOST_THREAD_RV_REF(T) rvalue_source_type; typedef T move_dest_type; #else typedef T& source_reference_type; typedef typename conditional<boost::thread_detail::is_convertible<T&,BOOST_THREAD_RV_REF(T) >::value, BOOST_THREAD_RV_REF(T),T const&>::type rvalue_source_type; typedef typename conditional<boost::thread_detail::is_convertible<T&,BOOST_THREAD_RV_REF(T) >::value, BOOST_THREAD_RV_REF(T),T>::type move_dest_type; #endif typedef const T& shared_future_get_result_type; storage_type result; shared_state(): result() {} shared_state(exceptional_ptr const& ex): detail::shared_state_base(ex), result() {} // locating this definition on the template avoid the ODR issue. See https://github.com/boostorg/thread/issues/193 BOOST_THREAD_DO_CONTINUATION void mark_finished_with_result_internal(source_reference_type result_, boost::unique_lock<boost::mutex>& lock) { #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result = result_; #else result.reset(new T(result_)); #endif this->mark_finished_internal(lock); } void mark_finished_with_result_internal(rvalue_source_type result_, boost::unique_lock<boost::mutex>& lock) { #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result = boost::move(result_); #elif ! defined BOOST_NO_CXX11_RVALUE_REFERENCES result.reset(new T(boost::move(result_))); #else result.reset(new T(static_cast<rvalue_source_type>(result_))); #endif this->mark_finished_internal(lock); } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template <class ...Args> void mark_finished_with_result_internal(boost::unique_lock<boost::mutex>& lock, BOOST_THREAD_FWD_REF(Args)... args) { #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result.emplace(boost::forward<Args>(args)...); #else result.reset(new T(boost::forward<Args>(args)...)); #endif this->mark_finished_internal(lock); } #endif void mark_finished_with_result(source_reference_type result_) { boost::unique_lock<boost::mutex> lock(this->mutex); this->mark_finished_with_result_internal(result_, lock); } void mark_finished_with_result(rvalue_source_type result_) { boost::unique_lock<boost::mutex> lock(this->mutex); #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES mark_finished_with_result_internal(boost::move(result_), lock); #else mark_finished_with_result_internal(static_cast<rvalue_source_type>(result_), lock); #endif } storage_type& get_storage(boost::unique_lock<boost::mutex>& lk) { wait_internal(lk); return result; } virtual move_dest_type get(boost::unique_lock<boost::mutex>& lk) { return boost::move(*get_storage(lk)); } move_dest_type get() { boost::unique_lock<boost::mutex> lk(this->mutex); return this->get(lk); } virtual shared_future_get_result_type get_sh(boost::unique_lock<boost::mutex>& lk) { return *get_storage(lk); } shared_future_get_result_type get_sh() { boost::unique_lock<boost::mutex> lk(this->mutex); return this->get_sh(lk); } void set_value_deferred(source_reference_type result_) { unique_lock<boost::mutex> lk(this->mutex); if (this->has_value(lk)) { throw_exception(promise_already_satisfied()); } #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result = result_; #else result.reset(new T(result_)); #endif this->is_constructed = true; } void set_value_deferred(rvalue_source_type result_) { unique_lock<boost::mutex> lk(this->mutex); if (this->has_value(lk)) { throw_exception(promise_already_satisfied()); } #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result = boost::move(result_); #else result.reset(new T(boost::move(result_))); #endif #else #if defined BOOST_THREAD_FUTURE_USES_OPTIONAL result = boost::move(result_); #else result.reset(new T(static_cast<rvalue_source_type>(result_))); #endif #endif this->is_constructed = true; } void set_value_at_thread_exit(source_reference_type result_) { set_value_deferred(result_); // unique_lock<boost::mutex> lk(this->mutex); // if (this->has_value(lk)) // { // throw_exception(promise_already_satisfied()); // } //#if defined BOOST_THREAD_FUTURE_USES_OPTIONAL // result = result_; //#else // result.reset(new T(result_)); //#endif // // this->is_constructed = true; detail::make_ready_at_thread_exit(shared_from_this()); } void set_value_at_thread_exit(rvalue_source_type result_) { set_value_deferred(boost::move(result_)); // unique_lock<boost::mutex> lk(this->mutex); // if (this->has_value(lk)) // throw_exception(promise_already_satisfied()); // //#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES //#if defined BOOST_THREAD_FUTURE_USES_OPTIONAL // result = boost::move(result_); //#else // result.reset(new T(boost::move(result_))); //#endif //#else //#if defined BOOST_THREAD_FUTURE_USES_OPTIONAL // result = boost::move(result_); //#else // result.reset(new T(static_cast<rvalue_source_type>(result_))); //#endif //#endif // this->is_constructed = true; detail::make_ready_at_thread_exit(shared_from_this()); } private: shared_state(shared_state const&); shared_state& operator=(shared_state const&); }; template<typename T> struct shared_state<T&>: detail::shared_state_base { typedef T* storage_type; typedef T& source_reference_type; typedef T& move_dest_type; typedef T& shared_future_get_result_type; T* result; shared_state(): result(0) {} shared_state(exceptional_ptr const& ex): detail::shared_state_base(ex), result(0) {} // locating this definition on the template avoid the ODR issue. See https://github.com/boostorg/thread/issues/193 BOOST_THREAD_DO_CONTINUATION void mark_finished_with_result_internal(source_reference_type result_, boost::unique_lock<boost::mutex>& lock) { result= &result_; mark_finished_internal(lock); } void mark_finished_with_result(source_reference_type result_) { boost::unique_lock<boost::mutex> lock(this->mutex); mark_finished_with_result_internal(result_, lock); } virtual T& get(boost::unique_lock<boost::mutex>& lock) { wait_internal(lock); return *result; } T& get() { boost::unique_lock<boost::mutex> lk(this->mutex); return get(lk); } virtual T& get_sh(boost::unique_lock<boost::mutex>& lock) { wait_internal(lock); return *result; } T& get_sh() { boost::unique_lock<boost::mutex> lock(this->mutex); return get_sh(lock); } void set_value_deferred(T& result_) { unique_lock<boost::mutex> lk(this->mutex); if (this->has_value(lk)) { throw_exception(promise_already_satisfied()); } result= &result_; this->is_constructed = true; } void set_value_at_thread_exit(T& result_) { set_value_deferred(result_); // unique_lock<boost::mutex> lk(this->mutex); // if (this->has_value(lk)) // throw_exception(promise_already_satisfied()); // result= &result_; // this->is_constructed = true; detail::make_ready_at_thread_exit(shared_from_this()); } private: shared_state(shared_state const&); shared_state& operator=(shared_state const&); }; template<> struct shared_state<void>: detail::shared_state_base { typedef void shared_future_get_result_type; typedef void move_dest_type; shared_state() {} shared_state(exceptional_ptr const& ex): detail::shared_state_base(ex) {} // locating this definition on the template avoid the ODR issue. See https://github.com/boostorg/thread/issues/193 BOOST_THREAD_DO_CONTINUATION void mark_finished_with_result_internal(boost::unique_lock<boost::mutex>& lock) { mark_finished_internal(lock); } void mark_finished_with_result() { boost::unique_lock<boost::mutex> lock(this->mutex); mark_finished_with_result_internal(lock); } virtual void get(boost::unique_lock<boost::mutex>& lock) { this->wait_internal(lock); } void get() { boost::unique_lock<boost::mutex> lock(this->mutex); this->get(lock); } virtual void get_sh(boost::unique_lock<boost::mutex>& lock) { this->wait_internal(lock); } void get_sh() { boost::unique_lock<boost::mutex> lock(this->mutex); this->get_sh(lock); } void set_value_deferred() { unique_lock<boost::mutex> lk(this->mutex); if (this->has_value(lk)) { throw_exception(promise_already_satisfied()); } this->is_constructed = true; } void set_value_at_thread_exit() { set_value_deferred(); // unique_lock<boost::mutex> lk(this->mutex); // if (this->has_value(lk)) // { // throw_exception(promise_already_satisfied()); // } // this->is_constructed = true; detail::make_ready_at_thread_exit(shared_from_this()); } private: shared_state(shared_state const&); shared_state& operator=(shared_state const&); }; ///////////////////////// /// future_async_shared_state_base ///////////////////////// template<typename Rp> struct future_async_shared_state_base: shared_state<Rp> { typedef shared_state<Rp> base_type; protected: #ifdef BOOST_THREAD_FUTURE_BLOCKING boost::thread thr_; void join() { if (this_thread::get_id() == thr_.get_id()) { thr_.detach(); return; } if (thr_.joinable()) thr_.join(); } #endif public: future_async_shared_state_base() { this->set_async(); } ~future_async_shared_state_base() { #ifdef BOOST_THREAD_FUTURE_BLOCKING join(); #elif defined BOOST_THREAD_ASYNC_FUTURE_WAITS unique_lock<boost::mutex> lk(this->mutex); this->waiters.wait(lk, boost::bind(&shared_state_base::is_done, boost::ref(*this))); #endif } virtual void wait(boost::unique_lock<boost::mutex>& lk, bool rethrow) { #ifdef BOOST_THREAD_FUTURE_BLOCKING { relocker rlk(lk); join(); } #endif this->base_type::wait(lk, rethrow); } }; ///////////////////////// /// future_async_shared_state ///////////////////////// template<typename Rp, typename Fp> struct future_async_shared_state: future_async_shared_state_base<Rp> { future_async_shared_state() { } void init(BOOST_THREAD_FWD_REF(Fp) f) { #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::forward<Fp>(f)); #else boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::forward<Fp>(f)).detach(); #endif } static void run(shared_ptr<future_async_shared_state> that, BOOST_THREAD_FWD_REF(Fp) f) { try { that->mark_finished_with_result(f()); } catch(...) { that->mark_exceptional_finish(); } } }; template<typename Fp> struct future_async_shared_state<void, Fp>: public future_async_shared_state_base<void> { void init(BOOST_THREAD_FWD_REF(Fp) f) { #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::move(f)); #else boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::move(f)).detach(); #endif } static void run(shared_ptr<future_async_shared_state> that, BOOST_THREAD_FWD_REF(Fp) f) { try { f(); that->mark_finished_with_result(); } catch(...) { that->mark_exceptional_finish(); } } }; template<typename Rp, typename Fp> struct future_async_shared_state<Rp&, Fp>: future_async_shared_state_base<Rp&> { void init(BOOST_THREAD_FWD_REF(Fp) f) { #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::move(f)); #else boost::thread(&future_async_shared_state::run, static_shared_from_this(this), boost::move(f)).detach(); #endif } static void run(shared_ptr<future_async_shared_state> that, BOOST_THREAD_FWD_REF(Fp) f) { try { that->mark_finished_with_result(f()); } catch(...) { that->mark_exceptional_finish(); } } }; ////////////////////////// /// future_deferred_shared_state ////////////////////////// template<typename Rp, typename Fp> struct future_deferred_shared_state: shared_state<Rp> { Fp func_; explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f) : func_(boost::move(f)) { this->set_deferred(); } virtual void execute(boost::unique_lock<boost::mutex>& lck) { try { Fp local_fuct=boost::move(func_); relocker relock(lck); Rp res = local_fuct(); relock.lock(); this->mark_finished_with_result_internal(boost::move(res), lck); } catch (...) { this->mark_exceptional_finish_internal(current_exception(), lck); } } }; template<typename Rp, typename Fp> struct future_deferred_shared_state<Rp&,Fp>: shared_state<Rp&> { Fp func_; explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f) : func_(boost::move(f)) { this->set_deferred(); } virtual void execute(boost::unique_lock<boost::mutex>& lck) { try { this->mark_finished_with_result_internal(func_(), lck); } catch (...) { this->mark_exceptional_finish_internal(current_exception(), lck); } } }; template<typename Fp> struct future_deferred_shared_state<void,Fp>: shared_state<void> { Fp func_; explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f) : func_(boost::move(f)) { this->set_deferred(); } virtual void execute(boost::unique_lock<boost::mutex>& lck) { try { Fp local_fuct=boost::move(func_); relocker relock(lck); local_fuct(); relock.lock(); this->mark_finished_with_result_internal(lck); } catch (...) { this->mark_exceptional_finish_internal(current_exception(), lck); } } }; class future_waiter { public: typedef std::vector<int>::size_type count_type; private: struct registered_waiter { boost::shared_ptr<detail::shared_state_base> future_; detail::shared_state_base::notify_when_ready_handle handle; count_type index; registered_waiter(boost::shared_ptr<detail::shared_state_base> const& a_future, detail::shared_state_base::notify_when_ready_handle handle_, count_type index_): future_(a_future),handle(handle_),index(index_) {} }; struct all_futures_lock { #ifdef _MANAGED typedef std::ptrdiff_t count_type_portable; #else typedef count_type count_type_portable; #endif count_type_portable count; boost::scoped_array<boost::unique_lock<boost::mutex> > locks; all_futures_lock(std::vector<registered_waiter>& futures): count(futures.size()),locks(new boost::unique_lock<boost::mutex>[count]) { for(count_type_portable i=0;i<count;++i) { locks[i]=BOOST_THREAD_MAKE_RV_REF(boost::unique_lock<boost::mutex>(futures[i].future_->mutex)); } } void lock() { boost::lock(locks.get(),locks.get()+count); } void unlock() { for(count_type_portable i=0;i<count;++i) { locks[i].unlock(); } } }; boost::condition_variable_any cv; std::vector<registered_waiter> futures_; count_type future_count; public: future_waiter(): future_count(0) {} template<typename F> void add(F& f) { if(f.future_) { registered_waiter waiter(f.future_,f.future_->notify_when_ready(cv),future_count); try { futures_.push_back(waiter); } catch(...) { f.future_->unnotify_when_ready(waiter.handle); throw; } } ++future_count; } #ifndef BOOST_NO_CXX11_VARIADIC_TEMPLATES template<typename F1, typename... Fs> void add(F1& f1, Fs&... fs) { add(f1); add(fs...); } #endif count_type wait() { all_futures_lock lk(futures_); for(;;) { for(count_type i=0;i<futures_.size();++i) { if(futures_[i].future_->done) { return futures_[i].index; } } cv.wait(lk); } } ~future_waiter() { for(count_type i=0;i<futures_.size();++i) { futures_[i].future_->unnotify_when_ready(futures_[i].handle); } } }; } template <typename R> class BOOST_THREAD_FUTURE; template <typename R> class shared_future; template<typename T> struct is_future_type<BOOST_THREAD_FUTURE<T> > : true_type { }; template<typename T> struct is_future_type<shared_future<T> > : true_type { }; // template<typename Iterator> // typename boost::disable_if<is_future_type<Iterator>,Iterator>::type wait_for_any(Iterator begin,Iterator end) // { // if(begin==end) // return end; // // detail::future_waiter waiter; // for(Iterator current=begin;current!=end;++current) // { // waiter.add(*current); // } // return boost::next(begin,waiter.wait()); // } #ifdef BOOST_NO_CXX11_VARIADIC_TEMPLATES template<typename F1,typename F2> typename boost::enable_if<is_future_type<F1>,typename detail::future_waiter::count_type>::type wait_for_any(F1& f1,F2& f2) { detail::future_waiter waiter; waiter.add(f1); waiter.add(f2); return waiter.wait(); } template<typename F1,typename F2,typename F3> typename detail::future_waiter::count_type wait_for_any(F1& f1,F2& f2,F3& f3) { detail::future_waiter waiter; waiter.add(f1); waiter.add(f2); waiter.add(f3); return waiter.wait(); } template<typename F1,typename F2,typename F3,typename F4> typename detail::future_waiter::count_type wait_for_any(F1& f1,F2& f2,F3& f3,F4& f4) { detail::future_waiter waiter; waiter.add(f1); waiter.add(f2); waiter.add(f3); waiter.add(f4); return waiter.wait(); } template<typename F1,typename F2,typename F3,typename F4,typename F5> typename detail::future_waiter::count_type wait_for_any(F1& f1,F2& f2,F3& f3,F4& f4,F5& f5) { detail::future_waiter waiter; waiter.add(f1); waiter.add(f2); waiter.add(f3); waiter.add(f4); waiter.add(f5); return waiter.wait(); } #else template<typename F1, typename... Fs> typename boost::enable_if<is_future_type<F1>, typename detail::future_waiter::count_type>::type wait_for_any(F1& f1, Fs&... fs) { detail::future_waiter waiter; waiter.add(f1, fs...); return waiter.wait(); } #endif // !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template <typename R> class promise; template <typename R> class packaged_task; namespace detail { /// Common implementation for all the futures independently of the return type class base_future { public: }; /// Common implementation for future and shared_future. template <typename R> class basic_future : public base_future { protected: public: typedef boost::shared_ptr<detail::shared_state<R> > future_ptr; typedef typename detail::shared_state<R>::move_dest_type move_dest_type; static //BOOST_CONSTEXPR future_ptr make_exceptional_future_ptr(exceptional_ptr const& ex) { return future_ptr(new detail::shared_state<R>(ex)); } future_ptr future_; basic_future(future_ptr a_future): future_(a_future) { } public: typedef future_state::state state; BOOST_THREAD_MOVABLE_ONLY(basic_future) basic_future(): future_() {} //BOOST_CONSTEXPR basic_future(exceptional_ptr const& ex) : future_(make_exceptional_future_ptr(ex)) { } ~basic_future() { } basic_future(BOOST_THREAD_RV_REF(basic_future) other) BOOST_NOEXCEPT: future_(BOOST_THREAD_RV(other).future_) { BOOST_THREAD_RV(other).future_.reset(); } basic_future& operator=(BOOST_THREAD_RV_REF(basic_future) other) BOOST_NOEXCEPT { future_=BOOST_THREAD_RV(other).future_; BOOST_THREAD_RV(other).future_.reset(); return *this; } void swap(basic_future& that) BOOST_NOEXCEPT { future_.swap(that.future_); } // functions to check state, and wait for ready state get_state(boost::unique_lock<boost::mutex>& lk) const { if(!future_) { return future_state::uninitialized; } return future_->get_state(lk); } state get_state() const { if(!future_) { return future_state::uninitialized; } return future_->get_state(); } bool is_ready() const { return get_state()==future_state::ready; } bool is_ready(boost::unique_lock<boost::mutex>& lk) const { return get_state(lk)==future_state::ready; } bool has_exception() const { return future_ && future_->has_exception(); } bool has_value() const { return future_ && future_->has_value(); } launch launch_policy(boost::unique_lock<boost::mutex>& lk) const { if ( future_ ) return future_->launch_policy(lk); else return launch(launch::none); } launch launch_policy() const { if ( future_ ) { boost::unique_lock<boost::mutex> lk(this->future_->mutex); return future_->launch_policy(lk); } else return launch(launch::none); } exception_ptr get_exception_ptr() { return future_ ? future_->get_exception_ptr() : exception_ptr(); } bool valid() const BOOST_NOEXCEPT { return future_.get() != 0 && future_->valid(); } void wait() const { if(!future_) { boost::throw_exception(future_uninitialized()); } future_->wait(false); } typedef detail::shared_state_base::notify_when_ready_handle notify_when_ready_handle; boost::mutex& mutex() { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->mutex; } notify_when_ready_handle notify_when_ready(boost::condition_variable_any& cv) { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->notify_when_ready(cv); } void unnotify_when_ready(notify_when_ready_handle h) { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->unnotify_when_ready(h); } #if defined BOOST_THREAD_USES_DATETIME template<typename Duration> bool timed_wait(Duration const& rel_time) const { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->timed_wait(rel_time); } bool timed_wait_until(boost::system_time const& abs_time) const { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->timed_wait_until(abs_time); } #endif #ifdef BOOST_THREAD_USES_CHRONO template <class Rep, class Period> future_status wait_for(const chrono::duration<Rep, Period>& rel_time) const { return wait_until(chrono::steady_clock::now() + rel_time); } template <class Clock, class Duration> future_status wait_until(const chrono::time_point<Clock, Duration>& abs_time) const { if(!future_) { boost::throw_exception(future_uninitialized()); } return future_->wait_until(abs_time); } #endif }; } // detail BOOST_THREAD_DCL_MOVABLE_BEG(R) detail::basic_future<R> BOOST_THREAD_DCL_MOVABLE_END namespace detail { #if (!defined _MSC_VER || _MSC_VER >= 1400) // _MSC_VER == 1400 on MSVC 2005 template <class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f); template <class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f); #endif // #if (!defined _MSC_VER || _MSC_VER >= 1400) #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template<typename F, typename Rp, typename Fp> struct future_deferred_continuation_shared_state; template<typename F, typename Rp, typename Fp> struct future_async_continuation_shared_state; template <class F, class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename Ex, typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template <class Rp, class Fp, class Executor> BOOST_THREAD_FUTURE<Rp> make_future_executor_shared_state(Executor& ex, BOOST_THREAD_FWD_REF(Fp) f); #endif #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP template<typename F, typename Rp> struct future_unwrap_shared_state; template <class F, class Rp> inline BOOST_THREAD_FUTURE<Rp> make_future_unwrap_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f); #endif } #if defined(BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY) template< typename InputIterator> typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_all(InputIterator first, InputIterator last); inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_all(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_all(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif template< typename InputIterator> typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_any(InputIterator first, InputIterator last); inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_any(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_any(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif #endif // BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY template <typename R> class BOOST_THREAD_FUTURE : public detail::basic_future<R> { private: typedef detail::basic_future<R> base_type; typedef typename base_type::future_ptr future_ptr; friend class shared_future<R>; friend class promise<R>; #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template <typename, typename, typename> friend struct detail::future_async_continuation_shared_state; template <typename, typename, typename> friend struct detail::future_deferred_continuation_shared_state; template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename Ex, typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template <class Rp, class Fp, class Executor> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_executor_shared_state(Executor& ex, BOOST_THREAD_FWD_REF(Fp) f); #endif #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP template<typename F, typename Rp> friend struct detail::future_unwrap_shared_state; template <class F, class Rp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_unwrap_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f); #endif #if defined(BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY) template< typename InputIterator> friend typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_all(InputIterator first, InputIterator last); //friend inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_all(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> friend BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_all(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif template< typename InputIterator> friend typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_any(InputIterator first, InputIterator last); //friend inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_any(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> friend BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_any(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif #endif // BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK template <class> friend class packaged_task; // todo check if this works in windows #else friend class packaged_task<R>; #endif friend class detail::future_waiter; template <class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f); template <class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f); typedef typename base_type::move_dest_type move_dest_type; BOOST_THREAD_FUTURE(future_ptr a_future): base_type(a_future) { } public: BOOST_THREAD_MOVABLE_ONLY(BOOST_THREAD_FUTURE) typedef future_state::state state; typedef R value_type; // EXTENSION BOOST_CONSTEXPR BOOST_THREAD_FUTURE() {} //BOOST_CONSTEXPR BOOST_THREAD_FUTURE(exceptional_ptr const& ex): base_type(ex) {} ~BOOST_THREAD_FUTURE() { } BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE) other) BOOST_NOEXCEPT: base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))) { } #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP inline explicit BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R> >) other); // EXTENSION #endif explicit BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(shared_future<R>) other) : base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))) {} BOOST_THREAD_FUTURE& operator=(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE) other) BOOST_NOEXCEPT { this->base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))); return *this; } shared_future<R> share() { return shared_future<R>(::boost::move(*this)); } void swap(BOOST_THREAD_FUTURE& other) { static_cast<base_type*>(this)->swap(other); } // todo this function must be private and friendship provided to the internal users. void set_async() { this->future_->set_async(); } // todo this function must be private and friendship provided to the internal users. void set_deferred() { this->future_->set_deferred(); } bool run_if_is_deferred() { return this->future_->run_if_is_deferred(); } bool run_if_is_deferred_or_ready() { return this->future_->run_if_is_deferred_or_ready(); } // retrieving the value move_dest_type get() { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif return this->future_->get(lk); } template <typename R2> typename boost::disable_if< is_void<R2>, move_dest_type>::type get_or(BOOST_THREAD_RV_REF(R2) v) { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } this->future_->wait(lk, false); #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif if (this->future_->has_value(lk)) { return this->future_->get(lk); } else { return boost::move(v); } } template <typename R2> typename boost::disable_if< is_void<R2>, move_dest_type>::type get_or(R2 const& v) // EXTENSION { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } this->future_->wait(lk, false); #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif if (this->future_->has_value(lk)) { return this->future_->get(lk); } else { return v; } } #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(BOOST_THREAD_FWD_REF(F) func); // EXTENSION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(launch policy, BOOST_THREAD_FWD_REF(F) func); // EXTENSION #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(Ex& ex, BOOST_THREAD_FWD_REF(F) func); // EXTENSION #endif template <typename R2> inline typename boost::disable_if< is_void<R2>, BOOST_THREAD_FUTURE<R> >::type fallback_to(BOOST_THREAD_RV_REF(R2) v); // EXTENSION template <typename R2> inline typename boost::disable_if< is_void<R2>, BOOST_THREAD_FUTURE<R> >::type fallback_to(R2 const& v); // EXTENSION #endif }; BOOST_THREAD_DCL_MOVABLE_BEG(T) BOOST_THREAD_FUTURE<T> BOOST_THREAD_DCL_MOVABLE_END template <typename R2> class BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> > : public detail::basic_future<BOOST_THREAD_FUTURE<R2> > { typedef BOOST_THREAD_FUTURE<R2> R; private: typedef detail::basic_future<R> base_type; typedef typename base_type::future_ptr future_ptr; friend class shared_future<R>; friend class promise<R>; #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template <typename, typename, typename> friend struct detail::future_async_continuation_shared_state; template <typename, typename, typename> friend struct detail::future_deferred_continuation_shared_state; template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template<typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template<typename Ex, typename F, typename Rp, typename Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_shared_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c); template <class Rp, class Fp, class Executor> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_executor_shared_state(Executor& ex, BOOST_THREAD_FWD_REF(Fp) f); #endif #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP template<typename F, typename Rp> friend struct detail::future_unwrap_shared_state; template <class F, class Rp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_unwrap_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f); #endif #if defined(BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY) template< typename InputIterator> friend typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_all(InputIterator first, InputIterator last); friend inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_all(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> friend BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_all(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif template< typename InputIterator> friend typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_any(InputIterator first, InputIterator last); friend inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_any(); #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> friend BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_any(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures); #endif #endif // BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK template <class> friend class packaged_task; // todo check if this works in windows #else friend class packaged_task<R>; #endif friend class detail::future_waiter; template <class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f); template <class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f); typedef typename base_type::move_dest_type move_dest_type; BOOST_THREAD_FUTURE(future_ptr a_future): base_type(a_future) { } public: BOOST_THREAD_MOVABLE_ONLY(BOOST_THREAD_FUTURE) typedef future_state::state state; typedef R value_type; // EXTENSION BOOST_CONSTEXPR BOOST_THREAD_FUTURE() {} //BOOST_CONSTEXPR BOOST_THREAD_FUTURE(exceptional_ptr const& ex): base_type(ex) {} ~BOOST_THREAD_FUTURE() { } BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE) other) BOOST_NOEXCEPT: base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))) { } BOOST_THREAD_FUTURE& operator=(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE) other) BOOST_NOEXCEPT { this->base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))); return *this; } shared_future<R> share() { return shared_future<R>(::boost::move(*this)); } void swap(BOOST_THREAD_FUTURE& other) { static_cast<base_type*>(this)->swap(other); } // todo this function must be private and friendship provided to the internal users. void set_async() { this->future_->set_async(); } // todo this function must be private and friendship provided to the internal users. void set_deferred() { this->future_->set_deferred(); } bool run_if_is_deferred() { return this->future_->run_if_is_deferred(); } bool run_if_is_deferred_or_ready() { return this->future_->run_if_is_deferred_or_ready(); } // retrieving the value move_dest_type get() { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif return this->future_->get(lk); } move_dest_type get_or(BOOST_THREAD_RV_REF(R) v) // EXTENSION { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } this->future_->wait(lk, false); #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif if (this->future_->has_value(lk)) return this->future_->get(lk); else return boost::move(v); } move_dest_type get_or(R const& v) // EXTENSION { if (this->future_.get() == 0) { boost::throw_exception(future_uninitialized()); } unique_lock<boost::mutex> lk(this->future_->mutex); if (! this->future_->valid(lk)) { boost::throw_exception(future_uninitialized()); } this->future_->wait(lk, false); #ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET this->future_->invalidate(lk); #endif if (this->future_->has_value(lk)) return this->future_->get(lk); else return v; } #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(BOOST_THREAD_FWD_REF(F) func); // EXTENSION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(launch policy, BOOST_THREAD_FWD_REF(F) func); // EXTENSION #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type> then(Ex &ex, BOOST_THREAD_FWD_REF(F) func); // EXTENSION #endif #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP inline BOOST_THREAD_FUTURE<R2> unwrap(); // EXTENSION #endif }; template <typename R> class shared_future : public detail::basic_future<R> { typedef detail::basic_future<R> base_type; typedef typename base_type::future_ptr future_ptr; friend class detail::future_waiter; friend class promise<R>; #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template <typename, typename, typename> friend struct detail::future_async_continuation_shared_state; template <typename, typename, typename> friend struct detail::future_deferred_continuation_shared_state; template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_sync_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); template <class F, class Rp, class Fp> friend BOOST_THREAD_FUTURE<Rp> detail::make_future_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c); #endif #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK template <class> friend class packaged_task;// todo check if this works in windows #else friend class packaged_task<R>; #endif shared_future(future_ptr a_future): base_type(a_future) {} public: BOOST_THREAD_COPYABLE_AND_MOVABLE(shared_future) typedef R value_type; // EXTENSION shared_future(shared_future const& other): base_type(other.future_) {} typedef future_state::state state; BOOST_CONSTEXPR shared_future() {} //BOOST_CONSTEXPR shared_future(exceptional_ptr const& ex): base_type(ex) {} ~shared_future() {} shared_future& operator=(BOOST_THREAD_COPY_ASSIGN_REF(shared_future) other) { this->future_ = other.future_; return *this; } shared_future(BOOST_THREAD_RV_REF(shared_future) other) BOOST_NOEXCEPT : base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))) { } shared_future(BOOST_THREAD_RV_REF( BOOST_THREAD_FUTURE<R> ) other) BOOST_NOEXCEPT : base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))) { } shared_future& operator=(BOOST_THREAD_RV_REF(shared_future) other) BOOST_NOEXCEPT { base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))); return *this; } shared_future& operator=(BOOST_THREAD_RV_REF( BOOST_THREAD_FUTURE<R> ) other) BOOST_NOEXCEPT { base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other)))); return *this; } void swap(shared_future& other) BOOST_NOEXCEPT { static_cast<base_type*>(this)->swap(other); } bool run_if_is_deferred() { return this->future_->run_if_is_deferred(); } bool run_if_is_deferred_or_ready() { return this->future_->run_if_is_deferred_or_ready(); } // retrieving the value typename detail::shared_state<R>::shared_future_get_result_type get() const { if(!this->future_) { boost::throw_exception(future_uninitialized()); } return this->future_->get_sh(); } template <typename R2> typename boost::disable_if< is_void<R2>, typename detail::shared_state<R>::shared_future_get_result_type>::type get_or(BOOST_THREAD_RV_REF(R2) v) const // EXTENSION { if(!this->future_) { boost::throw_exception(future_uninitialized()); } this->future_->wait(); if (this->future_->has_value()) return this->future_->get_sh(); else return boost::move(v); } #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future)>::type> then(BOOST_THREAD_FWD_REF(F) func) const; // EXTENSION template<typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future)>::type> then(launch policy, BOOST_THREAD_FWD_REF(F) func) const; // EXTENSION #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future)>::type> then(Ex& ex, BOOST_THREAD_FWD_REF(F) func) const; // EXTENSION #endif #endif }; BOOST_THREAD_DCL_MOVABLE_BEG(T) shared_future<T> BOOST_THREAD_DCL_MOVABLE_END template <typename R> class promise { typedef boost::shared_ptr<detail::shared_state<R> > future_ptr; typedef typename detail::shared_state<R>::source_reference_type source_reference_type; typedef typename detail::shared_state<R>::rvalue_source_type rvalue_source_type; typedef typename detail::shared_state<R>::move_dest_type move_dest_type; typedef typename detail::shared_state<R>::shared_future_get_result_type shared_future_get_result_type; future_ptr future_; bool future_obtained; void lazy_init() { #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY #include <boost/thread/detail/atomic_undef_macros.hpp> if(!atomic_load(&future_)) { future_ptr blank; atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<R>)); } #include <boost/thread/detail/atomic_redef_macros.hpp> #endif } public: BOOST_THREAD_MOVABLE_ONLY(promise) #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS template <class Allocator> promise(boost::allocator_arg_t, Allocator a) { typedef typename Allocator::template rebind<detail::shared_state<R> >::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<R>(), D(a2, 1) ); future_obtained = false; } #endif promise(): #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY future_(), #else future_(new detail::shared_state<R>()), #endif future_obtained(false) {} ~promise() { if(future_) { boost::unique_lock<boost::mutex> lock(future_->mutex); if(!future_->done && !future_->is_constructed) { future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock); } } } // Assignment promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT : future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained) { BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; } promise & operator=(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT { future_=BOOST_THREAD_RV(rhs).future_; future_obtained=BOOST_THREAD_RV(rhs).future_obtained; BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; return *this; } void swap(promise& other) { future_.swap(other.future_); std::swap(future_obtained,other.future_obtained); } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS void set_executor(executor_ptr_type aex) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } boost::lock_guard<boost::mutex> lk(future_->mutex); future_->set_executor_policy(aex, lk); } #endif // Result retrieval BOOST_THREAD_FUTURE<R> get_future() { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } if (future_obtained) { boost::throw_exception(future_already_retrieved()); } future_obtained=true; return BOOST_THREAD_FUTURE<R>(future_); } #if defined BOOST_NO_CXX11_RVALUE_REFERENCES template <class TR> typename boost::enable_if_c<is_copy_constructible<TR>::value && is_same<R, TR>::value, void>::type set_value(TR const & r) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_finished_with_result_internal(r, lock); } #else void set_value(source_reference_type r) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_finished_with_result_internal(r, lock); } #endif void set_value(rvalue_source_type r) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES future_->mark_finished_with_result_internal(boost::move(r), lock); #else future_->mark_finished_with_result_internal(static_cast<rvalue_source_type>(r), lock); #endif } #if defined BOOST_NO_CXX11_RVALUE_REFERENCES template <class TR> typename boost::enable_if_c<is_copy_constructible<TR>::value && is_same<R, TR>::value, void>::type set_value_deferred(TR const & r) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_deferred(r); } #else void set_value_deferred(source_reference_type r) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_deferred(r); } #endif void set_value_deferred(rvalue_source_type r) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES future_->set_value_deferred(boost::move(r)); #else future_->set_value_deferred(static_cast<rvalue_source_type>(r)); #endif } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template <class ...Args> void emplace(BOOST_THREAD_FWD_REF(Args) ...args) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_finished_with_result_internal(lock, boost::forward<Args>(args)...); } #endif void set_exception(boost::exception_ptr p) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_exceptional_finish_internal(p, lock); } template <typename E> void set_exception(E ex) { set_exception(boost::copy_exception(ex)); } void set_exception_deferred(boost::exception_ptr p) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_deferred(p); } template <typename E> void set_exception_deferred(E ex) { set_exception_deferred(boost::copy_exception(ex)); } // setting the result with deferred notification #if defined BOOST_NO_CXX11_RVALUE_REFERENCES template <class TR> typename boost::enable_if_c<is_copy_constructible<TR>::value && is_same<R, TR>::value, void>::type set_value_at_thread_exit(TR const& r) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_at_thread_exit(r); } #else void set_value_at_thread_exit(source_reference_type r) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_at_thread_exit(r); } #endif void set_value_at_thread_exit(BOOST_THREAD_RV_REF(R) r) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_at_thread_exit(boost::move(r)); } void set_exception_at_thread_exit(exception_ptr e) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_at_thread_exit(e); } template <typename E> void set_exception_at_thread_exit(E ex) { set_exception_at_thread_exit(boost::copy_exception(ex)); } template<typename F> void set_wait_callback(F f) { lazy_init(); future_->set_wait_callback(f,this); } void notify_deferred() { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->notify_deferred(); } }; template <typename R> class promise<R&> { typedef boost::shared_ptr<detail::shared_state<R&> > future_ptr; future_ptr future_; bool future_obtained; void lazy_init() { #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY #include <boost/thread/detail/atomic_undef_macros.hpp> if(!atomic_load(&future_)) { future_ptr blank; atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<R&>)); } #include <boost/thread/detail/atomic_redef_macros.hpp> #endif } public: BOOST_THREAD_MOVABLE_ONLY(promise) #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS template <class Allocator> promise(boost::allocator_arg_t, Allocator a) { typedef typename Allocator::template rebind<detail::shared_state<R&> >::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<R&>(), D(a2, 1) ); future_obtained = false; } #endif promise(): #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY future_(), #else future_(new detail::shared_state<R&>()), #endif future_obtained(false) {} ~promise() { if(future_) { boost::unique_lock<boost::mutex> lock(future_->mutex); if(!future_->done && !future_->is_constructed) { future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock); } } } // Assignment promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT : future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained) { BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; } promise & operator=(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT { future_=BOOST_THREAD_RV(rhs).future_; future_obtained=BOOST_THREAD_RV(rhs).future_obtained; BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; return *this; } void swap(promise& other) { future_.swap(other.future_); std::swap(future_obtained,other.future_obtained); } // Result retrieval BOOST_THREAD_FUTURE<R&> get_future() { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } if (future_obtained) { boost::throw_exception(future_already_retrieved()); } future_obtained=true; return BOOST_THREAD_FUTURE<R&>(future_); } void set_value(R& r) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_finished_with_result_internal(r, lock); } void set_value_deferred(R& r) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_already_satisfied()); } future_->set_value_deferred(r); } void set_exception(boost::exception_ptr p) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_exceptional_finish_internal(p, lock); } template <typename E> void set_exception(E ex) { set_exception(boost::copy_exception(ex)); } void set_exception_deferred(boost::exception_ptr p) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_deferred(p); } template <typename E> void set_exception_deferred(E ex) { set_exception_deferred(boost::copy_exception(ex)); } // setting the result with deferred notification void set_value_at_thread_exit(R& r) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_at_thread_exit(r); } void set_exception_at_thread_exit(exception_ptr e) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_at_thread_exit(e); } template <typename E> void set_exception_at_thread_exit(E ex) { set_exception_at_thread_exit(boost::copy_exception(ex)); } template<typename F> void set_wait_callback(F f) { lazy_init(); future_->set_wait_callback(f,this); } void notify_deferred() { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->notify_deferred(); } }; template <> class promise<void> { typedef boost::shared_ptr<detail::shared_state<void> > future_ptr; future_ptr future_; bool future_obtained; void lazy_init() { #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY if(!atomic_load(&future_)) { future_ptr blank; atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<void>)); } #endif } public: BOOST_THREAD_MOVABLE_ONLY(promise) #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS template <class Allocator> promise(boost::allocator_arg_t, Allocator a) { typedef typename Allocator::template rebind<detail::shared_state<void> >::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<void>(), D(a2, 1) ); future_obtained = false; } #endif promise(): #if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY future_(), #else future_(new detail::shared_state<void>), #endif future_obtained(false) {} ~promise() { if(future_) { boost::unique_lock<boost::mutex> lock(future_->mutex); if(!future_->done && !future_->is_constructed) { future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock); } } } // Assignment promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT : future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained) { // we need to release the future as shared_ptr doesn't implements move semantics BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; } promise & operator=(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT { future_=BOOST_THREAD_RV(rhs).future_; future_obtained=BOOST_THREAD_RV(rhs).future_obtained; BOOST_THREAD_RV(rhs).future_.reset(); BOOST_THREAD_RV(rhs).future_obtained=false; return *this; } void swap(promise& other) { future_.swap(other.future_); std::swap(future_obtained,other.future_obtained); } // Result retrieval BOOST_THREAD_FUTURE<void> get_future() { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } if(future_obtained) { boost::throw_exception(future_already_retrieved()); } future_obtained=true; //return BOOST_THREAD_MAKE_RV_REF(BOOST_THREAD_FUTURE<void>(future_)); return BOOST_THREAD_FUTURE<void>(future_); } void set_value() { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_finished_with_result_internal(lock); } void set_value_deferred() { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_deferred(); } void set_exception(boost::exception_ptr p) { lazy_init(); boost::unique_lock<boost::mutex> lock(future_->mutex); if(future_->done) { boost::throw_exception(promise_already_satisfied()); } future_->mark_exceptional_finish_internal(p,lock); } template <typename E> void set_exception(E ex) { set_exception(boost::copy_exception(ex)); } void set_exception_deferred(boost::exception_ptr p) { lazy_init(); if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_deferred(p); } template <typename E> void set_exception_deferred(E ex) { set_exception_deferred(boost::copy_exception(ex)); } // setting the result with deferred notification void set_value_at_thread_exit() { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_value_at_thread_exit(); } void set_exception_at_thread_exit(exception_ptr e) { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->set_exception_at_thread_exit(e); } template <typename E> void set_exception_at_thread_exit(E ex) { set_exception_at_thread_exit(boost::copy_exception(ex)); } template<typename F> void set_wait_callback(F f) { lazy_init(); future_->set_wait_callback(f,this); } void notify_deferred() { if (future_.get()==0) { boost::throw_exception(promise_moved()); } future_->notify_deferred(); } }; } #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS namespace boost { namespace container { template <class R, class Alloc> struct uses_allocator< ::boost::promise<R> , Alloc> : true_type { }; }} #if ! defined BOOST_NO_CXX11_ALLOCATOR namespace std { template <class R, class Alloc> struct uses_allocator< ::boost::promise<R> , Alloc> : true_type { }; } #endif #endif namespace boost { BOOST_THREAD_DCL_MOVABLE_BEG(T) promise<T> BOOST_THREAD_DCL_MOVABLE_END namespace detail { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK template<typename R> struct task_base_shared_state; #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename R, typename ...ArgTypes> struct task_base_shared_state<R(ArgTypes...)>: #else template<typename R> struct task_base_shared_state<R()>: #endif #else template<typename R> struct task_base_shared_state: #endif detail::shared_state<R> { bool started; task_base_shared_state(): started(false) {} void reset() { // todo The packaged_task::reset must be as if an assignemnt froma new packaged_task with the same function // the reset function is an optimization that avoids reallocating a new task. started=false; this->validate(); } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) virtual void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args)=0; void run(BOOST_THREAD_RV_REF(ArgTypes) ... args) #else virtual void do_run()=0; void run() #endif { { boost::lock_guard<boost::mutex> lk(this->mutex); if(started) { boost::throw_exception(task_already_started()); } started=true; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) do_run(boost::move(args)...); #else do_run(); #endif } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) virtual void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args)=0; void apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) #else virtual void do_apply()=0; void apply() #endif { { boost::lock_guard<boost::mutex> lk(this->mutex); if(started) { boost::throw_exception(task_already_started()); } started=true; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) do_apply(boost::move(args)...); #else do_apply(); #endif } void owner_destroyed() { boost::unique_lock<boost::mutex> lk(this->mutex); if(!started) { started=true; this->mark_exceptional_finish_internal(boost::copy_exception(boost::broken_promise()), lk); } } }; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK template<typename F, typename R> struct task_shared_state; #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename F, typename R, typename ...ArgTypes> struct task_shared_state<F, R(ArgTypes...)>: task_base_shared_state<R(ArgTypes...)> #else template<typename F, typename R> struct task_shared_state<F, R()>: task_base_shared_state<R()> #endif #else template<typename F, typename R> struct task_shared_state: task_base_shared_state<R> #endif { private: task_shared_state(task_shared_state&); public: F f; task_shared_state(F const& f_): f(f_) {} task_shared_state(BOOST_THREAD_RV_REF(F) f_): f(boost::move(f_)) {} F callable() { return boost::move(f); } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->set_value_at_thread_exit(f(boost::move(args)...)); } #else void do_apply() { try { this->set_value_at_thread_exit(f()); } #endif catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->mark_finished_with_result(f(boost::move(args)...)); } #else void do_run() { try { #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES R res((f())); this->mark_finished_with_result(boost::move(res)); #else this->mark_finished_with_result(f()); #endif } #endif catch(...) { this->mark_exceptional_finish(); } } }; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename F, typename R, typename ...ArgTypes> struct task_shared_state<F, R&(ArgTypes...)>: task_base_shared_state<R&(ArgTypes...)> #else template<typename F, typename R> struct task_shared_state<F, R&()>: task_base_shared_state<R&()> #endif #else template<typename F, typename R> struct task_shared_state<F,R&>: task_base_shared_state<R&> #endif { private: task_shared_state(task_shared_state&); public: F f; task_shared_state(F const& f_): f(f_) {} task_shared_state(BOOST_THREAD_RV_REF(F) f_): f(boost::move(f_)) {} F callable() { return f; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->set_value_at_thread_exit(f(boost::move(args)...)); } #else void do_apply() { try { this->set_value_at_thread_exit(f()); } #endif catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->mark_finished_with_result(f(boost::move(args)...)); } #else void do_run() { try { R& res((f())); this->mark_finished_with_result(res); } #endif catch(...) { this->mark_exceptional_finish(); } } }; #if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR) #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename R, typename ...ArgTypes> struct task_shared_state<R (*)(ArgTypes...), R(ArgTypes...)>: task_base_shared_state<R(ArgTypes...)> #else template<typename R> struct task_shared_state<R (*)(), R()>: task_base_shared_state<R()> #endif #else template<typename R> struct task_shared_state<R (*)(), R> : task_base_shared_state<R> #endif { private: task_shared_state(task_shared_state&); #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef R (*CallableType)(ArgTypes ... ); #else typedef R (*CallableType)(); #endif public: CallableType f; task_shared_state(CallableType f_): f(f_) {} CallableType callable() { return f; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->set_value_at_thread_exit(f(boost::move(args)...)); } #else void do_apply() { try { R r((f())); this->set_value_at_thread_exit(boost::move(r)); } #endif catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->mark_finished_with_result(f(boost::move(args)...)); } #else void do_run() { try { R res((f())); this->mark_finished_with_result(boost::move(res)); } #endif catch(...) { this->mark_exceptional_finish(); } } }; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename R, typename ...ArgTypes> struct task_shared_state<R& (*)(ArgTypes...), R&(ArgTypes...)>: task_base_shared_state<R&(ArgTypes...)> #else template<typename R> struct task_shared_state<R& (*)(), R&()>: task_base_shared_state<R&()> #endif #else template<typename R> struct task_shared_state<R& (*)(), R&> : task_base_shared_state<R&> #endif { private: task_shared_state(task_shared_state&); public: #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef R& (*CallableType)(BOOST_THREAD_RV_REF(ArgTypes) ... ); #else typedef R& (*CallableType)(); #endif CallableType f; task_shared_state(CallableType f_): f(f_) {} CallableType callable() { return boost::move(f); } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->set_value_at_thread_exit(f(boost::move(args)...)); } #else void do_apply() { try { this->set_value_at_thread_exit(f()); } #endif catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { this->mark_finished_with_result(f(boost::move(args)...)); } #else void do_run() { try { this->mark_finished_with_result(f()); } #endif catch(...) { this->mark_exceptional_finish(); } } }; #endif #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename F, typename ...ArgTypes> struct task_shared_state<F, void(ArgTypes...)>: task_base_shared_state<void(ArgTypes...)> #else template<typename F> struct task_shared_state<F, void()>: task_base_shared_state<void()> #endif #else template<typename F> struct task_shared_state<F,void>: task_base_shared_state<void> #endif { private: task_shared_state(task_shared_state&); public: typedef F CallableType; F f; task_shared_state(F const& f_): f(f_) {} task_shared_state(BOOST_THREAD_RV_REF(F) f_): f(boost::move(f_)) {} F callable() { return boost::move(f); } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { f(boost::move(args)...); #else void do_apply() { try { f(); #endif this->set_value_at_thread_exit(); } catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { f(boost::move(args)...); #else void do_run() { try { f(); #endif this->mark_finished_with_result(); } catch(...) { this->mark_exceptional_finish(); } } }; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename ...ArgTypes> struct task_shared_state<void (*)(ArgTypes...), void(ArgTypes...)>: task_base_shared_state<void(ArgTypes...)> #else template<> struct task_shared_state<void (*)(), void()>: task_base_shared_state<void()> #endif #else template<> struct task_shared_state<void (*)(),void>: task_base_shared_state<void> #endif { private: task_shared_state(task_shared_state&); #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef void (*CallableType)(ArgTypes...); #else typedef void (*CallableType)(); #endif public: CallableType f; task_shared_state(CallableType f_): f(f_) {} CallableType callable() { return f; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { f(boost::move(args)...); #else void do_apply() { try { f(); #endif this->set_value_at_thread_exit(); } catch(...) { this->set_exception_at_thread_exit(current_exception()); } } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void do_run(BOOST_THREAD_RV_REF(ArgTypes) ... args) { try { f(boost::move(args)...); #else void do_run() { try { f(); #endif this->mark_finished_with_result(); } catch(...) { this->mark_exceptional_finish(); } } }; } #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template<typename R, typename ...ArgTypes> class packaged_task<R(ArgTypes...)> { typedef boost::shared_ptr<detail::task_base_shared_state<R(ArgTypes...)> > task_ptr; boost::shared_ptr<detail::task_base_shared_state<R(ArgTypes...)> > task; #else template<typename R> class packaged_task<R()> { typedef boost::shared_ptr<detail::task_base_shared_state<R()> > task_ptr; boost::shared_ptr<detail::task_base_shared_state<R()> > task; #endif #else template<typename R> class packaged_task { typedef boost::shared_ptr<detail::task_base_shared_state<R> > task_ptr; boost::shared_ptr<detail::task_base_shared_state<R> > task; #endif bool future_obtained; struct dummy; public: typedef R result_type; BOOST_THREAD_MOVABLE_ONLY(packaged_task) packaged_task(): future_obtained(false) {} // construction and destruction #if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR) #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) explicit packaged_task(R(*f)(), BOOST_THREAD_FWD_REF(ArgTypes)... args) { typedef R(*FR)(BOOST_THREAD_FWD_REF(ArgTypes)...); typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type; task= task_ptr(new task_shared_state_type(f, boost::move(args)...)); future_obtained=false; } #else explicit packaged_task(R(*f)()) { typedef R(*FR)(); typedef detail::task_shared_state<FR,R()> task_shared_state_type; task= task_ptr(new task_shared_state_type(f)); future_obtained=false; } #endif #else explicit packaged_task(R(*f)()) { typedef R(*FR)(); typedef detail::task_shared_state<FR,R> task_shared_state_type; task= task_ptr(new task_shared_state_type(f)); future_obtained=false; } #endif #endif #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES template <class F> explicit packaged_task(BOOST_THREAD_FWD_REF(F) f , typename boost::disable_if<is_same<typename decay<F>::type, packaged_task>, dummy* >::type=0 ) { typedef typename decay<F>::type FR; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<FR,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<FR,R> task_shared_state_type; #endif task = task_ptr(new task_shared_state_type(boost::forward<F>(f))); future_obtained = false; } #else template <class F> explicit packaged_task(F const& f , typename boost::disable_if<is_same<typename decay<F>::type, packaged_task>, dummy* >::type=0 ) { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<F,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<F,R> task_shared_state_type; #endif task = task_ptr(new task_shared_state_type(f)); future_obtained=false; } template <class F> explicit packaged_task(BOOST_THREAD_RV_REF(F) f) { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type; task = task_ptr(new task_shared_state_type(boost::move(f))); #else typedef detail::task_shared_state<F,R()> task_shared_state_type; task = task_ptr(new task_shared_state_type(boost::move(f))); #endif #else typedef detail::task_shared_state<F,R> task_shared_state_type; task = task_ptr(new task_shared_state_type(boost::move(f))); #endif future_obtained=false; } #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS #if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR) template <class Allocator> packaged_task(boost::allocator_arg_t, Allocator a, R(*f)()) { typedef R(*FR)(); #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<FR,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<FR,R> task_shared_state_type; #endif typedef typename Allocator::template rebind<task_shared_state_type>::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(f), D(a2, 1) ); future_obtained = false; } #endif // BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES template <class F, class Allocator> packaged_task(boost::allocator_arg_t, Allocator a, BOOST_THREAD_FWD_REF(F) f) { typedef typename decay<F>::type FR; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<FR,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<FR,R> task_shared_state_type; #endif typedef typename Allocator::template rebind<task_shared_state_type>::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(boost::forward<F>(f)), D(a2, 1) ); future_obtained = false; } #else // ! defined BOOST_NO_CXX11_RVALUE_REFERENCES template <class F, class Allocator> packaged_task(boost::allocator_arg_t, Allocator a, const F& f) { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<F,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<F,R> task_shared_state_type; #endif typedef typename Allocator::template rebind<task_shared_state_type>::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(f), D(a2, 1) ); future_obtained = false; } template <class F, class Allocator> packaged_task(boost::allocator_arg_t, Allocator a, BOOST_THREAD_RV_REF(F) f) { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type; #else typedef detail::task_shared_state<F,R()> task_shared_state_type; #endif #else typedef detail::task_shared_state<F,R> task_shared_state_type; #endif typedef typename Allocator::template rebind<task_shared_state_type>::other A2; A2 a2(a); typedef thread_detail::allocator_destructor<A2> D; task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(boost::move(f)), D(a2, 1) ); future_obtained = false; } #endif //BOOST_NO_CXX11_RVALUE_REFERENCES #endif // BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS ~packaged_task() { if(task) { task->owner_destroyed(); } } // assignment packaged_task(BOOST_THREAD_RV_REF(packaged_task) other) BOOST_NOEXCEPT : future_obtained(BOOST_THREAD_RV(other).future_obtained) { task.swap(BOOST_THREAD_RV(other).task); BOOST_THREAD_RV(other).future_obtained=false; } packaged_task& operator=(BOOST_THREAD_RV_REF(packaged_task) other) BOOST_NOEXCEPT { #if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES packaged_task temp(boost::move(other)); #else packaged_task temp(static_cast<BOOST_THREAD_RV_REF(packaged_task)>(other)); #endif swap(temp); return *this; } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS void set_executor(executor_ptr_type aex) { if (!valid()) boost::throw_exception(task_moved()); boost::lock_guard<boost::mutex> lk(task->mutex); task->set_executor_policy(aex, lk); } #endif void reset() { if (!valid()) boost::throw_exception(future_error(system::make_error_code(future_errc::no_state))); // As if *this = packaged_task(task->callable()); task->reset(); future_obtained=false; } void swap(packaged_task& other) BOOST_NOEXCEPT { task.swap(other.task); std::swap(future_obtained,other.future_obtained); } bool valid() const BOOST_NOEXCEPT { return task.get()!=0; } // result retrieval BOOST_THREAD_FUTURE<R> get_future() { if(!task) { boost::throw_exception(task_moved()); } else if(!future_obtained) { future_obtained=true; return BOOST_THREAD_FUTURE<R>(task); } else { boost::throw_exception(future_already_retrieved()); } } // execution #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) void operator()(ArgTypes... args) { if(!task) { boost::throw_exception(task_moved()); } task->run(boost::move(args)...); } void make_ready_at_thread_exit(ArgTypes... args) { if(!task) { boost::throw_exception(task_moved()); } if (task->has_value()) { boost::throw_exception(promise_already_satisfied()); } task->apply(boost::move(args)...); } #else void operator()() { if(!task) { boost::throw_exception(task_moved()); } task->run(); } void make_ready_at_thread_exit() { if(!task) { boost::throw_exception(task_moved()); } if (task->has_value()) boost::throw_exception(promise_already_satisfied()); task->apply(); } #endif template<typename F> void set_wait_callback(F f) { task->set_wait_callback(f,this); } }; } #if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS namespace boost { namespace container { template <class R, class Alloc> struct uses_allocator< ::boost::packaged_task<R> , Alloc> : true_type {}; }} #if ! defined BOOST_NO_CXX11_ALLOCATOR namespace std { template <class R, class Alloc> struct uses_allocator< ::boost::packaged_task<R> , Alloc> : true_type {}; } #endif #endif namespace boost { BOOST_THREAD_DCL_MOVABLE_BEG(T) packaged_task<T> BOOST_THREAD_DCL_MOVABLE_END namespace detail { //////////////////////////////// // make_future_deferred_shared_state //////////////////////////////// template <class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f) { shared_ptr<future_deferred_shared_state<Rp, Fp> > h(new future_deferred_shared_state<Rp, Fp>(boost::forward<Fp>(f))); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_future_async_shared_state //////////////////////////////// template <class Rp, class Fp> BOOST_THREAD_FUTURE<Rp> make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f) { shared_ptr<future_async_shared_state<Rp, Fp> > h(new future_async_shared_state<Rp, Fp>()); h->init(boost::forward<Fp>(f)); return BOOST_THREAD_FUTURE<Rp>(h); } } //////////////////////////////// // template <class F, class... ArgTypes> // future<R> async(launch policy, F&&, ArgTypes&&...); //////////////////////////////// #if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class R, class... ArgTypes> BOOST_THREAD_FUTURE<R> async(launch policy, R(*f)(BOOST_THREAD_FWD_REF(ArgTypes)...), BOOST_THREAD_FWD_REF(ArgTypes)... args) { typedef R(*F)(BOOST_THREAD_FWD_REF(ArgTypes)...); typedef detail::invoker<typename decay<F>::type, typename decay<ArgTypes>::type...> BF; typedef typename BF::result_type Rp; if (underlying_cast<int>(policy) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_async_shared_state<Rp>( BF( f , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_deferred_shared_state<Rp>( BF( f , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } else { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); } } #else // defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class R> BOOST_THREAD_FUTURE<R> async(launch policy, R(*f)()) { #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK typedef packaged_task<R()> packaged_task_type; #else typedef packaged_task<R> packaged_task_type; #endif if (underlying_cast<int>(policy) & int(launch::async)) { packaged_task_type pt( f ); BOOST_THREAD_FUTURE<R> ret = BOOST_THREAD_MAKE_RV_REF(pt.get_future()); ret.set_async(); boost::thread( boost::move(pt) ).detach(); return ::boost::move(ret); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); } else { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); } } #endif #endif // defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR) #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class F, class ...ArgTypes> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type( typename decay<ArgTypes>::type... )>::type> async(launch policy, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(ArgTypes)... args) { typedef detail::invoker<typename decay<F>::type, typename decay<ArgTypes>::type...> BF; typedef typename BF::result_type Rp; if (underlying_cast<int>(policy) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_async_shared_state<Rp>( BF( thread_detail::decay_copy(boost::forward<F>(f)) , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_deferred_shared_state<Rp>( BF( thread_detail::decay_copy(boost::forward<F>(f)) , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } else { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); } } #else // defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class F> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type()>::type> async(launch policy, BOOST_THREAD_FWD_REF(F) f) { typedef typename boost::result_of<typename decay<F>::type()>::type R; #if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK typedef packaged_task<R()> packaged_task_type; #else // defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK typedef packaged_task<R> packaged_task_type; #endif // defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK if (underlying_cast<int>(policy) & int(launch::async)) { packaged_task_type pt( boost::forward<F>(f) ); BOOST_THREAD_FUTURE<R> ret = pt.get_future(); ret.set_async(); boost::thread( boost::move(pt) ).detach(); return ::boost::move(ret); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); // return boost::detail::make_future_deferred_shared_state<Rp>( // BF( // thread_detail::decay_copy(boost::forward<F>(f)) // ) // ); } else { std::terminate(); //BOOST_THREAD_FUTURE<R> ret; //return ::boost::move(ret); } } #endif // defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) #ifdef BOOST_THREAD_PROVIDES_EXECUTORS namespace detail { ///////////////////////// /// shared_state_nullary_task ///////////////////////// template<typename Rp, typename Fp> struct shared_state_nullary_task { typedef shared_ptr<shared_state_base > storage_type; storage_type that; Fp f_; public: shared_state_nullary_task(storage_type st, BOOST_THREAD_FWD_REF(Fp) f) : that(st), f_(boost::move(f)) {}; #if ! defined(BOOST_NO_CXX11_RVALUE_REFERENCES) BOOST_THREAD_COPYABLE_AND_MOVABLE(shared_state_nullary_task) shared_state_nullary_task(shared_state_nullary_task const& x) //BOOST_NOEXCEPT : that(x.that), f_(x.f_) {} shared_state_nullary_task& operator=(BOOST_THREAD_COPY_ASSIGN_REF(shared_state_nullary_task) x) //BOOST_NOEXCEPT { if (this != &x) { that=x.that; f_=x.f_; } return *this; } // move shared_state_nullary_task(BOOST_THREAD_RV_REF(shared_state_nullary_task) x) //BOOST_NOEXCEPT : that(x.that), f_(boost::move(x.f_)) { x.that.reset(); } shared_state_nullary_task& operator=(BOOST_THREAD_RV_REF(shared_state_nullary_task) x) //BOOST_NOEXCEPT { if (this != &x) { that=x.that; f_=boost::move(x.f_); x.that.reset(); } return *this; } #endif void operator()() { shared_ptr<shared_state<Rp> > that_ = static_pointer_cast<shared_state<Rp> >(that); try { that_->mark_finished_with_result(f_()); } catch(...) { that_->mark_exceptional_finish(); } } ~shared_state_nullary_task() { } }; template<typename Fp> struct shared_state_nullary_task<void, Fp> { typedef shared_ptr<shared_state_base > storage_type; storage_type that; Fp f_; public: shared_state_nullary_task(storage_type st, BOOST_THREAD_FWD_REF(Fp) f) : that(st), f_(boost::move(f)) {}; #if ! defined(BOOST_NO_CXX11_RVALUE_REFERENCES) BOOST_THREAD_COPYABLE_AND_MOVABLE(shared_state_nullary_task) shared_state_nullary_task(shared_state_nullary_task const& x) //BOOST_NOEXCEPT : that(x.that), f_(x.f_) {} shared_state_nullary_task& operator=(BOOST_THREAD_COPY_ASSIGN_REF(shared_state_nullary_task) x) //BOOST_NOEXCEPT { if (this != &x) { that=x.that; f_=x.f_; } return *this; } // move shared_state_nullary_task(BOOST_THREAD_RV_REF(shared_state_nullary_task) x) BOOST_NOEXCEPT : that(x.that), f_(boost::move(x.f_)) { x.that.reset(); } shared_state_nullary_task& operator=(BOOST_THREAD_RV_REF(shared_state_nullary_task) x) BOOST_NOEXCEPT { if (this != &x) { that=x.that; f_=boost::move(x.f_); x.that.reset(); } return *this; } #endif void operator()() { shared_ptr<shared_state<void> > that_ = static_pointer_cast<shared_state<void> >(that); try { f_(); that_->mark_finished_with_result(); } catch(...) { that_->mark_exceptional_finish(); } } }; } BOOST_THREAD_DCL_MOVABLE_BEG2(R,F) detail::shared_state_nullary_task<R,F> BOOST_THREAD_DCL_MOVABLE_END namespace detail { ///////////////////////// /// future_executor_shared_state_base ///////////////////////// template<typename Rp> struct future_executor_shared_state: shared_state<Rp> { typedef shared_state<Rp> base_type; protected: public: future_executor_shared_state() { } template <class Fp, class Executor> void init(Executor& ex, BOOST_THREAD_FWD_REF(Fp) f) { typedef typename decay<Fp>::type Cont; this->set_executor_policy(executor_ptr_type(new executor_ref<Executor>(ex))); shared_state_nullary_task<Rp,Cont> t(this->shared_from_this(), boost::forward<Fp>(f)); ex.submit(boost::move(t)); } ~future_executor_shared_state() {} }; //////////////////////////////// // make_future_executor_shared_state //////////////////////////////// template <class Rp, class Fp, class Executor> BOOST_THREAD_FUTURE<Rp> make_future_executor_shared_state(Executor& ex, BOOST_THREAD_FWD_REF(Fp) f) { shared_ptr<future_executor_shared_state<Rp> > h(new future_executor_shared_state<Rp>()); h->init(ex, boost::forward<Fp>(f)); return BOOST_THREAD_FUTURE<Rp>(h); } } // detail //////////////////////////////// // template <class Executor, class F, class... ArgTypes> // future<R> async(Executor& ex, F&&, ArgTypes&&...); //////////////////////////////// //#if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) #if defined(BOOST_THREAD_PROVIDES_INVOKE) && ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && ! defined(BOOST_NO_CXX11_HDR_TUPLE) #if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR template <class Executor, class R, class... ArgTypes> BOOST_THREAD_FUTURE<R> async(Executor& ex, R(*f)(BOOST_THREAD_FWD_REF(ArgTypes)...), BOOST_THREAD_FWD_REF(ArgTypes)... args) { typedef R(*F)(BOOST_THREAD_FWD_REF(ArgTypes)...); typedef detail::invoker<typename decay<F>::type, typename decay<ArgTypes>::type...> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( f , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } #endif // defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR template <class Executor, class F, class ...ArgTypes> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type( typename decay<ArgTypes>::type... )>::type> async(Executor& ex, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(ArgTypes)... args) { typedef detail::invoker<typename decay<F>::type, typename decay<ArgTypes>::type...> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( thread_detail::decay_copy(boost::forward<F>(f)) , thread_detail::decay_copy(boost::forward<ArgTypes>(args))... ) )); } #else // ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) #if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR template <class Executor, class R> BOOST_THREAD_FUTURE<R> async(Executor& ex, R(*f)()) { typedef R(*F)(); typedef detail::invoker<F> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( f ) )); } template <class Executor, class R, class A1> BOOST_THREAD_FUTURE<R> async(Executor& ex, R(*f)(BOOST_THREAD_FWD_REF(A1)), BOOST_THREAD_FWD_REF(A1) a1) { typedef R(*F)(BOOST_THREAD_FWD_REF(A1)); typedef detail::invoker<F, typename decay<A1>::type> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( f , thread_detail::decay_copy(boost::forward<A1>(a1)) ) )); } #endif // defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR template <class Executor, class F> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type()>::type> async(Executor& ex, BOOST_THREAD_FWD_REF(F) f) { typedef detail::invoker<typename decay<F>::type> BF; typedef typename BF::result_type Rp; return boost::detail::make_future_executor_shared_state<Rp>(ex, BF( thread_detail::decay_copy(boost::forward<F>(f)) ) ); } template <class Executor, class F, class A1> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type( typename decay<A1>::type )>::type> async(Executor& ex, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(A1) a1) { typedef detail::invoker<typename decay<F>::type, typename decay<A1>::type> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( thread_detail::decay_copy(boost::forward<F>(f)) , thread_detail::decay_copy(boost::forward<A1>(a1)) ) )); } template <class Executor, class F, class A1, class A2> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type( typename decay<A1>::type, typename decay<A2>::type )>::type> async(Executor& ex, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(A1) a1, BOOST_THREAD_FWD_REF(A2) a2) { typedef detail::invoker<typename decay<F>::type, typename decay<A1>::type, typename decay<A2>::type> BF; typedef typename BF::result_type Rp; return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_executor_shared_state<Rp>(ex, BF( thread_detail::decay_copy(boost::forward<F>(f)) , thread_detail::decay_copy(boost::forward<A1>(a1)) , thread_detail::decay_copy(boost::forward<A2>(a2)) ) )); } #endif //! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) #endif //////////////////////////////// // template <class F, class... ArgTypes> // future<R> async(F&&, ArgTypes&&...); //////////////////////////////// #if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNCTION_PTR #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class R, class... ArgTypes> BOOST_THREAD_FUTURE<R> async(R(*f)(BOOST_THREAD_FWD_REF(ArgTypes)...), BOOST_THREAD_FWD_REF(ArgTypes)... args) { return BOOST_THREAD_MAKE_RV_REF(async(launch(launch::any), f, boost::forward<ArgTypes>(args)...)); } #else template <class R> BOOST_THREAD_FUTURE<R> async(R(*f)()) { return BOOST_THREAD_MAKE_RV_REF(async(launch(launch::any), f)); } #endif #endif #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD) template <class F, class ...ArgTypes> BOOST_THREAD_FUTURE<typename boost::result_of<typename decay<F>::type( typename decay<ArgTypes>::type... )>::type> async(BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(ArgTypes)... args) { return BOOST_THREAD_MAKE_RV_REF(async(launch(launch::any), boost::forward<F>(f), boost::forward<ArgTypes>(args)...)); } #else template <class F> BOOST_THREAD_FUTURE<typename boost::result_of<F()>::type> async(BOOST_THREAD_FWD_REF(F) f) { return BOOST_THREAD_MAKE_RV_REF(async(launch(launch::any), boost::forward<F>(f))); } #endif //////////////////////////////// // make_future deprecated //////////////////////////////// template <typename T> BOOST_THREAD_FUTURE<typename decay<T>::type> make_future(BOOST_THREAD_FWD_REF(T) value) { typedef typename decay<T>::type future_value_type; promise<future_value_type> p; p.set_value(boost::forward<future_value_type>(value)); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } #if defined BOOST_THREAD_USES_MOVE inline BOOST_THREAD_FUTURE<void> make_future() { promise<void> p; p.set_value(); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } #endif //////////////////////////////// // make_ready_future //////////////////////////////// namespace detail { template <class T> struct deduced_type_impl { typedef T type; }; template <class T> struct deduced_type_impl<reference_wrapper<T> const> { typedef T& type; }; template <class T> struct deduced_type_impl<reference_wrapper<T> > { typedef T& type; }; #if __cplusplus > 201103L template <class T> struct deduced_type_impl<std::reference_wrapper<T> > { typedef T& type; }; #endif template <class T> struct deduced_type { typedef typename detail::deduced_type_impl<typename decay<T>::type>::type type; }; } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template <int = 0, int..., class T> #else template <class T> #endif BOOST_THREAD_FUTURE<typename detail::deduced_type<T>::type> make_ready_future(BOOST_THREAD_FWD_REF(T) value) { typedef typename detail::deduced_type<T>::type future_value_type; promise<future_value_type> p; p.set_value(boost::forward<T>(value)); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } // explicit overloads template <class T> BOOST_THREAD_FUTURE<T> make_ready_future(typename remove_reference<T>::type & x) { promise<T> p; p.set_value(x); return p.get_future(); } template <class T> BOOST_THREAD_FUTURE<T> make_ready_future(BOOST_THREAD_FWD_REF(typename remove_reference<T>::type) x) { promise<T> p; p.set_value(forward<typename remove_reference<T>::type>(x)); return p.get_future(); } // variadic overload #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template <class T, class ...Args> BOOST_THREAD_FUTURE<T> make_ready_future(Args&&... args) { promise<T> p; p.emplace(forward<Args>(args)...); return p.get_future(); } #endif template <typename T, typename T1> BOOST_THREAD_FUTURE<T> make_ready_no_decay_future(T1 value) { typedef T future_value_type; promise<future_value_type> p; p.set_value(value); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || defined BOOST_THREAD_USES_MOVE inline BOOST_THREAD_FUTURE<void> make_ready_future() { promise<void> p; p.set_value(); return p.get_future(); } #endif template <typename T> BOOST_THREAD_FUTURE<T> make_exceptional_future(exception_ptr ex) { promise<T> p; p.set_exception(ex); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } template <typename T, typename E> BOOST_THREAD_FUTURE<T> make_exceptional_future(E ex) { promise<T> p; p.set_exception(boost::copy_exception(ex)); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } template <typename T> BOOST_THREAD_FUTURE<T> make_exceptional_future() { promise<T> p; p.set_exception(boost::current_exception()); return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } template <typename T> BOOST_THREAD_FUTURE<T> make_ready_future(exception_ptr ex) { return make_exceptional_future<T>(ex); } #if 0 template<typename CLOSURE> make_future(CLOSURE closure) -> BOOST_THREAD_FUTURE<decltype(closure())> { typedef decltype(closure()) T; promise<T> p; try { p.set_value(closure()); } catch(...) { p.set_exception(std::current_exception()); } return BOOST_THREAD_MAKE_RV_REF(p.get_future()); } #endif //////////////////////////////// // make_shared_future deprecated //////////////////////////////// template <typename T> shared_future<typename decay<T>::type> make_shared_future(BOOST_THREAD_FWD_REF(T) value) { typedef typename decay<T>::type future_type; promise<future_type> p; p.set_value(boost::forward<T>(value)); return BOOST_THREAD_MAKE_RV_REF(p.get_future().share()); } inline shared_future<void> make_shared_future() { promise<void> p; return BOOST_THREAD_MAKE_RV_REF(p.get_future().share()); } //////////////////////////////// // detail::future_async_continuation_shared_state //////////////////////////////// #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION namespace detail { ////////////////////// // detail::continuation_shared_state ////////////////////// template<typename F, typename Rp, typename Fp, class ShSt=shared_state<Rp> > struct continuation_shared_state: ShSt { F parent; Fp continuation; public: continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : parent(boost::move(f)), continuation(boost::move(c)) { } void init(boost::unique_lock<boost::mutex> &lock) { parent.future_->set_continuation_ptr(this->shared_from_this(), lock); } void call() { try { this->mark_finished_with_result(this->continuation(boost::move(this->parent))); } catch(...) { this->mark_exceptional_finish(); } // make sure parent is really cleared to prevent memory "leaks" this->parent = F(); } void call(boost::unique_lock<boost::mutex>& lck) { try { relocker relock(lck); // neither continuation nor parent are protected by the lock - call() must only // be called once, and no one else must modify it. Rp res = this->continuation(boost::move(this->parent)); // make sure parent is really cleared to prevent memory "leaks" this->parent = F(); relock.lock(); this->mark_finished_with_result_internal(boost::move(res), lck); } catch (...) { this->mark_exceptional_finish_internal(current_exception(), lck); // make sure parent is really cleared to prevent memory "leaks" relocker relock(lck); this->parent = F(); } } static void run(shared_ptr<boost::detail::shared_state_base> that_) { continuation_shared_state* that = static_cast<continuation_shared_state*>(that_.get()); that->call(); } ~continuation_shared_state() {} }; template<typename F, typename Fp, class ShSt> struct continuation_shared_state<F, void, Fp, ShSt>: ShSt { F parent; Fp continuation; public: continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : parent(boost::move(f)), continuation(boost::move(c)) { } void init(boost::unique_lock<boost::mutex> &lock) { parent.future_->set_continuation_ptr(this->shared_from_this(), lock); } void call() { try { this->continuation(boost::move(this->parent)); this->mark_finished_with_result(); } catch(...) { this->mark_exceptional_finish(); } // make sure parent is really cleared to prevent memory "leaks" this->parent = F(); } void call(boost::unique_lock<boost::mutex>& lck) { try { { relocker relock(lck); // neither continuation nor parent are protected by the lock - call() must only // be called once, and no one else must modify it. this->continuation(boost::move(this->parent)); // make sure parent is really cleared to prevent memory "leaks" this->parent = F(); } this->mark_finished_with_result_internal(lck); } catch (...) { this->mark_exceptional_finish_internal(current_exception(), lck); // make sure parent is really cleared to prevent memory "leaks" relocker relock(lck); this->parent = F(); } } static void run(shared_ptr<boost::detail::shared_state_base> that_) { continuation_shared_state* that = static_cast<continuation_shared_state*>(that_.get()); that->call(); } ~continuation_shared_state() {} }; ///////////////////////// /// future_async_continuation_shared_state ///////////////////////// template<typename F, typename Rp, typename Fp> struct future_async_continuation_shared_state: continuation_shared_state<F,Rp,Fp,future_async_shared_state_base<Rp> > { typedef continuation_shared_state<F,Rp,Fp,future_async_shared_state_base<Rp> > base_type; public: future_async_continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } void launch_continuation() { #if defined BOOST_THREAD_FUTURE_BLOCKING boost::lock_guard<boost::mutex> lk(this->mutex); this->thr_ = boost::thread(&future_async_continuation_shared_state::run, static_shared_from_this(this)); #else boost::thread(&base_type::run, static_shared_from_this(this)).detach(); #endif } }; ///////////////////////// /// future_sync_continuation_shared_state ///////////////////////// template<typename F, typename Rp, typename Fp> struct future_sync_continuation_shared_state: continuation_shared_state<F,Rp,Fp,shared_state<Rp> > { typedef continuation_shared_state<F,Rp,Fp,shared_state<Rp> > base_type; public: future_sync_continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } void launch_continuation() { this->call(); } }; ///////////////////////// /// future_executor_continuation_shared_state ///////////////////////// #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template <typename FutureExecutorContinuationSharedState> struct run_it { shared_ptr<FutureExecutorContinuationSharedState> that_; #if ! defined(BOOST_NO_CXX11_RVALUE_REFERENCES) BOOST_THREAD_COPYABLE_AND_MOVABLE(run_it) run_it(run_it const& x) //BOOST_NOEXCEPT : that_(x.that_) {} run_it& operator=(BOOST_THREAD_COPY_ASSIGN_REF(run_it) x) //BOOST_NOEXCEPT { if (this != &x) { that_=x.that_; } return *this; } // move run_it(BOOST_THREAD_RV_REF(run_it) x) BOOST_NOEXCEPT : that_(x.that_) { x.that_.reset(); } run_it& operator=(BOOST_THREAD_RV_REF(run_it) x) BOOST_NOEXCEPT { if (this != &x) { that_=x.that; x.that_.reset(); } return *this; } #endif run_it(shared_ptr<FutureExecutorContinuationSharedState> that) : that_ (that) {} void operator()() { that_->run(that_); } }; } BOOST_THREAD_DCL_MOVABLE_BEG(F) detail::run_it<F> BOOST_THREAD_DCL_MOVABLE_END namespace detail { template<typename F, typename Rp, typename Fp> struct future_executor_continuation_shared_state: continuation_shared_state<F,Rp,Fp> { typedef continuation_shared_state<F,Rp,Fp> base_type; public: future_executor_continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } template <class Ex> void init(boost::unique_lock<boost::mutex> &lk, Ex& ex) { this->set_executor_policy(executor_ptr_type(new executor_ref<Ex>(ex)), lk); this->base_type::init(lk); } void launch_continuation() { run_it<base_type> fct(static_shared_from_this(this)); this->get_executor()->submit(boost::move(fct)); } ~future_executor_continuation_shared_state() {} }; #endif ///////////////////////// /// shared_future_async_continuation_shared_state ///////////////////////// template<typename F, typename Rp, typename Fp> struct shared_future_async_continuation_shared_state: continuation_shared_state<F,Rp,Fp,future_async_shared_state_base<Rp> > { typedef continuation_shared_state<F,Rp,Fp,future_async_shared_state_base<Rp> > base_type; public: shared_future_async_continuation_shared_state(F f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } void launch_continuation() { #if defined BOOST_THREAD_FUTURE_BLOCKING boost::lock_guard<boost::mutex> lk(this->mutex); this->thr_ = boost::thread(&base_type::run, static_shared_from_this(this)); #else boost::thread(&base_type::run, static_shared_from_this(this)).detach(); #endif } }; ///////////////////////// /// shared_future_async_continuation_shared_state ///////////////////////// template<typename F, typename Rp, typename Fp> struct shared_future_sync_continuation_shared_state: continuation_shared_state<F,Rp,Fp,shared_state<Rp> > { typedef continuation_shared_state<F,Rp,Fp,shared_state<Rp> > base_type; public: shared_future_sync_continuation_shared_state(F f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } void launch_continuation() { this->call(); } }; ///////////////////////// /// shared_future_executor_continuation_shared_state ///////////////////////// #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename F, typename Rp, typename Fp> struct shared_future_executor_continuation_shared_state: continuation_shared_state<F,Rp,Fp> { typedef continuation_shared_state<F,Rp,Fp> base_type; public: shared_future_executor_continuation_shared_state(F f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { } template <class Ex> void init(boost::unique_lock<boost::mutex> &lk, Ex& ex) { this->set_executor_policy(executor_ptr_type(new executor_ref<Ex>(ex)), lk); this->base_type::init(lk); } void launch_continuation() { run_it<base_type> fct(static_shared_from_this(this)); this->get_executor()->submit(boost::move(fct)); } ~shared_future_executor_continuation_shared_state() {} }; #endif ////////////////////////// /// future_deferred_continuation_shared_state ////////////////////////// template<typename F, typename Rp, typename Fp> struct future_deferred_continuation_shared_state: continuation_shared_state<F,Rp,Fp> { typedef continuation_shared_state<F,Rp,Fp> base_type; public: future_deferred_continuation_shared_state(BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { this->set_deferred(); } virtual void execute(boost::unique_lock<boost::mutex>& lk) { this->parent.wait(); this->call(lk); } virtual void launch_continuation() { } }; ////////////////////////// /// shared_future_deferred_continuation_shared_state ////////////////////////// template<typename F, typename Rp, typename Fp> struct shared_future_deferred_continuation_shared_state: continuation_shared_state<F,Rp,Fp> { typedef continuation_shared_state<F,Rp,Fp> base_type; public: shared_future_deferred_continuation_shared_state(F f, BOOST_THREAD_FWD_REF(Fp) c) : base_type(boost::move(f), boost::forward<Fp>(c)) { this->set_deferred(); } virtual void execute(boost::unique_lock<boost::mutex>& lk) { this->parent.wait(); this->call(lk); } virtual void launch_continuation() { } }; //////////////////////////////// // make_future_deferred_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_future_deferred_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<future_deferred_continuation_shared_state<F, Rp, Cont> > h(new future_deferred_continuation_shared_state<F, Rp, Cont>(boost::move(f), boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_future_async_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_future_async_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<future_async_continuation_shared_state<F,Rp, Cont> > h(new future_async_continuation_shared_state<F,Rp, Cont>(boost::move(f), boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_future_sync_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_future_sync_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<future_sync_continuation_shared_state<F,Rp, Cont> > h(new future_sync_continuation_shared_state<F,Rp, Cont>(boost::move(f), boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_future_executor_continuation_shared_state //////////////////////////////// #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<future_executor_continuation_shared_state<F,Rp, Cont> > h(new future_executor_continuation_shared_state<F,Rp, Cont>(boost::move(f), boost::forward<Fp>(c))); h->init(lock, ex); return BOOST_THREAD_FUTURE<Rp>(h); } #endif //////////////////////////////// // make_shared_future_deferred_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_deferred_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<shared_future_deferred_continuation_shared_state<F, Rp, Cont> > h(new shared_future_deferred_continuation_shared_state<F, Rp, Cont>(f, boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_shared_future_async_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_async_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<shared_future_async_continuation_shared_state<F,Rp, Cont> > h(new shared_future_async_continuation_shared_state<F,Rp, Cont>(f, boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_shared_future_sync_continuation_shared_state //////////////////////////////// template<typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_sync_continuation_shared_state( boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<shared_future_sync_continuation_shared_state<F,Rp, Cont> > h(new shared_future_sync_continuation_shared_state<F,Rp, Cont>(f, boost::forward<Fp>(c))); h->init(lock); return BOOST_THREAD_FUTURE<Rp>(h); } //////////////////////////////// // make_shared_future_executor_continuation_shared_state //////////////////////////////// #ifdef BOOST_THREAD_PROVIDES_EXECUTORS template<typename Ex, typename F, typename Rp, typename Fp> BOOST_THREAD_FUTURE<Rp> make_shared_future_executor_continuation_shared_state(Ex& ex, boost::unique_lock<boost::mutex> &lock, F f, BOOST_THREAD_FWD_REF(Fp) c) { typedef typename decay<Fp>::type Cont; shared_ptr<shared_future_executor_continuation_shared_state<F, Rp, Cont> > h(new shared_future_executor_continuation_shared_state<F, Rp, Cont>(f, boost::forward<Fp>(c))); h->init(lock, ex); return BOOST_THREAD_FUTURE<Rp>(h); } #endif } //////////////////////////////// // template<typename F> // auto future<R>::then(launch policy, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type> BOOST_THREAD_FUTURE<R>::then(launch policy, BOOST_THREAD_FWD_REF(F) func) { typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); if (underlying_cast<int>(policy) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_sync_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy) & int(launch::executor)) { assert(this->future_->get_executor()); typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); #endif } else if (underlying_cast<int>(policy) & int(launch::inherit)) { launch policy_ = this->launch_policy(lock); if (underlying_cast<int>(policy_) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_sync_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy_) & int(launch::executor)) { assert(this->future_->get_executor()); typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); #endif } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS //////////////////////////////// // template<typename Ex, typename F> // auto future<future<R2> >::then(Ex&, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type> BOOST_THREAD_FUTURE<R>::then(Ex& ex, BOOST_THREAD_FWD_REF(F) func) { typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); } #endif //////////////////////////////// // template<typename F> // auto future<future<R2> >::then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type> BOOST_THREAD_FUTURE<R>::then(BOOST_THREAD_FWD_REF(F) func) { #ifndef BOOST_THREAD_CONTINUATION_SYNC return this->then(this->launch_policy(), boost::forward<F>(func)); #else typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); launch policy = this->launch_policy(lock); if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } #endif } //////////////////////////////// // template<typename F> // auto future<future<R2> >::then(launch, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R2> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >)>::type> BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >::then(launch policy, BOOST_THREAD_FWD_REF(F) func) { typedef BOOST_THREAD_FUTURE<R2> R; typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); if (underlying_cast<int>(policy) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_sync_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy) & int(launch::executor)) { assert(this->future_->get_executor()); typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); #endif } else if (underlying_cast<int>(policy) & int(launch::inherit)) { launch policy_ = this->launch_policy(lock); if (underlying_cast<int>(policy_) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_sync_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy_) & int(launch::executor)) { assert(this->future_->get_executor()); typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); #endif } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS //////////////////////////////// // template<typename Ex, typename F> // auto future<future<R2> >::then(Ex&, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R2> template <typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >)>::type> BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >::then(Ex& ex, BOOST_THREAD_FWD_REF(F) func) { typedef BOOST_THREAD_FUTURE<R2> R; typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_executor_continuation_shared_state<Ex, BOOST_THREAD_FUTURE<R>, future_type>(ex, lock, boost::move(*this), boost::forward<F>(func) ))); } #endif //////////////////////////////// // template<typename F> // auto future<future<R2> >::then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R2> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >)>::type> BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >::then(BOOST_THREAD_FWD_REF(F) func) { #ifndef BOOST_THREAD_CONTINUATION_SYNC return this->then(this->launch_policy(), boost::forward<F>(func)); #else typedef BOOST_THREAD_FUTURE<R2> R; typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); launch policy = this->launch_policy(lock); if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_sync_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type>( lock, boost::move(*this), boost::forward<F>(func) ))); } #endif } //////////////////////////////// // template<typename F> // auto shared_future<R>::then(launch policy, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future<R>)>::type> shared_future<R>::then(launch policy, BOOST_THREAD_FWD_REF(F) func) const { typedef typename boost::result_of<F(shared_future<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); boost::unique_lock<boost::mutex> lock(this->future_->mutex); if (underlying_cast<int>(policy) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_async_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_deferred_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_sync_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy) & int(launch::executor)) { typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_executor_continuation_shared_state<Ex, shared_future<R>, future_type>(ex, lock, *this, boost::forward<F>(func) ))); #endif } else if (underlying_cast<int>(policy) & int(launch::inherit)) { launch policy_ = this->launch_policy(lock); if (underlying_cast<int>(policy_) & int(launch::async)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_async_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_deferred_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } else if (underlying_cast<int>(policy_) & int(launch::sync)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_sync_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); #ifdef BOOST_THREAD_PROVIDES_EXECUTORS } else if (underlying_cast<int>(policy_) & int(launch::executor)) { typedef executor Ex; Ex& ex = *(this->future_->get_executor()); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_executor_continuation_shared_state<Ex, shared_future<R>, future_type>(ex, lock, *this, boost::forward<F>(func) ))); #endif } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_async_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_async_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } } #ifdef BOOST_THREAD_PROVIDES_EXECUTORS //////////////////////////////// // template<typename Ex, typename F> // auto shared_future<R>::then(Ex&, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename Ex, typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future<R>)>::type> shared_future<R>::then(Ex& ex, BOOST_THREAD_FWD_REF(F) func) const { typedef typename boost::result_of<F(shared_future<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); boost::unique_lock<boost::mutex> lock(this->future_->mutex); return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_executor_continuation_shared_state<Ex, shared_future<R>, future_type>(ex, lock, *this, boost::forward<F>(func) ))); } #endif //////////////////////////////// // template<typename F> // auto shared_future<R>::then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>; //////////////////////////////// template <typename R> template <typename F> inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future<R>)>::type> shared_future<R>::then(BOOST_THREAD_FWD_REF(F) func) const { #ifndef BOOST_THREAD_CONTINUATION_SYNC return this->then(this->launch_policy(), boost::forward<F>(func)); #else typedef typename boost::result_of<F(shared_future<R>)>::type future_type; BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); boost::unique_lock<boost::mutex> lock(this->future_->mutex); launch policy = this->launch_policy(lock); if (underlying_cast<int>(policy) & int(launch::deferred)) { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_deferred_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } else { return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_shared_future_sync_continuation_shared_state<shared_future<R>, future_type>( lock, *this, boost::forward<F>(func) ))); } #endif } namespace detail { template <typename T> struct mfallbacker_to { T value_; typedef T result_type; mfallbacker_to(BOOST_THREAD_RV_REF(T) v) : value_(boost::move(v)) {} T operator()(BOOST_THREAD_FUTURE<T> fut) { return fut.get_or(boost::move(value_)); } }; template <typename T> struct cfallbacker_to { T value_; typedef T result_type; cfallbacker_to(T const& v) : value_(v) {} T operator()(BOOST_THREAD_FUTURE<T> fut) const { return fut.get_or(value_); } }; } //////////////////////////////// // future<R> future<R>::fallback_to(R&& v); //////////////////////////////// template <typename R> template <typename R2> inline typename boost::disable_if< is_void<R2>, BOOST_THREAD_FUTURE<R> >::type BOOST_THREAD_FUTURE<R>::fallback_to(BOOST_THREAD_RV_REF(R2) v) { return then(detail::mfallbacker_to<R>(boost::move(v))); } template <typename R> template <typename R2> inline typename boost::disable_if< is_void<R2>, BOOST_THREAD_FUTURE<R> >::type BOOST_THREAD_FUTURE<R>::fallback_to(R2 const& v) { return then(detail::cfallbacker_to<R>(v)); } #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP namespace detail { ///////////////////////// /// future_unwrap_shared_state ///////////////////////// template<typename F, typename Rp> struct future_unwrap_shared_state: shared_state<Rp> { F wrapped; typename F::value_type unwrapped; public: explicit future_unwrap_shared_state(BOOST_THREAD_RV_REF(F) f) : wrapped(boost::move(f)) { } void launch_continuation() { boost::unique_lock<boost::mutex> lk(this->mutex); // assert(wrapped.is_ready()); if (! unwrapped.valid() ) { if (wrapped.has_exception()) { this->mark_exceptional_finish_internal(wrapped.get_exception_ptr(), lk); } else { unwrapped = wrapped.get(); if (unwrapped.valid()) { lk.unlock(); boost::unique_lock<boost::mutex> lk2(unwrapped.future_->mutex); unwrapped.future_->set_continuation_ptr(this->shared_from_this(), lk2); } else { this->mark_exceptional_finish_internal(boost::copy_exception(future_uninitialized()), lk); } } } else { // assert(unwrapped.is_ready()); if (unwrapped.has_exception()) { this->mark_exceptional_finish_internal(unwrapped.get_exception_ptr(), lk); } else { this->mark_finished_with_result_internal(unwrapped.get(), lk); } } } }; template<typename F> struct future_unwrap_shared_state<F,void>: shared_state<void> { F wrapped; typename F::value_type unwrapped; public: explicit future_unwrap_shared_state(BOOST_THREAD_RV_REF(F) f) : wrapped(boost::move(f)) { } void launch_continuation() { boost::unique_lock<boost::mutex> lk(this->mutex); // assert(wrapped.is_ready()); if (! unwrapped.valid() ) { if (wrapped.has_exception()) { this->mark_exceptional_finish_internal(wrapped.get_exception_ptr(), lk); } else { unwrapped = wrapped.get(); if (unwrapped.valid()) { lk.unlock(); boost::unique_lock<boost::mutex> lk2(unwrapped.future_->mutex); unwrapped.future_->set_continuation_ptr(this->shared_from_this(), lk2); } else { this->mark_exceptional_finish_internal(boost::copy_exception(future_uninitialized()), lk); } } } else { // assert(unwrapped.is_ready()); if (unwrapped.has_exception()) { this->mark_exceptional_finish_internal(unwrapped.get_exception_ptr(), lk); } else { this->mark_finished_with_result_internal(lk); } } } }; template <class F, class Rp> BOOST_THREAD_FUTURE<Rp> make_future_unwrap_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f) { shared_ptr<future_unwrap_shared_state<F, Rp> > h(new future_unwrap_shared_state<F, Rp>(boost::move(f))); h->wrapped.future_->set_continuation_ptr(h, lock); return BOOST_THREAD_FUTURE<Rp>(h); } } template <typename R> inline BOOST_THREAD_FUTURE<R>::BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R> >) other) : base_type(other.unwrap()) {} template <typename R2> BOOST_THREAD_FUTURE<R2> BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >::unwrap() { BOOST_THREAD_ASSERT_PRECONDITION(this->future_.get()!=0, future_uninitialized()); // keep state alive as we move ourself but hold the lock shared_ptr<detail::shared_state_base> sentinel(this->future_); boost::unique_lock<boost::mutex> lock(sentinel->mutex); return boost::detail::make_future_unwrap_shared_state<BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >, R2>(lock, boost::move(*this)); } #endif #if defined BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY namespace detail { struct input_iterator_tag {}; struct vector_tag {}; struct values_tag {}; template <typename T> struct alias_t { typedef T type; }; BOOST_CONSTEXPR_OR_CONST input_iterator_tag input_iterator_tag_value = {}; BOOST_CONSTEXPR_OR_CONST vector_tag vector_tag_value = {}; BOOST_CONSTEXPR_OR_CONST values_tag values_tag_value = {}; //////////////////////////////// // detail::future_async_when_all_shared_state //////////////////////////////// template<typename F> struct future_when_all_vector_shared_state: future_async_shared_state_base<csbl::vector<F> > { typedef csbl::vector<F> vector_type; typedef typename F::value_type value_type; vector_type vec_; static void run(shared_ptr<boost::detail::shared_state_base> that_) { future_when_all_vector_shared_state* that = static_cast<future_when_all_vector_shared_state*>(that_.get()); try { boost::wait_for_all(that->vec_.begin(), that->vec_.end()); that->mark_finished_with_result(boost::move(that->vec_)); } catch(...) { that->mark_exceptional_finish(); } } bool run_deferred() { bool res = false; for (typename csbl::vector<F>::iterator it = vec_.begin(); it != vec_.end(); ++it) { if (! it->run_if_is_deferred()) { res = true; } } return res; } void init() { if (! run_deferred()) { future_when_all_vector_shared_state::run(this->shared_from_this()); return; } #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_when_all_vector_shared_state::run, this->shared_from_this()); #else boost::thread(&future_when_all_vector_shared_state::run, this->shared_from_this()).detach(); #endif } public: template< typename InputIterator> future_when_all_vector_shared_state(input_iterator_tag, InputIterator first, InputIterator last) : vec_(std::make_move_iterator(first), std::make_move_iterator(last)) { } future_when_all_vector_shared_state(vector_tag, BOOST_THREAD_RV_REF(csbl::vector<F>) v) : vec_(boost::move(v)) { } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> future_when_all_vector_shared_state(values_tag, BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures) { vec_.push_back(boost::forward<T0>(f)); typename alias_t<char[]>::type{ ( //first part of magic unpacker vec_.push_back(boost::forward<T>(futures)),'0' )..., '0' }; //second part of magic unpacker } #endif ~future_when_all_vector_shared_state() {} }; //////////////////////////////// // detail::future_async_when_any_shared_state //////////////////////////////// template<typename F> struct future_when_any_vector_shared_state: future_async_shared_state_base<csbl::vector<F> > { typedef csbl::vector<F> vector_type; typedef typename F::value_type value_type; vector_type vec_; static void run(shared_ptr<boost::detail::shared_state_base> that_) { future_when_any_vector_shared_state* that = static_cast<future_when_any_vector_shared_state*>(that_.get()); try { boost::wait_for_any(that->vec_.begin(), that->vec_.end()); that->mark_finished_with_result(boost::move(that->vec_)); } catch(...) { that->mark_exceptional_finish(); } } bool run_deferred() { for (typename csbl::vector<F>::iterator it = vec_.begin(); it != vec_.end(); ++it) { if (it->run_if_is_deferred_or_ready()) { return true; } } return false; } void init() { if (run_deferred()) { future_when_any_vector_shared_state::run(this->shared_from_this()); return; } #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_when_any_vector_shared_state::run, this->shared_from_this()); #else boost::thread(&future_when_any_vector_shared_state::run, this->shared_from_this()).detach(); #endif } public: template< typename InputIterator> future_when_any_vector_shared_state(input_iterator_tag, InputIterator first, InputIterator last) : vec_(std::make_move_iterator(first), std::make_move_iterator(last)) { } future_when_any_vector_shared_state(vector_tag, BOOST_THREAD_RV_REF(csbl::vector<F>) v) : vec_(boost::move(v)) { } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> future_when_any_vector_shared_state(values_tag, BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures ) { vec_.push_back(boost::forward<T0>(f)); typename alias_t<char[]>::type{ ( //first part of magic unpacker vec_.push_back(boost::forward<T>(futures)) ,'0' )..., '0' }; //second part of magic unpacker } #endif ~future_when_any_vector_shared_state() {} }; #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) struct wait_for_all_fctr { template <class ...T> void operator()(T&&... v) { boost::wait_for_all(boost::forward<T>(v)...); } }; struct wait_for_any_fctr { template <class ...T> void operator()(T&&... v) { boost::wait_for_any(boost::forward<T>(v)...); } }; template <class Tuple, std::size_t i=csbl::tuple_size<Tuple>::value> struct accumulate_run_if_is_deferred { bool operator ()(Tuple& t) { return (! csbl::get<i-1>(t).run_if_is_deferred()) || accumulate_run_if_is_deferred<Tuple,i-1>()(t); } }; template <class Tuple> struct accumulate_run_if_is_deferred<Tuple, 0> { bool operator ()(Tuple& ) { return false; } }; template< typename Tuple, typename T0, typename ...T> struct future_when_all_tuple_shared_state: future_async_shared_state_base<Tuple> { Tuple tup_; typedef typename make_tuple_indices<1+sizeof...(T)>::type Index; static void run(shared_ptr<boost::detail::shared_state_base> that_) { future_when_all_tuple_shared_state* that = static_cast<future_when_all_tuple_shared_state*>(that_.get()); try { // TODO make use of apply(that->tup_, boost::detail::wait_for_all_fctor()); that->wait_for_all(Index()); that->mark_finished_with_result(boost::move(that->tup_)); } catch(...) { that->mark_exceptional_finish(); } } template <size_t ...Indices> void wait_for_all(tuple_indices<Indices...>) { #if defined BOOST_THREAD_PROVIDES_INVOKE return invoke<void>(wait_for_all_fctr(), csbl::get<Indices>(tup_)...); #else return wait_for_all_fctr()(csbl::get<Indices>(tup_)...); #endif } bool run_deferred() { return accumulate_run_if_is_deferred<Tuple>()(tup_); } void init() { if (! run_deferred()) { future_when_all_tuple_shared_state::run(this->shared_from_this()); return; } #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_when_all_tuple_shared_state::run, this->shared_from_this()); #else boost::thread(&future_when_all_tuple_shared_state::run, this->shared_from_this()).detach(); #endif } public: template< typename F, typename ...Fs> future_when_all_tuple_shared_state(values_tag, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(Fs) ... futures) : tup_(boost::csbl::make_tuple(boost::forward<F>(f), boost::forward<Fs>(futures)...)) { } ~future_when_all_tuple_shared_state() {} }; template <class Tuple, std::size_t i=csbl::tuple_size<Tuple>::value> struct apply_any_run_if_is_deferred_or_ready { bool operator ()(Tuple& t) { if (csbl::get<i-1>(t).run_if_is_deferred_or_ready()) return true; return apply_any_run_if_is_deferred_or_ready<Tuple,i-1>()(t); } }; template <class Tuple> struct apply_any_run_if_is_deferred_or_ready<Tuple, 0> { bool operator ()(Tuple& ) { return false; } }; template< typename Tuple, typename T0, typename ...T > struct future_when_any_tuple_shared_state: future_async_shared_state_base<Tuple> { Tuple tup_; typedef typename make_tuple_indices<1+sizeof...(T)>::type Index; static void run(shared_ptr<boost::detail::shared_state_base> that_) { future_when_any_tuple_shared_state* that = static_cast<future_when_any_tuple_shared_state*>(that_.get()); try { // TODO make use of apply(that->tup_, wait_for_any_fctr); that->wait_for_any(Index()); that->mark_finished_with_result(boost::move(that->tup_)); } catch(...) { that->mark_exceptional_finish(); } } template <size_t ...Indices> void wait_for_any(tuple_indices<Indices...>) { #if defined BOOST_THREAD_PROVIDES_INVOKE return invoke<void>(wait_for_any_fctr(), csbl::get<Indices>(tup_)...); #else return wait_for_any_fctr()(csbl::get<Indices>(tup_)...); #endif } bool run_deferred() { return apply_any_run_if_is_deferred_or_ready<Tuple>()(tup_); } void init() { if (run_deferred()) { future_when_any_tuple_shared_state::run(this->shared_from_this()); return; } #ifdef BOOST_THREAD_FUTURE_BLOCKING this->thr_ = boost::thread(&future_when_any_tuple_shared_state::run, this->shared_from_this()); #else boost::thread(&future_when_any_tuple_shared_state::run, this->shared_from_this()).detach(); #endif } public: template< typename F, typename ...Fs> future_when_any_tuple_shared_state(values_tag, BOOST_THREAD_FWD_REF(F) f, BOOST_THREAD_FWD_REF(Fs) ... futures ) : tup_(boost::csbl::make_tuple(boost::forward<F>(f), boost::forward<Fs>(futures)...)) { } ~future_when_any_tuple_shared_state() {} }; #endif } template< typename InputIterator> typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_all(InputIterator first, InputIterator last) { typedef typename InputIterator::value_type value_type; typedef csbl::vector<value_type> container_type; typedef detail::future_when_all_vector_shared_state<value_type> factory_type; if (first==last) return make_ready_future(container_type()); shared_ptr<factory_type > h(new factory_type(detail::input_iterator_tag_value, first,last)); h->init(); return BOOST_THREAD_FUTURE<container_type>(h); } inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_all() { return make_ready_future(csbl::tuple<>()); } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_all(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures) { typedef csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> container_type; typedef detail::future_when_all_tuple_shared_state<container_type, typename decay<T0>::type, typename decay<T>::type...> factory_type; shared_ptr<factory_type> h(new factory_type(detail::values_tag_value, boost::forward<T0>(f), boost::forward<T>(futures)...)); h->init(); return BOOST_THREAD_FUTURE<container_type>(h); } #endif template< typename InputIterator> typename boost::disable_if<is_future_type<InputIterator>, BOOST_THREAD_FUTURE<csbl::vector<typename InputIterator::value_type> > >::type when_any(InputIterator first, InputIterator last) { typedef typename InputIterator::value_type value_type; typedef csbl::vector<value_type> container_type; typedef detail::future_when_any_vector_shared_state<value_type> factory_type; if (first==last) return make_ready_future(container_type()); shared_ptr<factory_type > h(new factory_type(detail::input_iterator_tag_value, first,last)); h->init(); return BOOST_THREAD_FUTURE<container_type>(h); } inline BOOST_THREAD_FUTURE<csbl::tuple<> > when_any() { return make_ready_future(csbl::tuple<>()); } #if ! defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) template< typename T0, typename ...T> BOOST_THREAD_FUTURE<csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> > when_any(BOOST_THREAD_FWD_REF(T0) f, BOOST_THREAD_FWD_REF(T) ... futures) { typedef csbl::tuple<typename decay<T0>::type, typename decay<T>::type...> container_type; typedef detail::future_when_any_tuple_shared_state<container_type, typename decay<T0>::type, typename decay<T>::type...> factory_type; shared_ptr<factory_type> h(new factory_type(detail::values_tag_value, boost::forward<T0>(f), boost::forward<T>(futures)...)); h->init(); return BOOST_THREAD_FUTURE<container_type>(h); } #endif #endif // BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY } #endif // BOOST_NO_EXCEPTIONS #endif // header