2018-12-24 17:58:57 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
2019-07-12 15:18:26 -07:00
|
|
|
// Copyright (C) 2016-2019 Jason Volk <jason@zemos.net>
|
2018-12-24 17:58:57 -08:00
|
|
|
//
|
|
|
|
// 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
|
2019-09-30 12:30:18 -07:00
|
|
|
#define HAVE_IRCD_CTX_CONCURRENT_H
|
2018-12-24 17:58:57 -08:00
|
|
|
|
|
|
|
namespace ircd::ctx
|
|
|
|
{
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value> struct concurrent;
|
2018-12-24 17:58:57 -08:00
|
|
|
}
|
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value>
|
2019-07-12 15:18:26 -07:00
|
|
|
struct ircd::ctx::concurrent
|
2018-12-24 17:58:57 -08:00
|
|
|
{
|
2019-07-15 12:46:19 -07:00
|
|
|
using closure = std::function<void (value)>;
|
2018-12-24 17:58:57 -08:00
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
pool &p;
|
2018-12-24 17:58:57 -08:00
|
|
|
closure c;
|
|
|
|
dock d;
|
2019-07-15 12:46:19 -07:00
|
|
|
size_t snd {0};
|
|
|
|
size_t rcv {0};
|
|
|
|
size_t fin {0};
|
2018-12-24 17:58:57 -08:00
|
|
|
std::exception_ptr eptr;
|
2018-12-26 17:12:28 -08:00
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class V> void operator()(V&&);
|
|
|
|
void wait();
|
2018-12-24 17:58:57 -08:00
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
concurrent(pool &, closure);
|
2019-07-12 15:18:26 -07:00
|
|
|
~concurrent() noexcept;
|
2018-12-24 17:58:57 -08:00
|
|
|
};
|
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value>
|
|
|
|
ircd::ctx::concurrent<value>::concurrent(pool &p,
|
|
|
|
closure c)
|
|
|
|
|
|
|
|
:p{p}
|
2018-12-24 17:58:57 -08:00
|
|
|
,c{std::move(c)}
|
2019-07-15 12:46:19 -07:00
|
|
|
{}
|
2018-12-24 17:58:57 -08:00
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value>
|
|
|
|
ircd::ctx::concurrent<value>::~concurrent()
|
2018-12-24 17:58:57 -08:00
|
|
|
noexcept
|
|
|
|
{
|
|
|
|
const uninterruptible::nothrow ui;
|
2019-07-15 12:46:19 -07:00
|
|
|
this->wait();
|
2018-12-26 17:12:28 -08:00
|
|
|
}
|
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value>
|
2018-12-26 17:12:28 -08:00
|
|
|
void
|
2019-07-15 12:46:19 -07:00
|
|
|
ircd::ctx::concurrent<value>::wait()
|
2018-12-24 17:58:57 -08:00
|
|
|
{
|
|
|
|
d.wait([this]
|
|
|
|
{
|
2019-07-15 12:46:19 -07:00
|
|
|
return snd == fin;
|
2018-12-24 17:58:57 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
template<class value>
|
|
|
|
template<class V>
|
2018-12-24 17:58:57 -08:00
|
|
|
void
|
2019-07-15 12:46:19 -07:00
|
|
|
ircd::ctx::concurrent<value>::operator()(V&& v)
|
2018-12-24 17:58:57 -08:00
|
|
|
{
|
2019-07-15 12:46:19 -07:00
|
|
|
++snd;
|
|
|
|
p([this, v(std::move(v))]
|
2018-12-24 17:58:57 -08:00
|
|
|
{
|
2019-07-15 12:46:19 -07:00
|
|
|
++rcv; try
|
|
|
|
{
|
|
|
|
c(std::move(v));
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
eptr = std::current_exception();
|
|
|
|
}
|
|
|
|
|
|
|
|
++fin;
|
|
|
|
d.notify_all();
|
2018-12-24 17:58:57 -08:00
|
|
|
});
|
2019-01-21 11:50:46 -08:00
|
|
|
|
2019-07-15 12:46:19 -07:00
|
|
|
if(eptr)
|
|
|
|
std::rethrow_exception(eptr);
|
2019-01-21 11:50:46 -08:00
|
|
|
}
|