464 lines
11 KiB
C++
464 lines
11 KiB
C++
|
|
// Copyright Oliver Kowalke 2013.
|
|
// 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_FIBERS_FUTURE_HPP
|
|
#define BOOST_FIBERS_FUTURE_HPP
|
|
|
|
#include <algorithm>
|
|
#include <chrono>
|
|
#include <exception>
|
|
|
|
#include <boost/config.hpp>
|
|
|
|
#include <boost/fiber/detail/config.hpp>
|
|
#include <boost/fiber/exceptions.hpp>
|
|
#include <boost/fiber/future/detail/shared_state.hpp>
|
|
#include <boost/fiber/future/future_status.hpp>
|
|
|
|
namespace boost {
|
|
namespace fibers {
|
|
namespace detail {
|
|
|
|
template< typename R >
|
|
struct future_base {
|
|
typedef typename shared_state< R >::ptr_t ptr_t;
|
|
|
|
ptr_t state_{};
|
|
|
|
future_base() noexcept = default;
|
|
|
|
explicit future_base( ptr_t const& p) noexcept :
|
|
state_{ p } {
|
|
}
|
|
|
|
~future_base() = default;
|
|
|
|
future_base( future_base const& other) :
|
|
state_{ other.state_ } {
|
|
}
|
|
|
|
future_base( future_base && other) noexcept :
|
|
state_{ other.state_ } {
|
|
other.state_.reset();
|
|
}
|
|
|
|
future_base & operator=( future_base const& other) noexcept {
|
|
if ( this == & other) return * this;
|
|
state_ = other.state_;
|
|
return * this;
|
|
}
|
|
|
|
future_base & operator=( future_base && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
state_ = other.state_;
|
|
other.state_.reset();
|
|
return * this;
|
|
}
|
|
|
|
bool valid() const noexcept {
|
|
return nullptr != state_.get();
|
|
}
|
|
|
|
std::exception_ptr get_exception_ptr() {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return state_->get_exception_ptr();
|
|
}
|
|
|
|
void wait() const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
state_->wait();
|
|
}
|
|
|
|
template< typename Rep, typename Period >
|
|
future_status wait_for( std::chrono::duration< Rep, Period > const& timeout_duration) const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return state_->wait_for( timeout_duration);
|
|
}
|
|
|
|
template< typename Clock, typename Duration >
|
|
future_status wait_until( std::chrono::time_point< Clock, Duration > const& timeout_time) const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return state_->wait_until( timeout_time);
|
|
}
|
|
};
|
|
|
|
template< typename R >
|
|
struct promise_base;
|
|
|
|
}
|
|
|
|
template< typename R >
|
|
class shared_future;
|
|
|
|
template< typename Signature >
|
|
class packaged_task;
|
|
|
|
template< typename R >
|
|
class future : private detail::future_base< R > {
|
|
private:
|
|
typedef detail::future_base< R > base_t;
|
|
|
|
friend struct detail::promise_base< R >;
|
|
friend class shared_future< R >;
|
|
template< typename Signature >
|
|
friend class packaged_task;
|
|
|
|
explicit future( typename base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
future() noexcept = default;
|
|
|
|
future( future const&) = delete;
|
|
future & operator=( future const&) = delete;
|
|
|
|
future( future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
future & operator=( future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
shared_future< R > share();
|
|
|
|
R get() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
typename base_t::ptr_t tmp{};
|
|
tmp.swap( base_t::state_);
|
|
return std::move( tmp->get() );
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
template< typename R >
|
|
class future< R & > : private detail::future_base< R & > {
|
|
private:
|
|
typedef detail::future_base< R & > base_t;
|
|
|
|
friend struct detail::promise_base< R & >;
|
|
friend class shared_future< R & >;
|
|
template< typename Signature >
|
|
friend class packaged_task;
|
|
|
|
explicit future( typename base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
future() noexcept = default;
|
|
|
|
future( future const&) = delete;
|
|
future & operator=( future const&) = delete;
|
|
|
|
future( future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
future & operator=( future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
shared_future< R & > share();
|
|
|
|
R & get() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
typename base_t::ptr_t tmp{};
|
|
tmp.swap( base_t::state_);
|
|
return tmp->get();
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
template<>
|
|
class future< void > : private detail::future_base< void > {
|
|
private:
|
|
typedef detail::future_base< void > base_t;
|
|
|
|
friend struct detail::promise_base< void >;
|
|
friend class shared_future< void >;
|
|
template< typename Signature >
|
|
friend class packaged_task;
|
|
|
|
explicit future( base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
future() noexcept = default;
|
|
|
|
future( future const&) = delete;
|
|
future & operator=( future const&) = delete;
|
|
|
|
inline
|
|
future( future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
inline
|
|
future & operator=( future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
shared_future< void > share();
|
|
|
|
inline
|
|
void get() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
base_t::ptr_t tmp{};
|
|
tmp.swap( base_t::state_);
|
|
tmp->get();
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
|
|
template< typename R >
|
|
class shared_future : private detail::future_base< R > {
|
|
private:
|
|
typedef detail::future_base< R > base_t;
|
|
|
|
explicit shared_future( typename base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
shared_future() noexcept = default;
|
|
|
|
~shared_future() = default;
|
|
|
|
shared_future( shared_future const& other) :
|
|
base_t{ other } {
|
|
}
|
|
|
|
shared_future( shared_future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
shared_future( future< R > && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
shared_future & operator=( shared_future const& other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( other);
|
|
return * this;
|
|
}
|
|
|
|
shared_future & operator=( shared_future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
shared_future & operator=( future< R > && other) noexcept {
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
R const& get() const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return base_t::state_->get();
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
template< typename R >
|
|
class shared_future< R & > : private detail::future_base< R & > {
|
|
private:
|
|
typedef detail::future_base< R & > base_t;
|
|
|
|
explicit shared_future( typename base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
shared_future() noexcept = default;
|
|
|
|
~shared_future() = default;
|
|
|
|
shared_future( shared_future const& other) :
|
|
base_t{ other } {
|
|
}
|
|
|
|
shared_future( shared_future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
shared_future( future< R & > && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
shared_future & operator=( shared_future const& other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( other);
|
|
return * this;
|
|
}
|
|
|
|
shared_future & operator=( shared_future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
shared_future & operator=( future< R & > && other) noexcept {
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
R & get() const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return base_t::state_->get();
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
template<>
|
|
class shared_future< void > : private detail::future_base< void > {
|
|
private:
|
|
typedef detail::future_base< void > base_t;
|
|
|
|
explicit shared_future( base_t::ptr_t const& p) noexcept :
|
|
base_t{ p } {
|
|
}
|
|
|
|
public:
|
|
shared_future() noexcept = default;
|
|
|
|
~shared_future() = default;
|
|
|
|
inline
|
|
shared_future( shared_future const& other) :
|
|
base_t{ other } {
|
|
}
|
|
|
|
inline
|
|
shared_future( shared_future && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
inline
|
|
shared_future( future< void > && other) noexcept :
|
|
base_t{ std::move( other) } {
|
|
}
|
|
|
|
inline
|
|
shared_future & operator=( shared_future const& other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( other);
|
|
return * this;
|
|
}
|
|
|
|
inline
|
|
shared_future & operator=( shared_future && other) noexcept {
|
|
if ( this == & other) return * this;
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
inline
|
|
shared_future & operator=( future< void > && other) noexcept {
|
|
base_t::operator=( std::move( other) );
|
|
return * this;
|
|
}
|
|
|
|
inline
|
|
void get() const {
|
|
if ( ! valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
base_t::state_->get();
|
|
}
|
|
|
|
using base_t::valid;
|
|
using base_t::get_exception_ptr;
|
|
using base_t::wait;
|
|
using base_t::wait_for;
|
|
using base_t::wait_until;
|
|
};
|
|
|
|
|
|
template< typename R >
|
|
shared_future< R >
|
|
future< R >::share() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return shared_future< R >{ std::move( * this) };
|
|
}
|
|
|
|
template< typename R >
|
|
shared_future< R & >
|
|
future< R & >::share() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return shared_future< R & >{ std::move( * this) };
|
|
}
|
|
|
|
inline
|
|
shared_future< void >
|
|
future< void >::share() {
|
|
if ( ! base_t::valid() ) {
|
|
throw future_uninitialized{};
|
|
}
|
|
return shared_future< void >{ std::move( * this) };
|
|
}
|
|
|
|
}}
|
|
|
|
#endif
|