2018-02-03 18:22:01 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
2016-09-18 23:31:56 -07:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_CTX_FUTURE_H
|
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
namespace ircd::ctx
|
|
|
|
{
|
|
|
|
template<class T = void> class future;
|
|
|
|
template<> class future<void>;
|
2018-10-30 11:18:03 -07:00
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
template<class... T> struct scoped_future;
|
2018-10-30 11:18:03 -07:00
|
|
|
|
|
|
|
IRCD_EXCEPTION(future_error, future_already_retrieved)
|
|
|
|
IRCD_OVERLOAD(use_future)
|
2019-02-15 16:47:00 -08:00
|
|
|
|
|
|
|
template<class T> const shared_state<T> &state(const future<T> &);
|
|
|
|
template<class T> shared_state<T> &state(future<T> &);
|
|
|
|
|
|
|
|
template<class T,
|
|
|
|
class time_point>
|
2019-07-19 14:42:46 -07:00
|
|
|
bool _wait_until(const future<T> &, const time_point &, std::nothrow_t);
|
2019-02-15 16:47:00 -08:00
|
|
|
|
|
|
|
template<class T,
|
|
|
|
class time_point>
|
2019-07-19 14:42:46 -07:00
|
|
|
void _wait_until(const future<T> &, const time_point &);
|
2017-08-28 14:51:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
2018-03-10 10:57:27 -08:00
|
|
|
struct ircd::ctx::future
|
2018-03-14 11:53:25 -07:00
|
|
|
:private shared_state<T>
|
2016-09-18 23:31:56 -07:00
|
|
|
{
|
|
|
|
using value_type = typename shared_state<T>::value_type;
|
|
|
|
using pointer_type = typename shared_state<T>::pointer_type;
|
|
|
|
using reference_type = typename shared_state<T>::reference_type;
|
|
|
|
|
2018-03-14 11:53:25 -07:00
|
|
|
const shared_state<T> &state() const { return *this; }
|
|
|
|
shared_state<T> &state() { return *this; }
|
|
|
|
|
2018-03-15 11:45:01 -07:00
|
|
|
bool valid() const { return !is(state(), future_state::INVALID); }
|
2016-09-19 20:07:30 -07:00
|
|
|
bool operator!() const { return !valid(); }
|
2019-08-12 20:43:04 -07:00
|
|
|
explicit operator T() { return get(); }
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2018-04-05 21:19:16 -07:00
|
|
|
template<class U, class time_point> friend bool wait_until(const future<U> &, const time_point &, std::nothrow_t);
|
|
|
|
template<class U, class time_point> friend void wait_until(const future<U> &, const time_point &);
|
|
|
|
template<class time_point> bool wait_until(const time_point &, std::nothrow_t) const;
|
|
|
|
template<class time_point> void wait_until(const time_point &) const;
|
|
|
|
template<class duration> bool wait(const duration &d, std::nothrow_t) const;
|
|
|
|
template<class duration> void wait(const duration &d) const;
|
2016-09-18 23:31:56 -07:00
|
|
|
void wait() const;
|
|
|
|
|
|
|
|
T get();
|
2019-08-12 20:43:39 -07:00
|
|
|
template<class duration> T get(const duration &d);
|
|
|
|
template<class time_point> T get_until(const time_point &);
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2019-08-31 23:05:06 -07:00
|
|
|
using shared_state<T>::shared_state;
|
|
|
|
using shared_state<T>::operator=;
|
2016-09-18 23:31:56 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
2018-03-10 10:57:27 -08:00
|
|
|
struct ircd::ctx::future<void>
|
2018-03-14 11:53:25 -07:00
|
|
|
:private shared_state<void>
|
2016-09-18 23:31:56 -07:00
|
|
|
{
|
|
|
|
using value_type = typename shared_state<void>::value_type;
|
|
|
|
|
2018-03-14 11:53:25 -07:00
|
|
|
const shared_state<void> &state() const { return *this; }
|
|
|
|
shared_state<void> &state() { return *this; }
|
|
|
|
|
2018-03-15 11:45:01 -07:00
|
|
|
bool valid() const { return !is(state(), future_state::INVALID); }
|
2016-09-19 20:07:30 -07:00
|
|
|
bool operator!() const { return !valid(); }
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2018-04-05 21:19:16 -07:00
|
|
|
template<class U, class time_point> friend bool wait_until(const future<U> &, const time_point &, std::nothrow_t);
|
|
|
|
template<class U, class time_point> friend void wait_until(const future<U> &, const time_point &);
|
|
|
|
template<class time_point> bool wait_until(const time_point &, std::nothrow_t) const;
|
|
|
|
template<class time_point> void wait_until(const time_point &) const;
|
|
|
|
template<class duration> bool wait(const duration &d, std::nothrow_t) const;
|
|
|
|
template<class duration> void wait(const duration &d) const;
|
2016-09-18 23:31:56 -07:00
|
|
|
void wait() const;
|
|
|
|
|
2019-08-31 23:05:06 -07:00
|
|
|
using shared_state<void>::shared_state;
|
|
|
|
using shared_state<void>::operator=;
|
2016-09-18 23:31:56 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<class... T>
|
2017-08-28 14:51:22 -07:00
|
|
|
struct ircd::ctx::scoped_future
|
|
|
|
:future<T...>
|
2016-09-18 23:31:56 -07:00
|
|
|
{
|
2019-08-31 23:05:06 -07:00
|
|
|
using future<T...>::future;
|
2016-09-18 23:31:56 -07:00
|
|
|
~scoped_future() noexcept;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class... T>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline
|
2017-08-28 14:51:22 -07:00
|
|
|
ircd::ctx::scoped_future<T...>::~scoped_future()
|
2016-09-18 23:31:56 -07:00
|
|
|
noexcept
|
|
|
|
{
|
2019-08-31 23:05:06 -07:00
|
|
|
if(std::uncaught_exceptions() || !this->valid())
|
2016-09-18 23:31:56 -07:00
|
|
|
return;
|
|
|
|
|
2019-08-31 23:05:06 -07:00
|
|
|
this->wait();
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2019-08-12 20:43:39 -07:00
|
|
|
template<class T>
|
|
|
|
template<class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline T
|
2019-08-12 20:43:39 -07:00
|
|
|
ircd::ctx::future<T>::get_until(const time_point &tp)
|
|
|
|
{
|
|
|
|
this->wait_until(tp);
|
|
|
|
return this->get();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
template<class duration>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline T
|
2019-08-12 20:43:39 -07:00
|
|
|
ircd::ctx::future<T>::get(const duration &d)
|
|
|
|
{
|
|
|
|
this->wait(d);
|
|
|
|
return this->get();
|
|
|
|
}
|
|
|
|
|
2016-09-18 23:31:56 -07:00
|
|
|
template<class T>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline T
|
2017-08-28 14:51:22 -07:00
|
|
|
ircd::ctx::future<T>::get()
|
2016-09-18 23:31:56 -07:00
|
|
|
{
|
|
|
|
wait();
|
2018-03-15 11:45:01 -07:00
|
|
|
if(unlikely(is(state(), future_state::RETRIEVED)))
|
2018-03-10 17:32:54 -08:00
|
|
|
throw future_already_retrieved{};
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2018-03-15 11:45:01 -07:00
|
|
|
set(state(), future_state::RETRIEVED);
|
2018-03-14 11:53:25 -07:00
|
|
|
if(bool(state().eptr))
|
|
|
|
std::rethrow_exception(state().eptr);
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2019-08-26 13:10:12 -07:00
|
|
|
return std::move(state().val);
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-10 10:57:27 -08:00
|
|
|
template<class T>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<T>::wait()
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
this->wait_until(system_point::max());
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-10 10:57:27 -08:00
|
|
|
inline void
|
|
|
|
ircd::ctx::future<void>::wait()
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
this->wait_until(system_point::max());
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-10 10:57:27 -08:00
|
|
|
template<class T>
|
2016-09-18 23:31:56 -07:00
|
|
|
template<class duration>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<T>::wait(const duration &d)
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
this->wait_until(now<system_point>() + d);
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-07 08:17:50 -08:00
|
|
|
template<class duration>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<void>::wait(const duration &d)
|
2018-03-07 08:17:50 -08:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
this->wait_until(now<system_point>() + d);
|
2018-03-07 08:17:50 -08:00
|
|
|
}
|
|
|
|
|
2016-09-18 23:31:56 -07:00
|
|
|
template<class T>
|
|
|
|
template<class duration>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline bool
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<T>::wait(const duration &d,
|
|
|
|
std::nothrow_t)
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
return this->wait_until(now<system_point>() + d, std::nothrow);
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-07 08:17:50 -08:00
|
|
|
template<class duration>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline bool
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<void>::wait(const duration &d,
|
|
|
|
std::nothrow_t)
|
2018-03-07 08:17:50 -08:00
|
|
|
const
|
|
|
|
{
|
2019-09-22 20:02:23 -07:00
|
|
|
return this->wait_until(now<system_point>() + d, std::nothrow);
|
2018-03-07 08:17:50 -08:00
|
|
|
}
|
|
|
|
|
2016-09-23 14:50:24 -07:00
|
|
|
template<class T>
|
2016-09-18 23:31:56 -07:00
|
|
|
template<class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2017-08-28 14:51:22 -07:00
|
|
|
ircd::ctx::future<T>::wait_until(const time_point &tp)
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-07-19 18:08:37 -07:00
|
|
|
if(!this->wait_until(tp, std::nothrow))
|
|
|
|
throw timeout{};
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
|
|
|
|
2018-03-07 08:17:50 -08:00
|
|
|
template<class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<void>::wait_until(const time_point &tp)
|
2018-03-07 08:17:50 -08:00
|
|
|
const
|
|
|
|
{
|
2018-04-05 21:19:16 -07:00
|
|
|
if(!this->wait_until(tp, std::nothrow))
|
2018-03-10 17:32:54 -08:00
|
|
|
throw timeout{};
|
2018-03-07 08:17:50 -08:00
|
|
|
}
|
|
|
|
|
2018-03-10 10:57:27 -08:00
|
|
|
template<class T>
|
2016-09-18 23:31:56 -07:00
|
|
|
template<class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline bool
|
2018-03-10 10:57:27 -08:00
|
|
|
ircd::ctx::future<T>::wait_until(const time_point &tp,
|
|
|
|
std::nothrow_t)
|
2016-09-18 23:31:56 -07:00
|
|
|
const
|
|
|
|
{
|
2019-07-19 14:42:46 -07:00
|
|
|
return _wait_until(*this, tp, std::nothrow);
|
2016-09-23 14:50:24 -07:00
|
|
|
}
|
|
|
|
|
2018-03-07 08:17:50 -08:00
|
|
|
template<class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline bool
|
2018-03-07 08:17:50 -08:00
|
|
|
ircd::ctx::future<void>::wait_until(const time_point &tp,
|
|
|
|
std::nothrow_t)
|
|
|
|
const
|
|
|
|
{
|
2019-07-19 14:42:46 -07:00
|
|
|
if(_wait_until(*this, tp, std::nothrow))
|
2018-03-10 17:32:54 -08:00
|
|
|
{
|
2019-08-31 23:05:06 -07:00
|
|
|
auto &state(mutable_cast(this->state()));
|
2018-03-15 11:45:01 -07:00
|
|
|
set(state, future_state::RETRIEVED);
|
2018-03-14 11:53:25 -07:00
|
|
|
if(bool(state.eptr))
|
|
|
|
std::rethrow_exception(state.eptr);
|
2018-03-10 17:32:54 -08:00
|
|
|
|
2018-04-05 21:19:16 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else return false;
|
2018-03-07 08:17:50 -08:00
|
|
|
}
|
|
|
|
|
2016-09-23 14:50:24 -07:00
|
|
|
template<class T,
|
|
|
|
class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline void
|
2017-08-28 14:51:22 -07:00
|
|
|
ircd::ctx::wait_until(const future<T> &f,
|
|
|
|
const time_point &tp)
|
2018-03-07 08:17:50 -08:00
|
|
|
{
|
2019-07-19 14:42:46 -07:00
|
|
|
if(!_wait_until(f, tp, std::nothrow))
|
2018-03-07 08:17:50 -08:00
|
|
|
throw timeout{};
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T,
|
|
|
|
class time_point>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline bool
|
2019-07-19 14:42:46 -07:00
|
|
|
ircd::ctx::_wait_until(const future<T> &f,
|
|
|
|
const time_point &tp,
|
|
|
|
std::nothrow_t)
|
2016-09-23 14:50:24 -07:00
|
|
|
{
|
2019-08-31 23:05:06 -07:00
|
|
|
auto &state(mutable_cast(f.state()));
|
2018-03-15 11:45:01 -07:00
|
|
|
if(unlikely(is(state, future_state::INVALID)))
|
2018-03-10 10:57:27 -08:00
|
|
|
throw no_state{};
|
2016-09-18 23:31:56 -07:00
|
|
|
|
2020-07-11 15:28:24 -07:00
|
|
|
return state.cond.wait_until(tp, [&state]() noexcept
|
2019-07-19 14:42:46 -07:00
|
|
|
{
|
2019-07-19 14:45:36 -07:00
|
|
|
return !is(state, future_state::PENDING);
|
|
|
|
});
|
2016-09-18 23:31:56 -07:00
|
|
|
}
|
2018-05-05 02:00:11 -07:00
|
|
|
|
|
|
|
template<class T>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline ircd::ctx::shared_state<T> &
|
2018-05-05 02:00:11 -07:00
|
|
|
ircd::ctx::state(future<T> &future)
|
|
|
|
{
|
|
|
|
return future.state();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
2020-07-11 15:29:13 -07:00
|
|
|
inline const ircd::ctx::shared_state<T> &
|
2018-05-05 02:00:11 -07:00
|
|
|
ircd::ctx::state(const future<T> &future)
|
|
|
|
{
|
|
|
|
return future.state();
|
|
|
|
}
|