2012-05-09 21:45:30 -07:00
|
|
|
// (C) Copyright 2008-10 Anthony Williams
|
2013-08-25 14:35:47 -07:00
|
|
|
// (C) Copyright 2011-2013 Vicente J. Botet Escriba
|
2012-05-09 21:45:30 -07:00
|
|
|
//
|
|
|
|
// 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
|
2012-07-21 11:37:40 -07:00
|
|
|
|
|
|
|
#include <boost/thread/detail/config.hpp>
|
2013-01-13 14:38:19 -08:00
|
|
|
|
|
|
|
// boost::thread::future requires exception handling
|
|
|
|
// due to boost::exception::exception_ptr dependency
|
|
|
|
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
//#include <boost/thread/detail/log.hpp>
|
2012-07-21 11:37:40 -07:00
|
|
|
#include <boost/detail/scoped_enum_emulation.hpp>
|
2012-05-09 21:45:30 -07:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <boost/thread/detail/move.hpp>
|
2013-02-13 20:03:52 -08:00
|
|
|
#include <boost/thread/detail/async_func.hpp>
|
2012-05-09 21:45:30 -07:00
|
|
|
#include <boost/thread/thread_time.hpp>
|
|
|
|
#include <boost/thread/mutex.hpp>
|
|
|
|
#include <boost/thread/condition_variable.hpp>
|
2013-02-13 20:03:52 -08:00
|
|
|
#include <boost/thread/lock_algorithms.hpp>
|
|
|
|
#include <boost/thread/lock_types.hpp>
|
2012-05-09 21:45:30 -07:00
|
|
|
#include <boost/exception_ptr.hpp>
|
|
|
|
#include <boost/shared_ptr.hpp>
|
|
|
|
#include <boost/scoped_ptr.hpp>
|
|
|
|
#include <boost/type_traits/is_fundamental.hpp>
|
2013-02-13 20:03:52 -08:00
|
|
|
#include <boost/thread/detail/is_convertible.hpp>
|
2012-07-21 11:37:40 -07:00
|
|
|
#include <boost/type_traits/remove_reference.hpp>
|
|
|
|
#include <boost/type_traits/remove_cv.hpp>
|
2014-03-01 11:00:20 -08:00
|
|
|
#include <boost/type_traits/is_void.hpp>
|
2012-05-09 21:45:30 -07:00
|
|
|
#include <boost/mpl/if.hpp>
|
|
|
|
#include <boost/config.hpp>
|
|
|
|
#include <boost/throw_exception.hpp>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <boost/function.hpp>
|
|
|
|
#include <boost/bind.hpp>
|
|
|
|
#include <boost/ref.hpp>
|
|
|
|
#include <boost/scoped_array.hpp>
|
2013-02-13 20:03:52 -08:00
|
|
|
#include <boost/enable_shared_from_this.hpp>
|
2012-05-09 21:45:30 -07:00
|
|
|
#include <boost/utility/enable_if.hpp>
|
|
|
|
#include <list>
|
|
|
|
#include <boost/next_prior.hpp>
|
|
|
|
#include <vector>
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
#include <boost/thread/future_error_code.hpp>
|
2012-07-21 11:37:40 -07:00
|
|
|
#ifdef BOOST_THREAD_USES_CHRONO
|
|
|
|
#include <boost/chrono/system_clocks.hpp>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
#include <boost/thread/detail/memory.hpp>
|
|
|
|
#endif
|
|
|
|
|
2013-01-13 14:38:19 -08:00
|
|
|
#include <boost/utility/result_of.hpp>
|
2013-08-25 14:35:47 -07:00
|
|
|
#include <boost/thread/thread_only.hpp>
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE
|
|
|
|
#define BOOST_THREAD_FUTURE future
|
|
|
|
#else
|
|
|
|
#define BOOST_THREAD_FUTURE unique_future
|
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
namespace boost
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
//enum class launch
|
|
|
|
BOOST_SCOPED_ENUM_DECLARE_BEGIN(launch)
|
|
|
|
{
|
|
|
|
none = 0,
|
|
|
|
async = 1,
|
|
|
|
deferred = 2,
|
|
|
|
any = async | deferred
|
|
|
|
}
|
|
|
|
BOOST_SCOPED_ENUM_DECLARE_END(launch)
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
//enum class future_status
|
|
|
|
BOOST_SCOPED_ENUM_DECLARE_BEGIN(future_status)
|
|
|
|
{
|
|
|
|
ready,
|
|
|
|
timeout,
|
|
|
|
deferred
|
|
|
|
}
|
|
|
|
BOOST_SCOPED_ENUM_DECLARE_END(future_status)
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
class BOOST_SYMBOL_VISIBLE future_error
|
|
|
|
: public std::logic_error
|
|
|
|
{
|
|
|
|
system::error_code ec_;
|
|
|
|
public:
|
|
|
|
future_error(system::error_code ec)
|
|
|
|
: logic_error(ec.message()),
|
|
|
|
ec_(ec)
|
|
|
|
{
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
const system::error_code& code() const BOOST_NOEXCEPT
|
|
|
|
{
|
|
|
|
return ec_;
|
|
|
|
}
|
|
|
|
};
|
2012-07-21 11:37:40 -07:00
|
|
|
|
|
|
|
class BOOST_SYMBOL_VISIBLE future_uninitialized:
|
|
|
|
public future_error
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
public:
|
2013-01-13 14:38:19 -08:00
|
|
|
future_uninitialized() :
|
2012-07-21 11:37:40 -07:00
|
|
|
future_error(system::make_error_code(future_errc::no_state))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
2012-07-21 11:37:40 -07:00
|
|
|
class BOOST_SYMBOL_VISIBLE broken_promise:
|
|
|
|
public future_error
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
broken_promise():
|
2012-07-21 11:37:40 -07:00
|
|
|
future_error(system::make_error_code(future_errc::broken_promise))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
2012-07-21 11:37:40 -07:00
|
|
|
class BOOST_SYMBOL_VISIBLE future_already_retrieved:
|
|
|
|
public future_error
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
future_already_retrieved():
|
2012-07-21 11:37:40 -07:00
|
|
|
future_error(system::make_error_code(future_errc::future_already_retrieved))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
2012-07-21 11:37:40 -07:00
|
|
|
class BOOST_SYMBOL_VISIBLE promise_already_satisfied:
|
|
|
|
public future_error
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
promise_already_satisfied():
|
2012-07-21 11:37:40 -07:00
|
|
|
future_error(system::make_error_code(future_errc::promise_already_satisfied))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
class BOOST_SYMBOL_VISIBLE task_already_started:
|
|
|
|
public future_error
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
task_already_started():
|
2012-07-21 11:37:40 -07:00
|
|
|
future_error(system::make_error_code(future_errc::promise_already_satisfied))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
class BOOST_SYMBOL_VISIBLE task_moved:
|
|
|
|
public future_error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
task_moved():
|
|
|
|
future_error(system::make_error_code(future_errc::no_state))
|
|
|
|
{}
|
|
|
|
};
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
class promise_moved:
|
|
|
|
public future_error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
promise_moved():
|
|
|
|
future_error(system::make_error_code(future_errc::no_state))
|
|
|
|
{}
|
|
|
|
};
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
namespace future_state
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
enum state { uninitialized, waiting, ready, moved, deferred };
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace detail
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
struct relocker
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex>& lock_;
|
|
|
|
bool unlocked_;
|
|
|
|
|
|
|
|
relocker(boost::unique_lock<boost::mutex>& lk):
|
|
|
|
lock_(lk)
|
|
|
|
{
|
|
|
|
lock_.unlock();
|
|
|
|
unlocked_=true;
|
|
|
|
}
|
|
|
|
~relocker()
|
|
|
|
{
|
|
|
|
if (unlocked_) {
|
|
|
|
lock_.lock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void lock() {
|
|
|
|
if (unlocked_) {
|
|
|
|
lock_.lock();
|
|
|
|
unlocked_=false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
relocker& operator=(relocker const&);
|
|
|
|
};
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
struct shared_state_base : enable_shared_from_this<shared_state_base>
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef std::list<boost::condition_variable_any*> waiter_list;
|
|
|
|
// 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;
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
boost::exception_ptr exception;
|
|
|
|
bool done;
|
2013-02-13 20:03:52 -08:00
|
|
|
bool is_deferred_;
|
|
|
|
launch policy_;
|
|
|
|
bool is_constructed;
|
2014-03-01 11:00:20 -08:00
|
|
|
mutable boost::mutex mutex;
|
2012-05-09 21:45:30 -07:00
|
|
|
boost::condition_variable waiters;
|
|
|
|
waiter_list external_waiters;
|
|
|
|
boost::function<void()> callback;
|
2014-03-01 11:00:20 -08:00
|
|
|
// This declaration should be only included conditionally if interruptions are allowed, but is included to maintain the same layout.
|
2013-02-13 20:03:52 -08:00
|
|
|
bool thread_was_interrupted;
|
2014-03-01 11:00:20 -08:00
|
|
|
// This declaration should be only included conditionally, but is included to maintain the same layout.
|
2013-08-25 14:35:47 -07:00
|
|
|
continuation_ptr_type continuation_ptr;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
// This declaration should be only included conditionally, but is included to maintain the same layout.
|
2013-08-25 14:35:47 -07:00
|
|
|
virtual void launch_continuation(boost::unique_lock<boost::mutex>&)
|
|
|
|
{
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
shared_state_base():
|
2012-07-21 11:37:40 -07:00
|
|
|
done(false),
|
2013-02-13 20:03:52 -08:00
|
|
|
is_deferred_(false),
|
|
|
|
policy_(launch::none),
|
2014-03-01 11:00:20 -08:00
|
|
|
is_constructed(false),
|
|
|
|
thread_was_interrupted(false),
|
|
|
|
continuation_ptr()
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual ~shared_state_base()
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_deferred()
|
|
|
|
{
|
|
|
|
is_deferred_ = true;
|
2014-03-01 11:00:20 -08:00
|
|
|
policy_ = launch::deferred;
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
void set_async()
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
is_deferred_ = false;
|
2014-03-01 11:00:20 -08:00
|
|
|
policy_ = launch::async;
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
waiter_list::iterator register_external_waiter(boost::condition_variable_any& cv)
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
do_callback(lock);
|
|
|
|
return external_waiters.insert(external_waiters.end(),&cv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void remove_external_waiter(waiter_list::iterator it)
|
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lock(mutex);
|
|
|
|
external_waiters.erase(it);
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
|
|
|
void do_continuation(boost::unique_lock<boost::mutex>& lock)
|
|
|
|
{
|
|
|
|
if (continuation_ptr) {
|
2013-08-25 14:35:47 -07:00
|
|
|
continuation_ptr->launch_continuation(lock);
|
|
|
|
if (! lock.owns_lock())
|
|
|
|
lock.lock();
|
|
|
|
continuation_ptr.reset();
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_continuation(boost::unique_lock<boost::mutex>&)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
2013-08-25 14:35:47 -07:00
|
|
|
void set_continuation_ptr(continuation_ptr_type continuation, boost::unique_lock<boost::mutex>& lock)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
continuation_ptr= continuation;
|
2013-02-13 20:03:52 -08:00
|
|
|
if (done) {
|
|
|
|
do_continuation(lock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
void mark_finished_internal(boost::unique_lock<boost::mutex>& lock)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
done=true;
|
|
|
|
waiters.notify_all();
|
|
|
|
for(waiter_list::const_iterator it=external_waiters.begin(),
|
|
|
|
end=external_waiters.end();it!=end;++it)
|
|
|
|
{
|
|
|
|
(*it)->notify_all();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
do_continuation(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
void make_ready()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
mark_finished_internal(lock);
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
void do_callback(boost::unique_lock<boost::mutex>& lock)
|
|
|
|
{
|
|
|
|
if(callback && !done)
|
|
|
|
{
|
|
|
|
boost::function<void()> local_callback=callback;
|
|
|
|
relocker relock(lock);
|
|
|
|
local_callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
void wait_internal(boost::unique_lock<boost::mutex> &lk, bool rethrow=true)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
do_callback(lk);
|
2013-08-25 14:35:47 -07:00
|
|
|
//if (!done) // fixme why this doesn't work?
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
if (is_deferred_)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
is_deferred_=false;
|
2014-03-01 11:00:20 -08:00
|
|
|
execute(lk);
|
|
|
|
//lk.unlock();
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
else
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
while(!done)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
waiters.wait(lk);
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
if(rethrow && thread_was_interrupted)
|
|
|
|
{
|
|
|
|
throw boost::thread_interrupted();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if(rethrow && exception)
|
|
|
|
{
|
|
|
|
boost::rethrow_exception(exception);
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
virtual void wait(bool rethrow=true)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
wait_internal(lock, rethrow);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_USES_DATETIME
|
2012-05-09 21:45:30 -07:00
|
|
|
bool timed_wait_until(boost::system_time const& target_time)
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
2013-02-13 20:03:52 -08:00
|
|
|
if (is_deferred_)
|
|
|
|
return false;
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
do_callback(lock);
|
|
|
|
while(!done)
|
|
|
|
{
|
|
|
|
bool const success=waiters.timed_wait(lock,target_time);
|
|
|
|
if(!success && !done)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-07-21 11:37:40 -07:00
|
|
|
#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(mutex);
|
2013-02-13 20:03:52 -08:00
|
|
|
if (is_deferred_)
|
|
|
|
return future_status::deferred;
|
2012-07-21 11:37:40 -07:00
|
|
|
do_callback(lock);
|
|
|
|
while(!done)
|
|
|
|
{
|
|
|
|
cv_status const st=waiters.wait_until(lock,abs_time);
|
|
|
|
if(st==cv_status::timeout && !done)
|
|
|
|
{
|
|
|
|
return future_status::timeout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return future_status::ready;
|
|
|
|
}
|
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
void mark_exceptional_finish_internal(boost::exception_ptr const& e, boost::unique_lock<boost::mutex>& lock)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
exception=e;
|
2013-02-13 20:03:52 -08:00
|
|
|
mark_finished_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
void mark_exceptional_finish()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
mark_exceptional_finish_internal(boost::current_exception(), lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2012-07-21 11:37:40 -07:00
|
|
|
void mark_interrupted_finish()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
2012-07-21 11:37:40 -07:00
|
|
|
thread_was_interrupted=true;
|
2013-02-13 20:03:52 -08:00
|
|
|
mark_finished_internal(lock);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_interrupted_at_thread_exit()
|
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(mutex);
|
|
|
|
thread_was_interrupted=true;
|
|
|
|
if (has_value(lk))
|
|
|
|
{
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_exception_at_thread_exit(exception_ptr e)
|
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(mutex);
|
|
|
|
if (has_value(lk))
|
|
|
|
{
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
}
|
|
|
|
exception=e;
|
2013-08-25 14:35:47 -07:00
|
|
|
this->is_constructed = true;
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
bool has_value() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lock(mutex);
|
2013-02-13 20:03:52 -08:00
|
|
|
return done && !(exception
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
|| thread_was_interrupted
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
bool has_value(unique_lock<boost::mutex>& ) const
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
return done && !(exception
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
|| thread_was_interrupted
|
|
|
|
#endif
|
|
|
|
);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
bool has_exception() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lock(mutex);
|
2013-02-13 20:03:52 -08:00
|
|
|
return done && (exception
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
|| thread_was_interrupted
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
bool has_exception(unique_lock<boost::mutex>&) const
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
return done && (exception
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
|| thread_was_interrupted
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
bool is_deferred(boost::lock_guard<boost::mutex>&) const {
|
2013-08-25 14:35:47 -07:00
|
|
|
return is_deferred_;
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
launch launch_policy(boost::unique_lock<boost::mutex>&) const
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
return policy_;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
future_state::state get_state() const
|
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> guard(mutex);
|
|
|
|
if(!done)
|
|
|
|
{
|
|
|
|
return future_state::waiting;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return future_state::ready;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
exception_ptr get_exception_ptr()
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
return get_exception_ptr(lock);
|
|
|
|
}
|
|
|
|
exception_ptr get_exception_ptr(boost::unique_lock<boost::mutex>& lock)
|
|
|
|
{
|
|
|
|
wait_internal(lock, false);
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
if(thread_was_interrupted)
|
|
|
|
{
|
|
|
|
return copy_exception(boost::thread_interrupted());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return exception;
|
|
|
|
}
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename F,typename U>
|
|
|
|
void set_wait_callback(F f,U* u)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::lock_guard<boost::mutex> lock(mutex);
|
2012-05-09 21:45:30 -07:00
|
|
|
callback=boost::bind(f,boost::ref(*u));
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
virtual void execute(boost::unique_lock<boost::mutex>&) {}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state_base(shared_state_base const&);
|
|
|
|
shared_state_base& operator=(shared_state_base const&);
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct future_traits
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef boost::scoped_ptr<T> storage_type;
|
|
|
|
struct dummy;
|
2013-01-13 14:38:19 -08:00
|
|
|
#ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef T const& source_reference_type;
|
|
|
|
//typedef typename boost::mpl::if_<boost::is_fundamental<T>,dummy&,BOOST_THREAD_RV_REF(T)>::type rvalue_source_type;
|
|
|
|
typedef BOOST_THREAD_RV_REF(T) rvalue_source_type;
|
|
|
|
//typedef typename boost::mpl::if_<boost::is_fundamental<T>,T,BOOST_THREAD_RV_REF(T)>::type move_dest_type;
|
|
|
|
typedef T move_dest_type;
|
2012-07-21 11:37:40 -07:00
|
|
|
#elif defined BOOST_THREAD_USES_MOVE
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef typename boost::mpl::if_c<boost::is_fundamental<T>::value,T,T&>::type source_reference_type;
|
|
|
|
//typedef typename boost::mpl::if_c<boost::is_fundamental<T>::value,T,BOOST_THREAD_RV_REF(T)>::type rvalue_source_type;
|
|
|
|
//typedef typename boost::mpl::if_c<boost::enable_move_utility_emulation<T>::value,BOOST_THREAD_RV_REF(T),T>::type move_dest_type;
|
|
|
|
typedef BOOST_THREAD_RV_REF(T) rvalue_source_type;
|
|
|
|
typedef T move_dest_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef T& source_reference_type;
|
|
|
|
typedef typename boost::mpl::if_<boost::thread_detail::is_convertible<T&,BOOST_THREAD_RV_REF(T) >,BOOST_THREAD_RV_REF(T),T const&>::type rvalue_source_type;
|
|
|
|
typedef typename boost::mpl::if_<boost::thread_detail::is_convertible<T&,BOOST_THREAD_RV_REF(T) >,BOOST_THREAD_RV_REF(T),T>::type move_dest_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef const T& shared_future_get_result_type;
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
static void init(storage_type& storage,source_reference_type t)
|
|
|
|
{
|
|
|
|
storage.reset(new T(t));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init(storage_type& storage,rvalue_source_type t)
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
|
|
storage.reset(new T(boost::forward<T>(t)));
|
|
|
|
#else
|
2012-07-21 11:37:40 -07:00
|
|
|
storage.reset(new T(static_cast<rvalue_source_type>(t)));
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void cleanup(storage_type& storage)
|
|
|
|
{
|
|
|
|
storage.reset();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct future_traits<T&>
|
|
|
|
{
|
|
|
|
typedef T* storage_type;
|
|
|
|
typedef T& source_reference_type;
|
2013-08-25 14:35:47 -07:00
|
|
|
//struct rvalue_source_type
|
|
|
|
//{};
|
2012-05-09 21:45:30 -07:00
|
|
|
typedef T& move_dest_type;
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef T& shared_future_get_result_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
static void init(storage_type& storage,T& t)
|
|
|
|
{
|
|
|
|
storage=&t;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cleanup(storage_type& storage)
|
|
|
|
{
|
|
|
|
storage=0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct future_traits<void>
|
|
|
|
{
|
|
|
|
typedef bool storage_type;
|
|
|
|
typedef void move_dest_type;
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef void shared_future_get_result_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
static void init(storage_type& storage)
|
|
|
|
{
|
|
|
|
storage=true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cleanup(storage_type& storage)
|
|
|
|
{
|
|
|
|
storage=false;
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
// Used to create stand-alone futures
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename T>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct shared_state:
|
|
|
|
detail::shared_state_base
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
typedef typename future_traits<T>::storage_type storage_type;
|
|
|
|
typedef typename future_traits<T>::source_reference_type source_reference_type;
|
|
|
|
typedef typename future_traits<T>::rvalue_source_type rvalue_source_type;
|
|
|
|
typedef typename future_traits<T>::move_dest_type move_dest_type;
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef typename future_traits<T>::shared_future_get_result_type shared_future_get_result_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
storage_type result;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state():
|
2012-05-09 21:45:30 -07:00
|
|
|
result(0)
|
|
|
|
{}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
~shared_state()
|
2013-08-25 14:35:47 -07:00
|
|
|
{}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
void mark_finished_with_result_internal(source_reference_type result_, boost::unique_lock<boost::mutex>& lock)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
future_traits<T>::init(result,result_);
|
2013-08-25 14:35:47 -07:00
|
|
|
this->mark_finished_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void mark_finished_with_result_internal(rvalue_source_type result_, boost::unique_lock<boost::mutex>& lock)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
|
|
future_traits<T>::init(result,boost::forward<T>(result_));
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
future_traits<T>::init(result,static_cast<rvalue_source_type>(result_));
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
|
|
|
this->mark_finished_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void mark_finished_with_result(source_reference_type result_)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
2013-08-25 14:35:47 -07:00
|
|
|
this->mark_finished_with_result_internal(result_, lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
void mark_finished_with_result(rvalue_source_type result_)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
|
|
mark_finished_with_result_internal(boost::forward<T>(result_), lock);
|
|
|
|
#else
|
2013-02-13 20:03:52 -08:00
|
|
|
mark_finished_with_result_internal(static_cast<rvalue_source_type>(result_), lock);
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
virtual move_dest_type get()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
wait();
|
2013-08-25 14:35:47 -07:00
|
|
|
return boost::move(*result);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual shared_future_get_result_type get_sh()
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
|
|
|
wait();
|
2013-08-25 14:35:47 -07:00
|
|
|
return *result;
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
//void set_value_at_thread_exit(const T & result_)
|
|
|
|
void set_value_at_thread_exit(source_reference_type result_)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(this->mutex);
|
|
|
|
if (this->has_value(lk))
|
|
|
|
{
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
//future_traits<T>::init(result,result_);
|
|
|
|
result.reset(new T(result_));
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
this->is_constructed = true;
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
//void set_value_at_thread_exit(BOOST_THREAD_RV_REF(T) result_)
|
|
|
|
void set_value_at_thread_exit(rvalue_source_type result_)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(this->mutex);
|
|
|
|
if (this->has_value(lk))
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
result.reset(new T(boost::move(result_)));
|
|
|
|
//future_traits<T>::init(result,static_cast<rvalue_source_type>(result_));
|
|
|
|
this->is_constructed = true;
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state(shared_state const&);
|
|
|
|
shared_state& operator=(shared_state const&);
|
2013-02-13 20:03:52 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct shared_state<T&>:
|
|
|
|
detail::shared_state_base
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef typename future_traits<T&>::storage_type storage_type;
|
|
|
|
typedef typename future_traits<T&>::source_reference_type source_reference_type;
|
|
|
|
typedef typename future_traits<T&>::move_dest_type move_dest_type;
|
|
|
|
typedef typename future_traits<T&>::shared_future_get_result_type shared_future_get_result_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
T* result;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state():
|
2013-02-13 20:03:52 -08:00
|
|
|
result(0)
|
|
|
|
{}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
~shared_state()
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
void mark_finished_with_result_internal(source_reference_type result_, boost::unique_lock<boost::mutex>& lock)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
//future_traits<T>::init(result,result_);
|
|
|
|
result= &result_;
|
|
|
|
mark_finished_internal(lock);
|
|
|
|
}
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
void mark_finished_with_result(source_reference_type result_)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
mark_finished_with_result_internal(result_, lock);
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual T& get()
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
wait();
|
|
|
|
return *result;
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual T& get_sh()
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
wait();
|
|
|
|
return *result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_value_at_thread_exit(T& result_)
|
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(this->mutex);
|
|
|
|
if (this->has_value(lk))
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
//future_traits<T>::init(result,result_);
|
|
|
|
result= &result_;
|
|
|
|
this->is_constructed = true;
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state(shared_state const&);
|
|
|
|
shared_state& operator=(shared_state const&);
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct shared_state<void>:
|
|
|
|
detail::shared_state_base
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef void shared_future_get_result_type;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state()
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void mark_finished_with_result_internal(boost::unique_lock<boost::mutex>& lock)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
mark_finished_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void mark_finished_with_result()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
mark_finished_with_result_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual void get()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
this->wait();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
virtual void get_sh()
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
|
|
|
wait();
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_value_at_thread_exit()
|
|
|
|
{
|
|
|
|
unique_lock<boost::mutex> lk(this->mutex);
|
|
|
|
if (this->has_value(lk))
|
|
|
|
{
|
|
|
|
throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
this->is_constructed = true;
|
|
|
|
detail::make_ready_at_thread_exit(shared_from_this());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_state(shared_state const&);
|
|
|
|
shared_state& operator=(shared_state const&);
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
/////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
/// future_async_shared_state_base
|
2013-02-13 20:03:52 -08:00
|
|
|
/////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
template<typename Rp>
|
|
|
|
struct future_async_shared_state_base: shared_state<Rp>
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef shared_state<Rp> base_type;
|
|
|
|
protected:
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::thread thr_;
|
2014-03-01 11:00:20 -08:00
|
|
|
void join()
|
|
|
|
{
|
|
|
|
if (thr_.joinable()) thr_.join();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
future_async_shared_state_base()
|
|
|
|
{
|
|
|
|
this->set_async();
|
|
|
|
}
|
|
|
|
explicit future_async_shared_state_base(BOOST_THREAD_RV_REF(boost::thread) th) :
|
|
|
|
thr_(boost::move(th))
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
this->set_async();
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
~future_async_shared_state_base()
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
join();
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
virtual void wait(bool rethrow)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
join();
|
|
|
|
this->base_type::wait(rethrow);
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
/////////////////////////
|
|
|
|
/// future_async_shared_state
|
|
|
|
/////////////////////////
|
|
|
|
template<typename Rp, typename Fp>
|
|
|
|
struct future_async_shared_state: future_async_shared_state_base<Rp>
|
|
|
|
{
|
|
|
|
typedef future_async_shared_state_base<Rp> base_type;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f) :
|
|
|
|
base_type(thread(&future_async_shared_state::run, this, boost::forward<Fp>(f)))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void run(future_async_shared_state* that, BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
that->mark_finished_with_result(f());
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
that->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
that->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_async_shared_state<void, Fp>: public future_async_shared_state_base<void>
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef future_async_shared_state_base<void> base_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
explicit future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f) :
|
|
|
|
base_type(thread(&future_async_shared_state::run, this, boost::forward<Fp>(f)))
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
static void run(future_async_shared_state* that, BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
f();
|
|
|
|
that->mark_finished_with_result();
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
that->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
that->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
template<typename Rp, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_async_shared_state<Rp&, Fp>: future_async_shared_state_base<Rp&>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef future_async_shared_state_base<Rp&> base_type;
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
explicit future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f) :
|
|
|
|
base_type(thread(&future_async_shared_state::run, this, boost::forward<Fp>(f)))
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
static void run(future_async_shared_state* that, BOOST_THREAD_FWD_REF(Fp) f)
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
that->mark_finished_with_result(f());
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
that->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
that->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
//////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
/// future_deferred_shared_state
|
2013-02-13 20:03:52 -08:00
|
|
|
//////////////////////////
|
|
|
|
template<typename Rp, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_deferred_shared_state: shared_state<Rp>
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef shared_state<Rp> base_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
Fp func_;
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f)
|
2013-08-25 14:35:47 -07:00
|
|
|
: func_(boost::forward<Fp>(f))
|
|
|
|
{
|
|
|
|
this->set_deferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void execute(boost::unique_lock<boost::mutex>& lck) {
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
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);
|
2013-08-25 14:35:47 -07:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish_internal(current_exception(), lck);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
template<typename Rp, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_deferred_shared_state<Rp&,Fp>: shared_state<Rp&>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef shared_state<Rp&> base_type;
|
2013-08-25 14:35:47 -07:00
|
|
|
Fp func_;
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
: func_(boost::forward<Fp>(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>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_deferred_shared_state<void,Fp>: shared_state<void>
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef shared_state<void> base_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
Fp func_;
|
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
explicit future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
: func_(boost::forward<Fp>(f))
|
|
|
|
{
|
|
|
|
this->set_deferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void execute(boost::unique_lock<boost::mutex>& lck) {
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
Fp local_fuct=boost::move(func_);
|
|
|
|
relocker relock(lck);
|
|
|
|
local_fuct();
|
|
|
|
relock.lock();
|
2013-02-13 20:03:52 -08:00
|
|
|
this->mark_finished_with_result_internal(lck);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish_internal(current_exception(), lck);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// template<typename T, typename Allocator>
|
2014-03-01 11:00:20 -08:00
|
|
|
// struct shared_state_alloc: public shared_state<T>
|
2013-02-13 20:03:52 -08:00
|
|
|
// {
|
2014-03-01 11:00:20 -08:00
|
|
|
// typedef shared_state<T> base;
|
2013-02-13 20:03:52 -08:00
|
|
|
// Allocator alloc_;
|
|
|
|
//
|
|
|
|
// public:
|
2014-03-01 11:00:20 -08:00
|
|
|
// explicit shared_state_alloc(const Allocator& a)
|
2013-02-13 20:03:52 -08:00
|
|
|
// : alloc_(a) {}
|
|
|
|
//
|
|
|
|
// };
|
|
|
|
class future_waiter
|
|
|
|
{
|
|
|
|
struct registered_waiter;
|
|
|
|
typedef std::vector<int>::size_type count_type;
|
|
|
|
|
|
|
|
struct registered_waiter
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
boost::shared_ptr<detail::shared_state_base> future_;
|
|
|
|
detail::shared_state_base::waiter_list::iterator wait_iterator;
|
2012-05-09 21:45:30 -07:00
|
|
|
count_type index;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
registered_waiter(boost::shared_ptr<detail::shared_state_base> const& a_future,
|
|
|
|
detail::shared_state_base::waiter_list::iterator wait_iterator_,
|
2012-05-09 21:45:30 -07:00
|
|
|
count_type index_):
|
2012-07-21 11:37:40 -07:00
|
|
|
future_(a_future),wait_iterator(wait_iterator_),index(index_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct all_futures_lock
|
|
|
|
{
|
2013-01-13 14:38:19 -08:00
|
|
|
#ifdef _MANAGED
|
2013-02-13 20:03:52 -08:00
|
|
|
typedef std::ptrdiff_t count_type_portable;
|
2013-01-13 14:38:19 -08:00
|
|
|
#else
|
2013-02-13 20:03:52 -08:00
|
|
|
typedef count_type count_type_portable;
|
2013-01-13 14:38:19 -08:00
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
count_type_portable count;
|
|
|
|
boost::scoped_array<boost::unique_lock<boost::mutex> > locks;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
all_futures_lock(std::vector<registered_waiter>& futures):
|
|
|
|
count(futures.size()),locks(new boost::unique_lock<boost::mutex>[count])
|
|
|
|
{
|
2013-01-13 14:38:19 -08:00
|
|
|
for(count_type_portable i=0;i<count;++i)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-01-13 14:38:19 -08:00
|
|
|
#if defined __DECCXX || defined __SUNPRO_CC || defined __hpux
|
2012-07-21 11:37:40 -07:00
|
|
|
locks[i]=boost::unique_lock<boost::mutex>(futures[i].future_->mutex).move();
|
2012-05-09 21:45:30 -07:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
locks[i]=boost::unique_lock<boost::mutex>(futures[i].future_->mutex);
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void lock()
|
|
|
|
{
|
|
|
|
boost::lock(locks.get(),locks.get()+count);
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlock()
|
|
|
|
{
|
2013-01-13 14:38:19 -08:00
|
|
|
for(count_type_portable i=0;i<count;++i)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
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)
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(f.future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
futures.push_back(registered_waiter(f.future_,f.future_->register_external_waiter(cv),future_count));
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
++future_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
count_type wait()
|
|
|
|
{
|
|
|
|
all_futures_lock lk(futures);
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
for(count_type i=0;i<futures.size();++i)
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(futures[i].future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
return futures[i].index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cv.wait(lk);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~future_waiter()
|
|
|
|
{
|
|
|
|
for(count_type i=0;i<futures.size();++i)
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
futures[i].future_->remove_external_waiter(futures[i].wait_iterator);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename R>
|
2012-07-21 11:37:40 -07:00
|
|
|
class BOOST_THREAD_FUTURE;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
template <typename R>
|
|
|
|
class shared_future;
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct is_future_type
|
|
|
|
{
|
|
|
|
BOOST_STATIC_CONSTANT(bool, value=false);
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef void type;
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
2012-07-21 11:37:40 -07:00
|
|
|
struct is_future_type<BOOST_THREAD_FUTURE<T> >
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
BOOST_STATIC_CONSTANT(bool, value=true);
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef T type;
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct is_future_type<shared_future<T> >
|
|
|
|
{
|
|
|
|
BOOST_STATIC_CONSTANT(bool, value=true);
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef T type;
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename Iterator>
|
|
|
|
typename boost::disable_if<is_future_type<Iterator>,void>::type wait_for_all(Iterator begin,Iterator end)
|
|
|
|
{
|
|
|
|
for(Iterator current=begin;current!=end;++current)
|
|
|
|
{
|
|
|
|
current->wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F1,typename F2>
|
|
|
|
typename boost::enable_if<is_future_type<F1>,void>::type wait_for_all(F1& f1,F2& f2)
|
|
|
|
{
|
|
|
|
f1.wait();
|
|
|
|
f2.wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F1,typename F2,typename F3>
|
|
|
|
void wait_for_all(F1& f1,F2& f2,F3& f3)
|
|
|
|
{
|
|
|
|
f1.wait();
|
|
|
|
f2.wait();
|
|
|
|
f3.wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F1,typename F2,typename F3,typename F4>
|
|
|
|
void wait_for_all(F1& f1,F2& f2,F3& f3,F4& f4)
|
|
|
|
{
|
|
|
|
f1.wait();
|
|
|
|
f2.wait();
|
|
|
|
f3.wait();
|
|
|
|
f4.wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F1,typename F2,typename F3,typename F4,typename F5>
|
|
|
|
void wait_for_all(F1& f1,F2& f2,F3& f3,F4& f4,F5& f5)
|
|
|
|
{
|
|
|
|
f1.wait();
|
|
|
|
f2.wait();
|
|
|
|
f3.wait();
|
|
|
|
f4.wait();
|
|
|
|
f5.wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F1,typename F2>
|
|
|
|
typename boost::enable_if<is_future_type<F1>,unsigned>::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>
|
|
|
|
unsigned 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>
|
|
|
|
unsigned 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>
|
|
|
|
unsigned 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();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename R>
|
|
|
|
class promise;
|
|
|
|
|
|
|
|
template <typename R>
|
|
|
|
class packaged_task;
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
namespace detail
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
/// Common implementation for all the futures independently of the return type
|
|
|
|
class base_future
|
|
|
|
{
|
|
|
|
//BOOST_THREAD_MOVABLE(base_future)
|
|
|
|
|
|
|
|
};
|
|
|
|
/// Common implementation for future and shared_future.
|
|
|
|
template <typename R>
|
|
|
|
class basic_future : public base_future
|
|
|
|
{
|
|
|
|
protected:
|
2014-03-01 11:00:20 -08:00
|
|
|
public:
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::shared_state<R> > future_ptr;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
future_ptr future_;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
basic_future(future_ptr a_future):
|
|
|
|
future_(a_future)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
// Copy construction from a shared_future
|
|
|
|
explicit basic_future(const shared_future<R>&) BOOST_NOEXCEPT;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
public:
|
2012-05-09 21:45:30 -07:00
|
|
|
typedef future_state::state state;
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
BOOST_THREAD_MOVABLE(basic_future)
|
|
|
|
basic_future(): future_() {}
|
|
|
|
~basic_future() {}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
basic_future(BOOST_THREAD_RV_REF(basic_future) other) BOOST_NOEXCEPT:
|
|
|
|
future_(BOOST_THREAD_RV(other).future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_RV(other).future_.reset();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
basic_future& operator=(BOOST_THREAD_RV_REF(basic_future) other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
future_=BOOST_THREAD_RV(other).future_;
|
|
|
|
BOOST_THREAD_RV(other).future_.reset();
|
2012-05-09 21:45:30 -07:00
|
|
|
return *this;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
void swap(basic_future& that) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
future_.swap(that.future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
// functions to check state, and wait for ready
|
2014-03-01 11:00:20 -08:00
|
|
|
state get_state() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(!future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
return future_state::uninitialized;
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
return future_->get_state();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
bool is_ready() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
return get_state()==future_state::ready;
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
bool has_exception() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
return future_ && future_->has_exception();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
bool has_value() const
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
return future_ && future_->has_value();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
launch launch_policy(boost::unique_lock<boost::mutex>& lk) const
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
if ( future_ ) return future_->launch_policy(lk);
|
2013-02-13 20:03:52 -08:00
|
|
|
else return launch(launch::none);
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
exception_ptr get_exception_ptr()
|
|
|
|
{
|
|
|
|
return future_
|
|
|
|
? future_->get_exception_ptr()
|
|
|
|
: exception_ptr();
|
|
|
|
}
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
bool valid() const BOOST_NOEXCEPT
|
|
|
|
{
|
|
|
|
return future_ != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
void wait() const
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(!future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
future_->wait(false);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_USES_DATETIME
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename Duration>
|
|
|
|
bool timed_wait(Duration const& rel_time) const
|
|
|
|
{
|
|
|
|
return timed_wait_until(boost::get_system_time()+rel_time);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool timed_wait_until(boost::system_time const& abs_time) const
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(!future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
return future_->timed_wait_until(abs_time);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-07-21 11:37:40 -07:00
|
|
|
#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);
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
|
|
|
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
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
} // detail
|
|
|
|
BOOST_THREAD_DCL_MOVABLE_BEG(R) detail::basic_future<R> BOOST_THREAD_DCL_MOVABLE_END
|
|
|
|
|
|
|
|
namespace detail
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
#if (!defined _MSC_VER || _MSC_VER >= 1400) // _MSC_VER == 1400 on MSVC 2005
|
2013-02-13 20:03:52 -08:00
|
|
|
template <class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f);
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
template <class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f);
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif // #if (!defined _MSC_VER || _MSC_VER >= 1400)
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
2014-03-01 11:00:20 -08:00
|
|
|
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;
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
template <class F, class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_async_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c);
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
template <class F, class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
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_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);
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
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>
|
2014-03-01 11:00:20 -08:00
|
|
|
friend struct detail::future_async_continuation_shared_state;
|
2013-08-25 14:35:47 -07:00
|
|
|
template <typename, typename, typename>
|
2014-03-01 11:00:20 -08:00
|
|
|
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_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_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);
|
2013-02-13 20:03:52 -08:00
|
|
|
#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
|
|
|
|
friend class detail::future_waiter;
|
|
|
|
|
|
|
|
template <class Rp, class Fp>
|
|
|
|
friend BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f);
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
template <class Rp, class Fp>
|
|
|
|
friend BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f);
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
typedef typename detail::future_traits<R>::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;
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef R value_type; // EXTENSION
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
BOOST_CONSTEXPR BOOST_THREAD_FUTURE() {}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
~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))))
|
|
|
|
{
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
inline BOOST_THREAD_FUTURE(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R> >) other); // EXTENSION
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
BOOST_THREAD_FUTURE& operator=(BOOST_THREAD_RV_REF(BOOST_THREAD_FUTURE) other) BOOST_NOEXCEPT
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
this->base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other))));
|
2013-02-13 20:03:52 -08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
// todo this function must be private and friendship provided to the internal users.
|
2013-08-25 14:35:47 -07:00
|
|
|
void set_async()
|
|
|
|
{
|
|
|
|
this->future_->set_async();
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
// todo this function must be private and friendship provided to the internal users.
|
2013-08-25 14:35:47 -07:00
|
|
|
void set_deferred()
|
|
|
|
{
|
|
|
|
this->future_->set_deferred();
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
// retrieving the value
|
|
|
|
move_dest_type get()
|
|
|
|
{
|
|
|
|
if(!this->future_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
future_ptr fut_=this->future_;
|
2014-03-01 11:00:20 -08:00
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
2013-02-13 20:03:52 -08:00
|
|
|
this->future_.reset();
|
2014-03-01 11:00:20 -08:00
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
return fut_->get();
|
2014-03-01 11:00:20 -08:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
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_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
this->future_->wait(false);
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
|
|
|
this->future_.reset();
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
if (fut_->has_value()) {
|
|
|
|
return fut_->get();
|
|
|
|
}
|
|
|
|
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_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
this->future_->wait(false);
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
|
|
|
this->future_.reset();
|
|
|
|
#endif
|
|
|
|
if (fut_->has_value()) {
|
|
|
|
return fut_->get();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return v;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
|
|
|
|
|
|
|
// template<typename F>
|
|
|
|
// auto then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>;
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
//#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(RF(*func)(BOOST_THREAD_FUTURE&));
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(launch policy, RF(*func)(BOOST_THREAD_FUTURE&));
|
|
|
|
//#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
template<typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type>
|
|
|
|
then(BOOST_THREAD_FWD_REF(F) func); // EXTENSION
|
2013-02-13 20:03:52 -08:00
|
|
|
template<typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE)>::type>
|
|
|
|
then(launch policy, BOOST_THREAD_FWD_REF(F) func); // EXTENSION
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
//#if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP
|
|
|
|
// inline
|
|
|
|
// typename boost::enable_if<
|
|
|
|
// is_future_type<value_type>,
|
|
|
|
// value_type
|
|
|
|
// //BOOST_THREAD_FUTURE<typename is_future_type<value_type>::type>
|
|
|
|
// >::type
|
|
|
|
// unwrap();
|
|
|
|
//#endif
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
};
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
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_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_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_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 detail::future_traits<R>::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_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();
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
move_dest_type get()
|
|
|
|
{
|
|
|
|
if(!this->future_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
|
|
|
this->future_.reset();
|
|
|
|
#endif
|
|
|
|
return fut_->get();
|
|
|
|
}
|
|
|
|
move_dest_type get_or(BOOST_THREAD_RV_REF(R) v) // EXTENSION
|
|
|
|
{
|
|
|
|
if(!this->future_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
this->future_->wait(false);
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
|
|
|
this->future_.reset();
|
|
|
|
#endif
|
|
|
|
if (fut_->has_value()) return fut_->get();
|
|
|
|
else return boost::move(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
move_dest_type get_or(R const& v) // EXTENSION
|
|
|
|
{
|
|
|
|
if(!this->future_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
this->future_->wait(false);
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
#ifdef BOOST_THREAD_PROVIDES_FUTURE_INVALID_AFTER_GET
|
|
|
|
this->future_.reset();
|
|
|
|
#endif
|
|
|
|
if (fut_->has_value()) return fut_->get();
|
|
|
|
else return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
|
|
|
|
|
|
|
// template<typename F>
|
|
|
|
// auto then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>;
|
|
|
|
|
|
|
|
//#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(RF(*func)(BOOST_THREAD_FUTURE&));
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(launch policy, RF(*func)(BOOST_THREAD_FUTURE&));
|
|
|
|
//#endif
|
|
|
|
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
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP
|
|
|
|
inline
|
|
|
|
BOOST_THREAD_FUTURE<R2>
|
|
|
|
unwrap(); // EXTENSION
|
|
|
|
#endif
|
|
|
|
|
|
|
|
};
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
template <typename R>
|
2013-02-13 20:03:52 -08:00
|
|
|
class shared_future : public detail::basic_future<R>
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
typedef detail::basic_future<R> base_type;
|
|
|
|
typedef typename base_type::future_ptr future_ptr;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
friend class detail::future_waiter;
|
|
|
|
friend class promise<R>;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
#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_deferred_continuation_shared_state(boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c);
|
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
#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
|
2012-07-21 11:37:40 -07:00
|
|
|
shared_future(future_ptr a_future):
|
2013-02-13 20:03:52 -08:00
|
|
|
base_type(a_future)
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
|
|
|
public:
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_MOVABLE(shared_future)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef R value_type; // EXTENSION
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
shared_future(shared_future const& other):
|
2013-02-13 20:03:52 -08:00
|
|
|
base_type(other)
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
|
|
|
typedef future_state::state state;
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
BOOST_CONSTEXPR shared_future()
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
|
|
|
~shared_future()
|
|
|
|
{}
|
|
|
|
|
|
|
|
shared_future& operator=(shared_future const& other)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
shared_future(other).swap(*this);
|
2012-05-09 21:45:30 -07:00
|
|
|
return *this;
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
shared_future(BOOST_THREAD_RV_REF(shared_future) other) BOOST_NOEXCEPT :
|
2013-02-13 20:03:52 -08:00
|
|
|
base_type(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other))))
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_RV(other).future_.reset();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
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))))
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
shared_future& operator=(BOOST_THREAD_RV_REF(shared_future) other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other))));
|
2012-05-09 21:45:30 -07:00
|
|
|
return *this;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
shared_future& operator=(BOOST_THREAD_RV_REF( BOOST_THREAD_FUTURE<R> ) other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
base_type::operator=(boost::move(static_cast<base_type&>(BOOST_THREAD_RV(other))));
|
2012-05-09 21:45:30 -07:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
void swap(shared_future& other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
static_cast<base_type*>(this)->swap(other);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// retrieving the value
|
2014-03-01 11:00:20 -08:00
|
|
|
typename detail::shared_state<R>::shared_future_get_result_type get()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
if(!this->future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
return this->future_->get_sh();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
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) // EXTENSION
|
|
|
|
{
|
|
|
|
if(!this->future_)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_uninitialized());
|
|
|
|
}
|
|
|
|
future_ptr fut_=this->future_;
|
|
|
|
fut_->wait();
|
|
|
|
if (fut_->has_value()) return fut_->get_sh();
|
|
|
|
else return boost::move(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
|
|
|
|
|
|
|
// template<typename F>
|
|
|
|
// auto then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>;
|
|
|
|
// template<typename F>
|
|
|
|
// auto then(launch, F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>;
|
|
|
|
|
|
|
|
//#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(RF(*func)(shared_future&));
|
|
|
|
// template<typename RF>
|
|
|
|
// inline BOOST_THREAD_FUTURE<RF> then(launch policy, RF(*func)(shared_future&));
|
|
|
|
//#endif
|
|
|
|
template<typename F>
|
|
|
|
inline BOOST_THREAD_FUTURE<typename boost::result_of<F(shared_future)>::type>
|
|
|
|
then(BOOST_THREAD_FWD_REF(F) func); // 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); // EXTENSION
|
|
|
|
#endif
|
|
|
|
//#if defined BOOST_THREAD_PROVIDES_FUTURE_UNWRAP
|
|
|
|
// inline
|
|
|
|
// typename boost::enable_if_c<
|
|
|
|
// is_future_type<value_type>::value,
|
|
|
|
// BOOST_THREAD_FUTURE<typename is_future_type<value_type>::type>
|
|
|
|
// >::type
|
|
|
|
// unwrap();
|
|
|
|
//#endif
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
BOOST_THREAD_DCL_MOVABLE_BEG(T) shared_future<T> BOOST_THREAD_DCL_MOVABLE_END
|
|
|
|
|
|
|
|
namespace detail
|
|
|
|
{
|
|
|
|
/// Copy construction from a shared_future
|
|
|
|
template <typename R>
|
|
|
|
inline basic_future<R>::basic_future(const shared_future<R>& other) BOOST_NOEXCEPT
|
|
|
|
: future_(other.future_)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename R>
|
|
|
|
class promise
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::shared_state<R> > future_ptr;
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
future_ptr future_;
|
|
|
|
bool future_obtained;
|
|
|
|
|
|
|
|
void lazy_init()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
2013-08-25 14:35:47 -07:00
|
|
|
#include <boost/detail/atomic_undef_macros.hpp>
|
|
|
|
if(!atomic_load(&future_))
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
future_ptr blank;
|
2014-03-01 11:00:20 -08:00
|
|
|
atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<R>));
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#include <boost/detail/atomic_redef_macros.hpp>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
public:
|
|
|
|
BOOST_THREAD_MOVABLE_ONLY(promise)
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
template <class Allocator>
|
|
|
|
promise(boost::allocator_arg_t, Allocator a)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<detail::shared_state<R> >::other A2;
|
2013-02-13 20:03:52 -08:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<R>(), D(a2, 1) );
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained = false;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
promise():
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
|
|
|
future_(),
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
future_(new detail::shared_state<R>()),
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
future_obtained(false)
|
|
|
|
{}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
~promise()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
if(future_)
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
if(!future_->done && !future_->is_constructed)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock);
|
|
|
|
}
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
// Assignment
|
|
|
|
promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT :
|
|
|
|
future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
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;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void swap(promise& other)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
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)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::throw_exception(promise_moved());
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
if (future_obtained)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_already_retrieved());
|
|
|
|
}
|
|
|
|
future_obtained=true;
|
|
|
|
return BOOST_THREAD_FUTURE<R>(future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_value(typename detail::future_traits<R>::source_reference_type r)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
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);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
// void set_value(R && r);
|
|
|
|
void set_value(typename detail::future_traits<R>::rvalue_source_type r)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
lazy_init();
|
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
|
|
|
if(future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::throw_exception(promise_already_satisfied());
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
|
|
future_->mark_finished_with_result_internal(boost::forward<R>(r), lock);
|
|
|
|
#else
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_finished_with_result_internal(static_cast<typename detail::future_traits<R>::rvalue_source_type>(r), lock);
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_exception(boost::exception_ptr p)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
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);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception(E ex)
|
|
|
|
{
|
|
|
|
set_exception(copy_exception(ex));
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit(const R& r)
|
|
|
|
{
|
|
|
|
if (future_.get()==0)
|
|
|
|
{
|
|
|
|
boost::throw_exception(promise_moved());
|
|
|
|
}
|
|
|
|
future_->set_value_at_thread_exit(r);
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
void set_value_at_thread_exit(BOOST_THREAD_RV_REF(R) r)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
if (future_.get()==0)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::throw_exception(promise_moved());
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
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);
|
2012-07-21 11:37:40 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception_at_thread_exit(E ex)
|
|
|
|
{
|
|
|
|
set_exception_at_thread_exit(copy_exception(ex));
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
template<typename F>
|
|
|
|
void set_wait_callback(F f)
|
|
|
|
{
|
|
|
|
lazy_init();
|
|
|
|
future_->set_wait_callback(f,this);
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
template <typename R>
|
2013-02-13 20:03:52 -08:00
|
|
|
class promise<R&>
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::shared_state<R&> > future_ptr;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
future_ptr future_;
|
2012-05-09 21:45:30 -07:00
|
|
|
bool future_obtained;
|
|
|
|
|
|
|
|
void lazy_init()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
2013-08-25 14:35:47 -07:00
|
|
|
#include <boost/detail/atomic_undef_macros.hpp>
|
2012-07-21 11:37:40 -07:00
|
|
|
if(!atomic_load(&future_))
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
future_ptr blank;
|
2014-03-01 11:00:20 -08:00
|
|
|
atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<R&>));
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#include <boost/detail/atomic_redef_macros.hpp>
|
2012-07-21 11:37:40 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_MOVABLE_ONLY(promise)
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
template <class Allocator>
|
|
|
|
promise(boost::allocator_arg_t, Allocator a)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<detail::shared_state<R&> >::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<R&>(), D(a2, 1) );
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
promise():
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
2012-07-21 11:37:40 -07:00
|
|
|
future_(),
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
future_(new detail::shared_state<R&>()),
|
2012-07-21 11:37:40 -07:00
|
|
|
#endif
|
|
|
|
future_obtained(false)
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
|
|
|
~promise()
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
if(!future_->done && !future_->is_constructed)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assignment
|
2012-07-21 11:37:40 -07:00
|
|
|
promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT :
|
|
|
|
future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_RV(rhs).future_.reset();
|
|
|
|
BOOST_THREAD_RV(rhs).future_obtained=false;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
promise & operator=(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
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;
|
2012-05-09 21:45:30 -07:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void swap(promise& other)
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
future_.swap(other.future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
std::swap(future_obtained,other.future_obtained);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Result retrieval
|
2013-02-13 20:03:52 -08:00
|
|
|
BOOST_THREAD_FUTURE<R&> get_future()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
lazy_init();
|
2012-07-21 11:37:40 -07:00
|
|
|
if (future_.get()==0)
|
|
|
|
{
|
|
|
|
boost::throw_exception(promise_moved());
|
|
|
|
}
|
|
|
|
if (future_obtained)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(future_already_retrieved());
|
|
|
|
}
|
|
|
|
future_obtained=true;
|
2013-02-13 20:03:52 -08:00
|
|
|
return BOOST_THREAD_FUTURE<R&>(future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_value(R& r)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
lazy_init();
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_finished_with_result_internal(r, lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
void set_exception(boost::exception_ptr p)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
lazy_init();
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_exceptional_finish_internal(p, lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception(E ex)
|
|
|
|
{
|
|
|
|
set_exception(copy_exception(ex));
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit(R& r)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
if (future_.get()==0)
|
|
|
|
{
|
|
|
|
boost::throw_exception(promise_moved());
|
|
|
|
}
|
|
|
|
future_->set_value_at_thread_exit(r);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
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);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception_at_thread_exit(E ex)
|
|
|
|
{
|
|
|
|
set_exception_at_thread_exit(copy_exception(ex));
|
|
|
|
}
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename F>
|
|
|
|
void set_wait_callback(F f)
|
|
|
|
{
|
|
|
|
lazy_init();
|
2012-07-21 11:37:40 -07:00
|
|
|
future_->set_wait_callback(f,this);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
template <>
|
|
|
|
class promise<void>
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::shared_state<void> > future_ptr;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
future_ptr future_;
|
2012-05-09 21:45:30 -07:00
|
|
|
bool future_obtained;
|
|
|
|
|
|
|
|
void lazy_init()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
2012-07-21 11:37:40 -07:00
|
|
|
if(!atomic_load(&future_))
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
future_ptr blank;
|
2014-03-01 11:00:20 -08:00
|
|
|
atomic_compare_exchange(&future_,&blank,future_ptr(new detail::shared_state<void>));
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
public:
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_MOVABLE_ONLY(promise)
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
template <class Allocator>
|
|
|
|
promise(boost::allocator_arg_t, Allocator a)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<detail::shared_state<void> >::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
future_ = future_ptr(::new(a2.allocate(1)) detail::shared_state<void>(), D(a2, 1) );
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
promise():
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_PROMISE_LAZY
|
2012-07-21 11:37:40 -07:00
|
|
|
future_(),
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
future_(new detail::shared_state<void>),
|
2012-07-21 11:37:40 -07:00
|
|
|
#endif
|
|
|
|
future_obtained(false)
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
|
|
|
~promise()
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
if(!future_->done && !future_->is_constructed)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_exceptional_finish_internal(boost::copy_exception(broken_promise()), lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assignment
|
2012-07-21 11:37:40 -07:00
|
|
|
promise(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT :
|
|
|
|
future_(BOOST_THREAD_RV(rhs).future_),future_obtained(BOOST_THREAD_RV(rhs).future_obtained)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
// 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;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
promise & operator=(BOOST_THREAD_RV_REF(promise) rhs) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
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;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void swap(promise& other)
|
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
future_.swap(other.future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
std::swap(future_obtained,other.future_obtained);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Result retrieval
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_FUTURE<void> get_future()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
lazy_init();
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
if (future_.get()==0)
|
|
|
|
{
|
|
|
|
boost::throw_exception(promise_moved());
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
if(future_obtained)
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_already_retrieved());
|
|
|
|
}
|
|
|
|
future_obtained=true;
|
2012-07-21 11:37:40 -07:00
|
|
|
return BOOST_THREAD_FUTURE<void>(future_);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void set_value()
|
|
|
|
{
|
|
|
|
lazy_init();
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_finished_with_result_internal(lock);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void set_exception(boost::exception_ptr p)
|
|
|
|
{
|
|
|
|
lazy_init();
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(future_->mutex);
|
2012-07-21 11:37:40 -07:00
|
|
|
if(future_->done)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
boost::throw_exception(promise_already_satisfied());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
future_->mark_exceptional_finish_internal(p,lock);
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception(E ex)
|
|
|
|
{
|
|
|
|
set_exception(copy_exception(ex));
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
// 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);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename E>
|
|
|
|
void set_exception_at_thread_exit(E ex)
|
|
|
|
{
|
|
|
|
set_exception_at_thread_exit(copy_exception(ex));
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
template<typename F>
|
|
|
|
void set_wait_callback(F f)
|
|
|
|
{
|
|
|
|
lazy_init();
|
2012-07-21 11:37:40 -07:00
|
|
|
future_->set_wait_callback(f,this);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
namespace container
|
|
|
|
{
|
|
|
|
template <class R, class Alloc>
|
|
|
|
struct uses_allocator<promise<R> , Alloc> : true_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_DCL_MOVABLE_BEG(T) promise<T> BOOST_THREAD_DCL_MOVABLE_END
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
namespace detail
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_base_shared_state;
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename R, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_base_shared_state<R(ArgTypes...)>:
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_base_shared_state<R()>:
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_base_shared_state:
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
detail::shared_state<R>
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
bool started;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
task_base_shared_state():
|
2012-05-09 21:45:30 -07:00
|
|
|
started(false)
|
|
|
|
{}
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
void reset()
|
|
|
|
{
|
|
|
|
started=false;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#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;
|
2012-05-09 21:45:30 -07:00
|
|
|
void run()
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
{
|
|
|
|
boost::lock_guard<boost::mutex> lk(this->mutex);
|
|
|
|
if(started)
|
|
|
|
{
|
|
|
|
boost::throw_exception(task_already_started());
|
|
|
|
}
|
|
|
|
started=true;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
do_run(boost::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
do_run();
|
2013-02-13 20:03:52 -08:00
|
|
|
#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::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
|
|
|
do_apply();
|
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void owner_destroyed()
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
boost::unique_lock<boost::mutex> lk(this->mutex);
|
2012-05-09 21:45:30 -07:00
|
|
|
if(!started)
|
|
|
|
{
|
|
|
|
started=true;
|
2013-02-13 20:03:52 -08:00
|
|
|
this->mark_exceptional_finish_internal(boost::copy_exception(boost::broken_promise()), lk);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
template<typename F, typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state;
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename F, typename R, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, R(ArgTypes...)>:
|
|
|
|
task_base_shared_state<R(ArgTypes...)>
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
|
|
|
template<typename F, typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, R()>:
|
|
|
|
task_base_shared_state<R()>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<typename F, typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state:
|
|
|
|
task_base_shared_state<R>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2012-07-21 11:37:40 -07:00
|
|
|
public:
|
2012-05-09 21:45:30 -07:00
|
|
|
F f;
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(F const& f_):
|
2013-02-13 20:03:52 -08:00
|
|
|
f(f_)
|
|
|
|
{}
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(BOOST_THREAD_RV_REF(F) f_):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(boost::move(f_))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
#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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
this->set_value_at_thread_exit(f());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
|
|
|
}
|
|
|
|
#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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
#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
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2012-07-21 11:37:40 -07:00
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2013-08-25 14:35:47 -07:00
|
|
|
template<typename F, typename R, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, R&(ArgTypes...)>:
|
|
|
|
task_base_shared_state<R&(ArgTypes...)>
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
template<typename F, typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, R&()>:
|
|
|
|
task_base_shared_state<R&()>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
template<typename F, typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F,R&>:
|
|
|
|
task_base_shared_state<R&>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2013-08-25 14:35:47 -07:00
|
|
|
public:
|
|
|
|
F f;
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(F const& f_):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(f_)
|
|
|
|
{}
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(BOOST_THREAD_RV_REF(F) f_):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(boost::move(f_))
|
|
|
|
{}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2013-08-25 14:35:47 -07:00
|
|
|
void do_apply(BOOST_THREAD_RV_REF(ArgTypes) ... args)
|
|
|
|
{
|
|
|
|
try
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
this->set_value_at_thread_exit(f(boost::forward<ArgTypes>(args)...));
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
this->set_value_at_thread_exit(f());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2013-08-25 14:35:47 -07:00
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
R& res((f()));
|
|
|
|
this->mark_finished_with_result(res);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename R, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R (*)(ArgTypes...), R(ArgTypes...)>:
|
|
|
|
task_base_shared_state<R(ArgTypes...)>
|
2013-08-25 14:35:47 -07:00
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R (*)(), R()>:
|
|
|
|
task_base_shared_state<R()>
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R (*)(), R> :
|
|
|
|
task_base_shared_state<R>
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2013-08-25 14:35:47 -07:00
|
|
|
public:
|
|
|
|
R (*f)();
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(R (*f_)()):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
R r((f()));
|
|
|
|
this->set_value_at_thread_exit(boost::move(r));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->set_exception_at_thread_exit(current_exception());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
#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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
2013-01-13 14:38:19 -08:00
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
R res((f()));
|
|
|
|
this->mark_finished_with_result(boost::move(res));
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2013-01-13 14:38:19 -08:00
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2013-08-25 14:35:47 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename R, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R& (*)(ArgTypes...), R&(ArgTypes...)>:
|
|
|
|
task_base_shared_state<R&(ArgTypes...)>
|
2013-08-25 14:35:47 -07:00
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R& (*)(), R&()>:
|
|
|
|
task_base_shared_state<R&()>
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<typename R>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<R& (*)(), R&> :
|
|
|
|
task_base_shared_state<R&>
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2013-08-25 14:35:47 -07:00
|
|
|
public:
|
|
|
|
R& (*f)();
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(R& (*f_)()):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
this->set_value_at_thread_exit(f());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-08-25 14:35:47 -07:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->set_exception_at_thread_exit(current_exception());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
#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::forward<ArgTypes>(args)...));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
this->mark_finished_with_result(f());
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename F, typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, void(ArgTypes...)>:
|
|
|
|
task_base_shared_state<void(ArgTypes...)>
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F, void()>:
|
|
|
|
task_base_shared_state<void()>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<F,void>:
|
|
|
|
task_base_shared_state<void>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2012-07-21 11:37:40 -07:00
|
|
|
public:
|
2012-05-09 21:45:30 -07:00
|
|
|
F f;
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(F const& f_):
|
2013-02-13 20:03:52 -08:00
|
|
|
f(f_)
|
|
|
|
{}
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(BOOST_THREAD_RV_REF(F) f_):
|
2013-08-25 14:35:47 -07:00
|
|
|
f(boost::move(f_))
|
2012-05-09 21:45:30 -07:00
|
|
|
{}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#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::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
f();
|
|
|
|
#endif
|
|
|
|
this->set_value_at_thread_exit();
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
|
|
|
}
|
|
|
|
#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
|
|
|
|
{
|
|
|
|
f(boost::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
f();
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
this->mark_finished_with_result();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2012-07-21 11:37:40 -07:00
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
template<typename ...ArgTypes>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<void (*)(ArgTypes...), void(ArgTypes...)>:
|
|
|
|
task_base_shared_state<void(ArgTypes...)>
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-01-13 14:38:19 -08:00
|
|
|
template<>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<void (*)(), void()>:
|
|
|
|
task_base_shared_state<void()>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template<>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct task_shared_state<void (*)(),void>:
|
|
|
|
task_base_shared_state<void>
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
{
|
|
|
|
private:
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(task_shared_state&);
|
2013-01-13 14:38:19 -08:00
|
|
|
public:
|
|
|
|
void (*f)();
|
2014-03-01 11:00:20 -08:00
|
|
|
task_shared_state(void (*f_)()):
|
2013-01-13 14:38:19 -08:00
|
|
|
f(f_)
|
|
|
|
{}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
#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::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
|
|
|
void do_apply()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
f();
|
|
|
|
#endif
|
|
|
|
this->set_value_at_thread_exit();
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->set_interrupted_at_thread_exit();
|
|
|
|
}
|
|
|
|
#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
|
|
|
|
{
|
|
|
|
f(boost::forward<ArgTypes>(args)...);
|
|
|
|
#else
|
2013-01-13 14:38:19 -08:00
|
|
|
void do_run()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
f();
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
this->mark_finished_with_result();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
2013-01-13 14:38:19 -08:00
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
this->mark_interrupted_finish();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#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...)>
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::task_base_shared_state<R(ArgTypes...)> > task_ptr;
|
|
|
|
boost::shared_ptr<detail::task_base_shared_state<R(ArgTypes...)> > task;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
|
|
|
template<typename R>
|
|
|
|
class packaged_task<R()>
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::task_base_shared_state<R()> > task_ptr;
|
|
|
|
boost::shared_ptr<detail::task_base_shared_state<R()> > task;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename R>
|
|
|
|
class packaged_task
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef boost::shared_ptr<detail::task_base_shared_state<R> > task_ptr;
|
|
|
|
boost::shared_ptr<detail::task_base_shared_state<R> > task;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
bool future_obtained;
|
2013-02-13 20:03:52 -08:00
|
|
|
struct dummy;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
public:
|
2012-07-21 11:37:40 -07:00
|
|
|
typedef R result_type;
|
|
|
|
BOOST_THREAD_MOVABLE_ONLY(packaged_task)
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
packaged_task():
|
|
|
|
future_obtained(false)
|
|
|
|
{}
|
|
|
|
|
|
|
|
// construction and destruction
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#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)...);
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type;
|
|
|
|
task= task_ptr(new task_shared_state_type(f, boost::forward<ArgTypes>(args)...));
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained=false;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
explicit packaged_task(R(*f)())
|
|
|
|
{
|
|
|
|
typedef R(*FR)();
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R()> task_shared_state_type;
|
|
|
|
task= task_ptr(new task_shared_state_type(f));
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained=false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
explicit packaged_task(R(*f)())
|
|
|
|
{
|
|
|
|
typedef R(*FR)();
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R> task_shared_state_type;
|
|
|
|
task= task_ptr(new task_shared_state_type(f));
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained=false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
#ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
|
2012-05-09 21:45:30 -07:00
|
|
|
template <class F>
|
2013-08-25 14:35:47 -07:00
|
|
|
explicit packaged_task(BOOST_THREAD_FWD_REF(F) f
|
2014-03-01 11:00:20 -08:00
|
|
|
, typename boost::disable_if<is_same<typename decay<F>::type, packaged_task>, dummy* >::type=0
|
2013-02-13 20:03:52 -08:00
|
|
|
)
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef typename remove_cv<typename remove_reference<F>::type>::type FR;
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(new task_shared_state_type(boost::forward<F>(f)));
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
#else
|
|
|
|
template <class F>
|
2013-02-13 20:03:52 -08:00
|
|
|
explicit packaged_task(F const& f
|
2014-03-01 11:00:20 -08:00
|
|
|
, typename boost::disable_if<is_same<typename decay<F>::type, packaged_task>, dummy* >::type=0
|
2013-02-13 20:03:52 -08:00
|
|
|
)
|
|
|
|
{
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(new task_shared_state_type(f));
|
2013-02-13 20:03:52 -08:00
|
|
|
future_obtained=false;
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
template <class F>
|
2013-02-13 20:03:52 -08:00
|
|
|
explicit packaged_task(BOOST_THREAD_RV_REF(F) f)
|
|
|
|
{
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type;
|
|
|
|
task = task_ptr(new task_shared_state_type(boost::forward<F>(f)));
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R()> task_shared_state_type;
|
|
|
|
task = task_ptr(new task_shared_state_type(boost::move(f))); // TODO forward
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R> task_shared_state_type;
|
|
|
|
task = task_ptr(new task_shared_state_type(boost::forward<F>(f)));
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
future_obtained=false;
|
|
|
|
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
2012-07-21 11:37:40 -07:00
|
|
|
template <class Allocator>
|
|
|
|
packaged_task(boost::allocator_arg_t, Allocator a, R(*f)())
|
|
|
|
{
|
|
|
|
typedef R(*FR)();
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<task_shared_state_type>::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(f), D(a2, 1) );
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif // BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR
|
|
|
|
|
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
2012-07-21 11:37:40 -07:00
|
|
|
template <class F, class Allocator>
|
2013-08-25 14:35:47 -07:00
|
|
|
packaged_task(boost::allocator_arg_t, Allocator a, BOOST_THREAD_FWD_REF(F) f)
|
2012-07-21 11:37:40 -07:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef typename remove_cv<typename remove_reference<F>::type>::type FR;
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<FR,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<task_shared_state_type>::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(boost::forward<F>(f)), D(a2, 1) );
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
#else // ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
2012-07-21 11:37:40 -07:00
|
|
|
template <class F, class Allocator>
|
|
|
|
packaged_task(boost::allocator_arg_t, Allocator a, const F& f)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<task_shared_state_type>::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
2012-05-09 21:45:30 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(f), D(a2, 1) );
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
|
|
|
template <class F, class Allocator>
|
|
|
|
packaged_task(boost::allocator_arg_t, Allocator a, BOOST_THREAD_RV_REF(F) f)
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R(ArgTypes...)> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R()> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef detail::task_shared_state<F,R> task_shared_state_type;
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename Allocator::template rebind<task_shared_state_type>::other A2;
|
2012-07-21 11:37:40 -07:00
|
|
|
A2 a2(a);
|
|
|
|
typedef thread_detail::allocator_destructor<A2> D;
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(boost::forward<F>(f)), D(a2, 1) );
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2014-03-01 11:00:20 -08:00
|
|
|
task = task_ptr(::new(a2.allocate(1)) task_shared_state_type(boost::move(f)), D(a2, 1) ); // TODO forward
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2012-07-21 11:37:40 -07:00
|
|
|
future_obtained = false;
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-01-13 14:38:19 -08:00
|
|
|
#endif //BOOST_NO_CXX11_RVALUE_REFERENCES
|
2012-07-21 11:37:40 -07:00
|
|
|
#endif // BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
~packaged_task()
|
|
|
|
{
|
|
|
|
if(task)
|
|
|
|
{
|
|
|
|
task->owner_destroyed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// assignment
|
2012-07-21 11:37:40 -07:00
|
|
|
packaged_task(BOOST_THREAD_RV_REF(packaged_task) other) BOOST_NOEXCEPT :
|
|
|
|
future_obtained(BOOST_THREAD_RV(other).future_obtained)
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
task.swap(BOOST_THREAD_RV(other).task);
|
|
|
|
BOOST_THREAD_RV(other).future_obtained=false;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
packaged_task& operator=(BOOST_THREAD_RV_REF(packaged_task) other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
// todo use forward
|
2013-08-25 14:35:47 -07:00
|
|
|
#if ! defined BOOST_NO_CXX11_RVALUE_REFERENCES
|
|
|
|
packaged_task temp(boost::move(other));
|
|
|
|
#else
|
2012-07-21 11:37:40 -07:00
|
|
|
packaged_task temp(static_cast<BOOST_THREAD_RV_REF(packaged_task)>(other));
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2012-05-09 21:45:30 -07:00
|
|
|
swap(temp);
|
|
|
|
return *this;
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
|
|
|
|
void reset()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
2012-07-21 11:37:40 -07:00
|
|
|
if (!valid())
|
|
|
|
throw future_error(system::make_error_code(future_errc::no_state));
|
|
|
|
task->reset();
|
|
|
|
future_obtained=false;
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
void swap(packaged_task& other) BOOST_NOEXCEPT
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
task.swap(other.task);
|
|
|
|
std::swap(future_obtained,other.future_obtained);
|
|
|
|
}
|
2012-07-21 11:37:40 -07:00
|
|
|
bool valid() const BOOST_NOEXCEPT
|
|
|
|
{
|
|
|
|
return task.get()!=0;
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
|
|
|
|
// result retrieval
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_FUTURE<R> get_future()
|
2012-05-09 21:45:30 -07:00
|
|
|
{
|
|
|
|
if(!task)
|
|
|
|
{
|
|
|
|
boost::throw_exception(task_moved());
|
|
|
|
}
|
|
|
|
else if(!future_obtained)
|
|
|
|
{
|
|
|
|
future_obtained=true;
|
2012-07-21 11:37:40 -07:00
|
|
|
return BOOST_THREAD_FUTURE<R>(task);
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
boost::throw_exception(future_already_retrieved());
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
//return BOOST_THREAD_FUTURE<R>();
|
2012-05-09 21:45:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// execution
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
|
|
|
void operator()(BOOST_THREAD_RV_REF(ArgTypes)... args)
|
|
|
|
{
|
|
|
|
if(!task)
|
|
|
|
{
|
|
|
|
boost::throw_exception(task_moved());
|
|
|
|
}
|
|
|
|
task->run(boost::forward<ArgTypes>(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::forward<ArgTypes>(args)...);
|
|
|
|
}
|
|
|
|
#else
|
2012-05-09 21:45:30 -07:00
|
|
|
void operator()()
|
|
|
|
{
|
|
|
|
if(!task)
|
|
|
|
{
|
|
|
|
boost::throw_exception(task_moved());
|
|
|
|
}
|
|
|
|
task->run();
|
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
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
|
2012-05-09 21:45:30 -07:00
|
|
|
template<typename F>
|
|
|
|
void set_wait_callback(F f)
|
|
|
|
{
|
|
|
|
task->set_wait_callback(f,this);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CTOR_ALLOCATORS
|
|
|
|
namespace container
|
|
|
|
{
|
|
|
|
template <class R, class Alloc>
|
|
|
|
struct uses_allocator<packaged_task<R>, Alloc>
|
|
|
|
: public true_type {};
|
|
|
|
}
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
|
|
|
|
2012-07-21 11:37:40 -07:00
|
|
|
BOOST_THREAD_DCL_MOVABLE_BEG(T) packaged_task<T> BOOST_THREAD_DCL_MOVABLE_END
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
namespace detail
|
|
|
|
{
|
|
|
|
////////////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
// make_future_deferred_shared_state
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
|
|
|
template <class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_deferred_shared_state(BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_ptr<future_deferred_shared_state<Rp, Fp> >
|
|
|
|
h(new future_deferred_shared_state<Rp, Fp>(boost::forward<Fp>(f)));
|
2013-02-13 20:03:52 -08:00
|
|
|
return BOOST_THREAD_FUTURE<Rp>(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
// make_future_async_shared_state
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
|
|
|
template <class Rp, class Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_async_shared_state(BOOST_THREAD_FWD_REF(Fp) f)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_ptr<future_async_shared_state<Rp, Fp> >
|
|
|
|
h(new future_async_shared_state<Rp, Fp>(boost::forward<Fp>(f)));
|
2013-02-13 20:03:52 -08:00
|
|
|
return BOOST_THREAD_FUTURE<Rp>(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////
|
|
|
|
// template <class F, class... ArgTypes>
|
2013-08-25 14:35:47 -07:00
|
|
|
// future<R> async(launch policy, F&&, ArgTypes&&...);
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
2012-07-21 11:37:40 -07:00
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#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)
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
typedef R(*F)(BOOST_THREAD_FWD_REF(ArgTypes)...);
|
|
|
|
typedef detail::async_func<typename decay<F>::type, typename decay<ArgTypes>::type...> BF;
|
|
|
|
typedef typename BF::result_type Rp;
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-01-13 14:38:19 -08:00
|
|
|
template <class R>
|
|
|
|
BOOST_THREAD_FUTURE<R>
|
|
|
|
async(launch policy, R(*f)())
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
typedef packaged_task<R()> packaged_task_type;
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
template <class R>
|
|
|
|
BOOST_THREAD_FUTURE<R>
|
|
|
|
async(launch policy, R(*f)())
|
|
|
|
{
|
|
|
|
typedef packaged_task<R> packaged_task_type;
|
|
|
|
#endif
|
|
|
|
if (int(policy) & int(launch::async))
|
2013-01-13 14:38:19 -08:00
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_async_shared_state<Rp>(
|
2013-08-25 14:35:47 -07:00
|
|
|
BF(
|
|
|
|
thread_detail::decay_copy(boost::forward<F>(f))
|
|
|
|
, thread_detail::decay_copy(boost::forward<ArgTypes>(args))...
|
|
|
|
)
|
|
|
|
));
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
packaged_task_type pt( f );
|
|
|
|
|
|
|
|
BOOST_THREAD_FUTURE<R> ret = BOOST_THREAD_MAKE_RV_REF(pt.get_future());
|
|
|
|
ret.set_async();
|
2013-01-13 14:38:19 -08:00
|
|
|
boost::thread( boost::move(pt) ).detach();
|
|
|
|
return ::boost::move(ret);
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
|
|
|
else if (int(policy) & int(launch::deferred))
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_deferred_shared_state<Rp>(
|
2013-08-25 14:35:47 -07:00
|
|
|
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);
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
} else {
|
2013-08-25 14:35:47 -07:00
|
|
|
std::terminate();
|
2013-01-13 14:38:19 -08:00
|
|
|
BOOST_THREAD_FUTURE<R> ret;
|
|
|
|
return ::boost::move(ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK
|
|
|
|
#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 typename boost::result_of<typename decay<F>::type(
|
|
|
|
typename decay<ArgTypes>::type...
|
|
|
|
)>::type R;
|
|
|
|
|
|
|
|
typedef detail::async_func<typename decay<F>::type, typename decay<ArgTypes>::type...> BF;
|
|
|
|
typedef typename BF::result_type Rp;
|
|
|
|
|
|
|
|
#else
|
|
|
|
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;
|
|
|
|
typedef packaged_task<R()> packaged_task_type;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
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;
|
|
|
|
typedef packaged_task<R> packaged_task_type;
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2013-01-13 14:38:19 -08:00
|
|
|
if (int(policy) & int(launch::async))
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_async_shared_state<Rp>(
|
2013-02-13 20:03:52 -08:00
|
|
|
BF(
|
|
|
|
thread_detail::decay_copy(boost::forward<F>(f))
|
|
|
|
, thread_detail::decay_copy(boost::forward<ArgTypes>(args))...
|
|
|
|
)
|
2013-08-25 14:35:47 -07:00
|
|
|
));
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
|
|
|
packaged_task_type pt( boost::forward<F>(f) );
|
2013-01-13 14:38:19 -08:00
|
|
|
|
|
|
|
BOOST_THREAD_FUTURE<R> ret = pt.get_future();
|
2013-08-25 14:35:47 -07:00
|
|
|
ret.set_async();
|
2013-01-13 14:38:19 -08:00
|
|
|
boost::thread( boost::move(pt) ).detach();
|
|
|
|
return ::boost::move(ret);
|
2013-02-13 20:03:52 -08:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
|
|
|
else if (int(policy) & int(launch::deferred))
|
|
|
|
{
|
2013-02-13 20:03:52 -08:00
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
|
2014-03-01 11:00:20 -08:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(boost::detail::make_future_deferred_shared_state<Rp>(
|
2013-02-13 20:03:52 -08:00
|
|
|
BF(
|
|
|
|
thread_detail::decay_copy(boost::forward<F>(f))
|
|
|
|
, thread_detail::decay_copy(boost::forward<ArgTypes>(args))...
|
|
|
|
)
|
2013-08-25 14:35:47 -07:00
|
|
|
));
|
2013-02-13 20:03:52 -08:00
|
|
|
#else
|
2013-08-25 14:35:47 -07:00
|
|
|
std::terminate();
|
2013-02-13 20:03:52 -08:00
|
|
|
BOOST_THREAD_FUTURE<R> ret;
|
|
|
|
return ::boost::move(ret);
|
2014-03-01 11:00:20 -08:00
|
|
|
// return boost::detail::make_future_deferred_shared_state<Rp>(
|
2013-02-13 20:03:52 -08:00
|
|
|
// BF(
|
|
|
|
// thread_detail::decay_copy(boost::forward<F>(f))
|
|
|
|
// )
|
|
|
|
// );
|
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
|
|
|
|
} else {
|
2013-08-25 14:35:47 -07:00
|
|
|
std::terminate();
|
2013-01-13 14:38:19 -08:00
|
|
|
BOOST_THREAD_FUTURE<R> ret;
|
|
|
|
return ::boost::move(ret);
|
|
|
|
}
|
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
////////////////////////////////
|
|
|
|
// template <class F, class... ArgTypes>
|
|
|
|
// future<R> async(F&&, ArgTypes&&...);
|
|
|
|
////////////////////////////////
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_SIGNATURE_PACKAGED_TASK && 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_SIGNATURE_PACKAGED_TASK && 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
|
2013-01-13 14:38:19 -08:00
|
|
|
template <class F>
|
|
|
|
BOOST_THREAD_FUTURE<typename boost::result_of<F()>::type>
|
|
|
|
async(BOOST_THREAD_RV_REF(F) f)
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(async(launch(launch::any), boost::forward<F>(f)));
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
2013-08-25 14:35:47 -07:00
|
|
|
// make_future deprecated
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
|
|
|
template <typename T>
|
|
|
|
BOOST_THREAD_FUTURE<typename decay<T>::type> make_future(BOOST_THREAD_FWD_REF(T) value)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename decay<T>::type future_value_type;
|
|
|
|
promise<future_value_type> p;
|
|
|
|
p.set_value(boost::forward<future_value_type>(value));
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
#if defined BOOST_THREAD_USES_MOVE
|
2013-02-13 20:03:52 -08:00
|
|
|
inline BOOST_THREAD_FUTURE<void> make_future()
|
|
|
|
{
|
|
|
|
promise<void> p;
|
2013-08-25 14:35:47 -07:00
|
|
|
p.set_value();
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
|
|
}
|
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
////////////////////////////////
|
|
|
|
// make_ready_future
|
|
|
|
////////////////////////////////
|
|
|
|
template <typename T>
|
|
|
|
BOOST_THREAD_FUTURE<typename decay<T>::type> make_ready_future(BOOST_THREAD_FWD_REF(T) value)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename decay<T>::type future_value_type;
|
|
|
|
promise<future_value_type> p;
|
|
|
|
p.set_value(boost::forward<future_value_type>(value));
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined BOOST_THREAD_USES_MOVE
|
|
|
|
inline BOOST_THREAD_FUTURE<void> make_ready_future()
|
|
|
|
{
|
|
|
|
promise<void> p;
|
|
|
|
p.set_value();
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
template <typename T>
|
2014-03-01 11:00:20 -08:00
|
|
|
BOOST_THREAD_FUTURE<T> make_ready_future(exception_ptr ex)
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
promise<T> p;
|
|
|
|
p.set_exception(ex);
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
template <typename T, typename E>
|
|
|
|
BOOST_THREAD_FUTURE<T> make_ready_future(E ex)
|
|
|
|
{
|
|
|
|
promise<T> p;
|
|
|
|
p.set_exception(boost::copy_exception(ex));
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future());
|
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
template<typename CLOSURE>
|
|
|
|
make_future(CLOSURE closure) -> BOOST_THREAD_FUTURE<decltype(closure())> {
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef decltype(closure()) T;
|
2013-08-25 14:35:47 -07:00
|
|
|
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
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
|
|
|
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));
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future().share());
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline shared_future<void> make_shared_future()
|
|
|
|
{
|
|
|
|
promise<void> p;
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF(p.get_future().share());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
// ////////////////////////////////
|
|
|
|
// // make_ready_shared_future
|
|
|
|
// ////////////////////////////////
|
|
|
|
// template <typename T>
|
|
|
|
// shared_future<typename decay<T>::type> make_ready_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 p.get_future().share();
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// inline shared_future<void> make_ready_shared_future()
|
|
|
|
// {
|
|
|
|
// promise<void> p;
|
|
|
|
// return BOOST_THREAD_MAKE_RV_REF(p.get_future().share());
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// ////////////////////////////////
|
|
|
|
// // make_exceptional_shared_future
|
|
|
|
// ////////////////////////////////
|
|
|
|
// template <typename T>
|
|
|
|
// shared_future<T> make_exceptional_shared_future(exception_ptr ex)
|
|
|
|
// {
|
|
|
|
// promise<T> p;
|
|
|
|
// p.set_exception(ex);
|
|
|
|
// return p.get_future().share();
|
|
|
|
// }
|
2013-02-13 20:03:52 -08:00
|
|
|
|
|
|
|
////////////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
// detail::future_async_continuation_shared_state
|
2013-02-13 20:03:52 -08:00
|
|
|
////////////////////////////////
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
|
|
|
|
namespace detail
|
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
/////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
/// future_async_continuation_shared_state
|
2013-08-25 14:35:47 -07:00
|
|
|
/////////////////////////
|
|
|
|
|
|
|
|
template<typename F, typename Rp, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_async_continuation_shared_state: future_async_shared_state_base<Rp>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
F parent;
|
|
|
|
Fp continuation;
|
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
future_async_continuation_shared_state(
|
|
|
|
BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
) :
|
2014-03-01 11:00:20 -08:00
|
|
|
parent(boost::move(f)),
|
|
|
|
continuation(boost::move(c))
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void launch_continuation(boost::unique_lock<boost::mutex>& lock)
|
|
|
|
{
|
|
|
|
lock.unlock();
|
2014-03-01 11:00:20 -08:00
|
|
|
this->thr_ = thread(&future_async_continuation_shared_state::run, this);
|
2013-08-25 14:35:47 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
static void run(future_async_continuation_shared_state* that)
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
that->mark_finished_with_result(that->continuation(boost::move(that->parent)));
|
2013-08-25 14:35:47 -07:00
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
that->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
that->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
~future_async_continuation_shared_state()
|
|
|
|
{
|
|
|
|
this->join();
|
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename F, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_async_continuation_shared_state<F, void, Fp>: public future_async_shared_state_base<void>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
F parent;
|
|
|
|
Fp continuation;
|
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
future_async_continuation_shared_state(
|
|
|
|
BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
) :
|
2014-03-01 11:00:20 -08:00
|
|
|
parent(boost::move(f)),
|
|
|
|
continuation(boost::move(c))
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void launch_continuation(boost::unique_lock<boost::mutex>& lk)
|
|
|
|
{
|
|
|
|
lk.unlock();
|
2014-03-01 11:00:20 -08:00
|
|
|
this->thr_ = thread(&future_async_continuation_shared_state::run, this);
|
2013-08-25 14:35:47 -07:00
|
|
|
}
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
static void run(future_async_continuation_shared_state* that)
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
that->continuation(boost::move(that->parent));
|
2013-08-25 14:35:47 -07:00
|
|
|
that->mark_finished_with_result();
|
|
|
|
}
|
|
|
|
#if defined BOOST_THREAD_PROVIDES_INTERRUPTIONS
|
|
|
|
catch(thread_interrupted& )
|
|
|
|
{
|
|
|
|
that->mark_interrupted_finish();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
that->mark_exceptional_finish();
|
|
|
|
}
|
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
~future_async_continuation_shared_state()
|
|
|
|
{
|
|
|
|
this->join();
|
|
|
|
}
|
2013-08-25 14:35:47 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
/// future_deferred_continuation_shared_state
|
2013-08-25 14:35:47 -07:00
|
|
|
//////////////////////////
|
|
|
|
template<typename F, typename Rp, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_deferred_continuation_shared_state: shared_state<Rp>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
F parent;
|
|
|
|
Fp continuation;
|
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
future_deferred_continuation_shared_state(
|
|
|
|
BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
) :
|
2014-03-01 11:00:20 -08:00
|
|
|
parent(boost::move(f)),
|
|
|
|
continuation(boost::move(c))
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
this->set_deferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void launch_continuation(boost::unique_lock<boost::mutex>& lk)
|
|
|
|
{
|
|
|
|
execute(lk);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void execute(boost::unique_lock<boost::mutex>& lck) {
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
Fp local_fuct=boost::move(continuation);
|
|
|
|
F ftmp = boost::move(parent);
|
|
|
|
relocker relock(lck);
|
|
|
|
Rp res = local_fuct(boost::move(ftmp));
|
|
|
|
relock.lock();
|
|
|
|
this->mark_finished_with_result_internal(boost::move(res), lck);
|
2013-08-25 14:35:47 -07:00
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish_internal(current_exception(), lck);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename F, typename Fp>
|
2014-03-01 11:00:20 -08:00
|
|
|
struct future_deferred_continuation_shared_state<F,void,Fp>: shared_state<void>
|
2013-08-25 14:35:47 -07:00
|
|
|
{
|
|
|
|
F parent;
|
|
|
|
Fp continuation;
|
|
|
|
|
|
|
|
public:
|
2014-03-01 11:00:20 -08:00
|
|
|
future_deferred_continuation_shared_state(
|
|
|
|
BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
):
|
2014-03-01 11:00:20 -08:00
|
|
|
parent(boost::move(f)),
|
2013-08-25 14:35:47 -07:00
|
|
|
continuation(boost::move(c))
|
|
|
|
{
|
|
|
|
this->set_deferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void launch_continuation(boost::unique_lock<boost::mutex>& lk)
|
|
|
|
{
|
|
|
|
execute(lk);
|
|
|
|
}
|
|
|
|
virtual void execute(boost::unique_lock<boost::mutex>& lck) {
|
|
|
|
try
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
Fp local_fuct=boost::move(continuation);
|
|
|
|
F ftmp = boost::move(parent);
|
|
|
|
relocker relock(lck);
|
|
|
|
local_fuct(boost::move(ftmp));
|
|
|
|
relock.lock();
|
2013-08-25 14:35:47 -07:00
|
|
|
this->mark_finished_with_result_internal(lck);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->mark_exceptional_finish_internal(current_exception(), lck);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
////////////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
// make_future_deferred_continuation_shared_state
|
2013-08-25 14:35:47 -07:00
|
|
|
////////////////////////////////
|
|
|
|
template<typename F, typename Rp, typename Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_deferred_continuation_shared_state(
|
2013-08-25 14:35:47 -07:00
|
|
|
boost::unique_lock<boost::mutex> &lock,
|
2014-03-01 11:00:20 -08:00
|
|
|
BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_ptr<future_deferred_continuation_shared_state<F, Rp, Fp> >
|
|
|
|
h(new future_deferred_continuation_shared_state<F, Rp, Fp>(boost::move(f), boost::forward<Fp>(c)));
|
|
|
|
h->parent.future_->set_continuation_ptr(h, lock);
|
2013-08-25 14:35:47 -07:00
|
|
|
return BOOST_THREAD_FUTURE<Rp>(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////
|
2014-03-01 11:00:20 -08:00
|
|
|
// make_future_async_continuation_shared_state
|
2013-08-25 14:35:47 -07:00
|
|
|
////////////////////////////////
|
|
|
|
template<typename F, typename Rp, typename Fp>
|
|
|
|
BOOST_THREAD_FUTURE<Rp>
|
2014-03-01 11:00:20 -08:00
|
|
|
make_future_async_continuation_shared_state(
|
|
|
|
boost::unique_lock<boost::mutex> &lock, BOOST_THREAD_RV_REF(F) f, BOOST_THREAD_FWD_REF(Fp) c
|
2013-08-25 14:35:47 -07:00
|
|
|
)
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
shared_ptr<future_async_continuation_shared_state<F,Rp, Fp> >
|
|
|
|
h(new future_async_continuation_shared_state<F,Rp, Fp>(boost::move(f), boost::forward<Fp>(c)));
|
|
|
|
h->parent.future_->set_continuation_ptr(h, lock);
|
2013-08-25 14:35:47 -07:00
|
|
|
|
|
|
|
return BOOST_THREAD_FUTURE<Rp>(h);
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////
|
|
|
|
// template<typename F>
|
|
|
|
// auto future<R>::then(F&& func) -> BOOST_THREAD_FUTURE<decltype(func(*this))>;
|
|
|
|
////////////////////////////////
|
|
|
|
|
|
|
|
template <typename R>
|
|
|
|
template <typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type>
|
2013-08-25 14:35:47 -07:00
|
|
|
BOOST_THREAD_FUTURE<R>::then(launch policy, BOOST_THREAD_FWD_REF(F) func)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type;
|
|
|
|
BOOST_THREAD_ASSERT_PRECONDITION(this->future_!=0, future_uninitialized());
|
2013-08-25 14:35:47 -07:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
if (int(policy) & int(launch::async))
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
else if (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, F>(
|
|
|
|
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, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
)));
|
|
|
|
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
template <typename R>
|
|
|
|
template <typename F>
|
2014-03-01 11:00:20 -08:00
|
|
|
inline BOOST_THREAD_FUTURE<typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type>
|
2013-08-25 14:35:47 -07:00
|
|
|
BOOST_THREAD_FUTURE<R>::then(BOOST_THREAD_FWD_REF(F) func)
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
typedef typename boost::result_of<F(BOOST_THREAD_FUTURE<R>)>::type future_type;
|
|
|
|
BOOST_THREAD_ASSERT_PRECONDITION(this->future_!=0, future_uninitialized());
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
2014-03-01 11:00:20 -08:00
|
|
|
if (int(this->launch_policy(lock)) & int(launch::async))
|
2013-02-13 20:03:52 -08:00
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
return boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
2013-08-25 14:35:47 -07:00
|
|
|
);
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
else if (int(this->launch_policy(lock)) & int(launch::deferred))
|
2013-01-13 14:38:19 -08:00
|
|
|
{
|
2013-08-25 14:35:47 -07:00
|
|
|
this->future_->wait_internal(lock);
|
2014-03-01 11:00:20 -08:00
|
|
|
return boost::detail::make_future_deferred_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
2013-08-25 14:35:47 -07:00
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-01 11:00:20 -08:00
|
|
|
return boost::detail::make_future_async_continuation_shared_state<BOOST_THREAD_FUTURE<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
);
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
2013-02-13 20:03:52 -08:00
|
|
|
}
|
2014-03-01 11:00:20 -08:00
|
|
|
|
|
|
|
|
2013-08-25 14:35:47 -07:00
|
|
|
//#if 0 && defined(BOOST_THREAD_RVALUE_REFERENCES_DONT_MATCH_FUNTION_PTR)
|
|
|
|
// template <typename R>
|
|
|
|
// template<typename RF>
|
|
|
|
// BOOST_THREAD_FUTURE<RF>
|
|
|
|
// BOOST_THREAD_FUTURE<R>::then(RF(*func)(BOOST_THREAD_FUTURE<R>&))
|
|
|
|
// {
|
|
|
|
//
|
|
|
|
// typedef RF future_type;
|
|
|
|
//
|
|
|
|
// if (this->future_)
|
|
|
|
// {
|
|
|
|
// boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
// detail::future_continuation<BOOST_THREAD_FUTURE<R>, future_type, RF(*)(BOOST_THREAD_FUTURE&) > *ptr =
|
|
|
|
// new detail::future_continuation<BOOST_THREAD_FUTURE<R>, future_type, RF(*)(BOOST_THREAD_FUTURE&)>(*this, func);
|
|
|
|
// if (ptr==0)
|
|
|
|
// {
|
|
|
|
// return BOOST_THREAD_MAKE_RV_REF(BOOST_THREAD_FUTURE<future_type>());
|
|
|
|
// }
|
|
|
|
// this->future_->set_continuation_ptr(ptr, lock);
|
|
|
|
// return ptr->get_future();
|
|
|
|
// } else {
|
|
|
|
// // fixme what to do when the future has no associated state?
|
|
|
|
// return BOOST_THREAD_MAKE_RV_REF(BOOST_THREAD_FUTURE<future_type>());
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
// template <typename R>
|
|
|
|
// template<typename RF>
|
|
|
|
// BOOST_THREAD_FUTURE<RF>
|
|
|
|
// BOOST_THREAD_FUTURE<R>::then(launch policy, RF(*func)(BOOST_THREAD_FUTURE<R>&))
|
|
|
|
// {
|
|
|
|
//
|
|
|
|
// typedef RF future_type;
|
|
|
|
//
|
|
|
|
// if (this->future_)
|
|
|
|
// {
|
|
|
|
// boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
// detail::future_continuation<BOOST_THREAD_FUTURE<R>, future_type, RF(*)(BOOST_THREAD_FUTURE&) > *ptr =
|
|
|
|
// new detail::future_continuation<BOOST_THREAD_FUTURE<R>, future_type, RF(*)(BOOST_THREAD_FUTURE&)>(*this, func, policy);
|
|
|
|
// if (ptr==0)
|
|
|
|
// {
|
|
|
|
// return BOOST_THREAD_MAKE_RV_REF(BOOST_THREAD_FUTURE<future_type>());
|
|
|
|
// }
|
|
|
|
// this->future_->set_continuation_ptr(ptr, lock);
|
|
|
|
// return ptr->get_future();
|
|
|
|
// } else {
|
|
|
|
// // fixme what to do when the future has no associated state?
|
|
|
|
// return BOOST_THREAD_MAKE_RV_REF(BOOST_THREAD_FUTURE<future_type>());
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
//#endif
|
2013-02-13 20:03:52 -08:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
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)
|
|
|
|
{
|
|
|
|
|
|
|
|
typedef typename boost::result_of<F(shared_future<R>)>::type future_type;
|
|
|
|
BOOST_THREAD_ASSERT_PRECONDITION(this->future_!=0, future_uninitialized());
|
|
|
|
|
|
|
|
boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
if (int(policy) & int(launch::async))
|
|
|
|
{
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
else if (int(policy) & int(launch::deferred))
|
|
|
|
{
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_deferred_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return BOOST_THREAD_MAKE_RV_REF((boost::detail::make_future_async_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
|
|
|
|
typedef typename boost::result_of<F(shared_future<R>)>::type future_type;
|
|
|
|
|
|
|
|
BOOST_THREAD_ASSERT_PRECONDITION(this->future_!=0, future_uninitialized());
|
|
|
|
|
|
|
|
boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
if (int(this->launch_policy(lock)) & int(launch::async))
|
|
|
|
{
|
|
|
|
return boost::detail::make_future_async_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (int(this->launch_policy(lock)) & int(launch::deferred))
|
|
|
|
{
|
|
|
|
this->future_->wait_internal(lock);
|
|
|
|
return boost::detail::make_future_deferred_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return boost::detail::make_future_async_continuation_shared_state<shared_future<R>, future_type, F>(
|
|
|
|
lock, boost::move(*this), boost::forward<F>(func)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2012-05-09 21:45:30 -07:00
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
|
2014-03-01 11:00:20 -08:00
|
|
|
#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 parent;
|
|
|
|
public:
|
|
|
|
explicit future_unwrap_shared_state(
|
|
|
|
BOOST_THREAD_RV_REF(F) f
|
|
|
|
) :
|
|
|
|
parent(boost::move(f))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
virtual void wait(bool ) // todo see if rethrow must be used
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
parent.get().wait();
|
|
|
|
}
|
|
|
|
virtual Rp get()
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(mutex);
|
|
|
|
return parent.get().get();
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
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->parent.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_!=0, future_uninitialized());
|
|
|
|
boost::unique_lock<boost::mutex> lock(this->future_->mutex);
|
|
|
|
return boost::detail::make_future_unwrap_shared_state<BOOST_THREAD_FUTURE<BOOST_THREAD_FUTURE<R2> >, R2>(lock, boost::move(*this));
|
|
|
|
}
|
|
|
|
#endif
|
2013-01-13 14:38:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif // BOOST_NO_EXCEPTION
|
|
|
|
#endif // header
|