5899 lines
		
	
	
		
			199 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			5899 lines
		
	
	
		
			199 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //  (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/next_prior.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
 |