0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-04 03:44:15 +01:00
construct/include/ircd/ctx/when.h
2019-09-03 15:56:14 -07:00

190 lines
5.3 KiB
C++

// 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.
#pragma once
#define HAVE_IRCD_CTX_WHEN_H
namespace ircd::ctx
{
template<class it, class F> future<void> when_all(it first, const it &last, F&& closure);
template<class it> future<void> when_all(it first, const it &last);
template<class it, class F> future<it> when_any(it first, const it &last, F&& closure);
template<class it> future<it> when_any(it first, const it &last);
}
// Internal interface
namespace ircd::ctx::when
{
template<class T> auto &state(const future<T> &);
void all_then(promise<void> &p);
template<class it, class F> void any_then(promise<it> &, it &, F&&);
template<class it, class F> void set_all_then(promise<void> &, it &, F&&);
template<class it, class F> void set_any_then(promise<it> &, it &, F&&);
}
/// Returns a future which becomes ready when any of the futures in the
/// iteration become ready or are already ready. The future that when_any()
/// eventually indicates is then considered "observed" which means you
/// are required to do nothing when including it in the next invocation of
/// when_any() and it won't be considered ready or pending again and the
/// collection does not have to be modified in any way.
///
/// The returned future's payload is an iterator into the collection as if
/// it were the result of an std::find() etc; thus to know its index an
/// std::distance is often satisfactory.
template<class it>
ircd::ctx::future<it>
ircd::ctx::when_any(it first,
const it &last)
{
return when_any(first, last, []
(auto &iterator) -> decltype(*iterator) &
{
return *iterator;
});
}
/// Implementation of when_any(); this requires a closure from the user which
/// knows how to use the iterable being passed. The closure must return a
/// a reference to the future. This allows for complex iterables which may
/// have pointers to pointers, etc. The default non-closure when_any() overload
/// supplies a closure that simply dereferences the argument (i.e `return *it;`)
template<class it,
class F>
ircd::ctx::future<it>
ircd::ctx::when_any(it first,
const it &last,
F&& closure)
{
const critical_assertion ca;
promise<it> p;
future<it> ret(p);
for(auto f(first); f != last; ++f)
if(is(state(closure(f)), future_state::READY))
{
set(when::state(closure(f)), future_state::OBSERVED);
p.set_value(f);
return ret;
}
for(; first != last; ++first)
if(is(state(closure(first)), future_state::PENDING))
when::set_any_then(p, first, closure);
if(promise<it>::refcount(p.state()) <= 1)
p.set_value(last);
return ret;
}
/// Returns a future which becomes ready when all of the futures in the
/// collection become ready. This future has a void payload to minimize
/// its cost since this indication is positively unate.
template<class it>
ircd::ctx::future<void>
ircd::ctx::when_all(it first,
const it &last)
{
return when_all(first, last, []
(auto &iterator) -> decltype(*iterator) &
{
return *iterator;
});
}
/// Implementation of when_all(); this requires a closure from the user which
/// knows how to use the iterable being passed. See related when_any() docs.
template<class it,
class F>
ircd::ctx::future<void>
ircd::ctx::when_all(it first,
const it &last,
F&& closure)
{
const critical_assertion ca;
promise<void> p;
future<void> ret(p);
for(; first != last; ++first)
if(is(state(closure(first)), future_state::PENDING))
when::set_all_then(p, first, closure);
if(promise<void>::refcount(p.state()) <= 1)
p.set_value();
return ret;
}
template<class it,
class F>
void
ircd::ctx::when::set_any_then(promise<it> &p,
it &f,
F&& closure)
{
when::state(closure(f)).then = [p, f, closure] // TODO: quash this alloc
(shared_state_base &sb) mutable
{
if(sb.then)
any_then(p, f, closure);
};
}
template<class it,
class F>
void
ircd::ctx::when::set_all_then(promise<void> &p,
it &f,
F&& closure)
{
when::state(closure(f)).then = [p] // TODO: quash this alloc
(shared_state_base &sb) mutable
{
if(sb.then)
all_then(p);
};
}
template<class it,
class F>
void
ircd::ctx::when::any_then(promise<it> &p,
it &f,
F&& closure)
{
if(!p.valid())
return;
set(when::state(closure(f)), future_state::OBSERVED);
p.set_value(f);
}
inline void
ircd::ctx::when::all_then(promise<void> &p)
{
if(!p.valid())
return;
if(promise<void>::refcount(p.state()) < 2)
return p.set_value();
return p.remove();
}
/// In order for this template to be reusable with std::set iterations we
/// have to make a const_cast at some point; this internal function does that.
template<class T>
auto &
ircd::ctx::when::state(const future<T> &f)
{
return const_cast<future<T> &>(f).state();
}