2018-02-04 03:22:01 +01: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.
|
2017-03-25 00:45:26 +01:00
|
|
|
|
2017-09-30 08:04:41 +02:00
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_IOS_H
|
|
|
|
|
2018-01-08 12:01:33 +01:00
|
|
|
// Boost headers are not exposed to our users unless explicitly included by a
|
|
|
|
// definition file. Other libircd headers may extend this namespace with more
|
|
|
|
// forward declarations.
|
|
|
|
|
2017-09-30 08:04:41 +02:00
|
|
|
/// Forward declarations for boost::asio because it is not included here.
|
|
|
|
namespace boost::asio
|
2017-08-28 23:51:22 +02:00
|
|
|
{
|
2020-02-27 19:11:59 +01:00
|
|
|
struct executor;
|
2017-12-29 23:53:39 +01:00
|
|
|
struct io_context;
|
2019-03-27 00:53:31 +01:00
|
|
|
|
|
|
|
template<class function>
|
|
|
|
void asio_handler_invoke(function&, ...);
|
2017-08-28 23:51:22 +02:00
|
|
|
}
|
2017-03-25 00:45:26 +01:00
|
|
|
|
2017-09-30 08:04:41 +02:00
|
|
|
namespace ircd
|
2017-03-25 00:45:26 +01:00
|
|
|
{
|
2018-10-17 14:12:10 +02:00
|
|
|
namespace asio = boost::asio; ///< Alias so that asio:: can be used.
|
2018-11-02 04:14:00 +01:00
|
|
|
|
2019-06-01 01:06:55 +02:00
|
|
|
extern const info::versions boost_version_api, boost_version_abi;
|
2018-10-17 14:12:10 +02:00
|
|
|
}
|
2017-03-25 00:45:26 +01:00
|
|
|
|
2018-10-17 14:12:10 +02:00
|
|
|
namespace ircd::ios
|
|
|
|
{
|
2019-03-27 00:53:31 +01:00
|
|
|
struct handler;
|
|
|
|
struct descriptor;
|
|
|
|
template<class function> struct handle;
|
|
|
|
|
2019-05-16 08:20:13 +02:00
|
|
|
IRCD_OVERLOAD(synchronous)
|
2019-04-16 08:46:12 +02:00
|
|
|
struct dispatch;
|
|
|
|
struct defer;
|
|
|
|
struct post;
|
|
|
|
|
2020-12-09 02:54:34 +01:00
|
|
|
constexpr bool profile_history {false};
|
|
|
|
constexpr bool profile_logging {false};
|
|
|
|
|
2020-12-07 20:19:33 +01:00
|
|
|
extern log::log log;
|
2018-10-17 14:12:10 +02:00
|
|
|
extern std::thread::id main_thread_id;
|
2020-02-27 19:11:59 +01:00
|
|
|
extern asio::executor user;
|
|
|
|
extern asio::executor main;
|
2017-10-12 02:38:24 +02:00
|
|
|
|
2020-02-27 19:11:59 +01:00
|
|
|
const string_view &name(const descriptor &);
|
|
|
|
const string_view &name(const handler &);
|
2020-06-19 04:31:17 +02:00
|
|
|
|
|
|
|
bool available() noexcept;
|
2019-08-14 04:37:30 +02:00
|
|
|
const uint64_t &epoch() noexcept;
|
2018-12-28 22:05:03 +01:00
|
|
|
|
2019-05-06 22:48:49 +02:00
|
|
|
void forked_parent();
|
|
|
|
void forked_child();
|
|
|
|
void forking();
|
2020-02-27 19:11:59 +01:00
|
|
|
void init(asio::executor &&);
|
2018-10-17 14:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace ircd
|
|
|
|
{
|
|
|
|
using ios::dispatch;
|
2019-04-13 01:26:34 +02:00
|
|
|
using ios::defer;
|
2018-10-17 14:12:10 +02:00
|
|
|
using ios::post;
|
2017-10-12 02:38:24 +02:00
|
|
|
}
|
|
|
|
|
2019-04-16 08:46:12 +02:00
|
|
|
struct ircd::ios::dispatch
|
|
|
|
{
|
|
|
|
dispatch(descriptor &, std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
dispatch(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-06 02:34:42 +02:00
|
|
|
dispatch(descriptor &, synchronous_t);
|
2019-04-16 08:46:12 +02:00
|
|
|
dispatch(std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
dispatch(synchronous_t, const std::function<void ()> &);
|
2019-04-16 08:46:12 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::ios::defer
|
|
|
|
{
|
|
|
|
defer(descriptor &, std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
defer(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-06 02:34:42 +02:00
|
|
|
defer(descriptor &, synchronous_t);
|
2019-04-16 08:46:12 +02:00
|
|
|
defer(std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
defer(synchronous_t, const std::function<void ()> &);
|
2019-04-16 08:46:12 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::ios::post
|
|
|
|
{
|
|
|
|
post(descriptor &, std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
post(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-06 02:34:42 +02:00
|
|
|
post(descriptor &, synchronous_t);
|
2019-04-16 08:46:12 +02:00
|
|
|
post(std::function<void ()>);
|
2019-05-16 08:20:13 +02:00
|
|
|
post(synchronous_t, const std::function<void ()> &);
|
2019-04-16 08:46:12 +02:00
|
|
|
};
|
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
struct ircd::ios::descriptor
|
|
|
|
:instance_list<descriptor>
|
|
|
|
{
|
2019-03-29 01:09:46 +01:00
|
|
|
struct stats;
|
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
static uint64_t ids;
|
|
|
|
|
2019-03-27 10:49:28 +01:00
|
|
|
static void *default_allocator(handler &, const size_t &);
|
2019-09-11 00:11:25 +02:00
|
|
|
static void default_deallocator(handler &, void *const &, const size_t &) noexcept;
|
2019-03-27 10:49:28 +01:00
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
string_view name;
|
|
|
|
uint64_t id {++ids};
|
2019-03-29 01:09:46 +01:00
|
|
|
std::unique_ptr<struct stats> stats;
|
2019-03-27 10:49:28 +01:00
|
|
|
std::function<void *(handler &, const size_t &)> allocator;
|
|
|
|
std::function<void (handler &, void *const &, const size_t &)> deallocator;
|
2020-08-02 14:45:35 +02:00
|
|
|
std::vector<std::array<uint64_t, 2>> history; // epoch, cycles
|
|
|
|
uint8_t history_pos {0};
|
|
|
|
bool continuation {false};
|
2019-03-27 10:49:28 +01:00
|
|
|
|
|
|
|
descriptor(const string_view &name,
|
|
|
|
const decltype(allocator) & = default_allocator,
|
2019-03-29 00:56:55 +01:00
|
|
|
const decltype(deallocator) & = default_deallocator,
|
|
|
|
const bool &continuation = false);
|
2019-03-27 10:49:28 +01:00
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
descriptor(descriptor &&) = delete;
|
|
|
|
descriptor(const descriptor &) = delete;
|
|
|
|
~descriptor() noexcept;
|
|
|
|
};
|
|
|
|
|
2019-03-29 01:09:46 +01:00
|
|
|
struct ircd::ios::descriptor::stats
|
|
|
|
{
|
2019-04-11 07:52:33 +02:00
|
|
|
uint64_t queued {0};
|
2019-03-29 01:09:46 +01:00
|
|
|
uint64_t calls {0};
|
|
|
|
uint64_t faults {0};
|
|
|
|
uint64_t allocs {0};
|
|
|
|
uint64_t alloc_bytes{0};
|
|
|
|
uint64_t frees {0};
|
|
|
|
uint64_t free_bytes{0};
|
|
|
|
uint64_t slice_total {0};
|
|
|
|
uint64_t slice_last {0};
|
2019-09-22 23:43:14 +02:00
|
|
|
uint64_t latency_total {0};
|
|
|
|
uint64_t latency_last {0};
|
2019-03-29 01:09:46 +01:00
|
|
|
|
|
|
|
stats &operator+=(const stats &) &;
|
|
|
|
|
|
|
|
stats();
|
|
|
|
~stats() noexcept;
|
|
|
|
};
|
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
struct ircd::ios::handler
|
|
|
|
{
|
2019-03-27 10:06:55 +01:00
|
|
|
static thread_local handler *current;
|
2019-07-16 01:02:40 +02:00
|
|
|
static thread_local uint64_t epoch;
|
2019-03-27 10:06:55 +01:00
|
|
|
|
2020-12-09 02:54:34 +01:00
|
|
|
static void enqueue(handler *const &) noexcept;
|
2019-03-27 05:22:22 +01:00
|
|
|
static void *allocate(handler *const &, const size_t &);
|
2019-07-16 01:01:35 +02:00
|
|
|
static void deallocate(handler *const &, void *const &, const size_t &) noexcept;
|
|
|
|
static bool continuation(handler *const &) noexcept;
|
|
|
|
static void enter(handler *const &) noexcept;
|
|
|
|
static void leave(handler *const &) noexcept;
|
|
|
|
static bool fault(handler *const &) noexcept;
|
2019-03-27 00:53:31 +01:00
|
|
|
|
2019-04-17 02:41:13 +02:00
|
|
|
ios::descriptor *descriptor {nullptr};
|
2019-09-22 23:43:14 +02:00
|
|
|
uint64_t ts {0}; // last tsc sample; for profiling each phase
|
2019-03-27 00:53:31 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
struct ircd::ios::handle
|
|
|
|
:handler
|
|
|
|
{
|
|
|
|
function f;
|
|
|
|
|
|
|
|
template<class... args>
|
|
|
|
void operator()(args&&... a) const;
|
|
|
|
|
2020-12-09 02:54:34 +01:00
|
|
|
handle(ios::descriptor &, function) noexcept;
|
2019-03-27 00:53:31 +01:00
|
|
|
};
|
|
|
|
|
2020-12-09 02:54:34 +01:00
|
|
|
// boost handlers
|
2019-03-27 00:53:31 +01:00
|
|
|
namespace ircd::ios
|
|
|
|
{
|
2019-03-27 05:22:22 +01:00
|
|
|
template<class function>
|
|
|
|
void asio_handler_deallocate(void *, size_t, handle<function> *);
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
void *asio_handler_allocate(size_t, handle<function> *);
|
|
|
|
|
2019-03-29 00:56:55 +01:00
|
|
|
template<class function>
|
|
|
|
bool asio_handler_is_continuation(handle<function> *);
|
|
|
|
|
2019-03-27 00:53:31 +01:00
|
|
|
template<class callable,
|
|
|
|
class function>
|
|
|
|
void asio_handler_invoke(callable &f, handle<function> *);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class callable,
|
|
|
|
class function>
|
2020-12-09 02:54:34 +01:00
|
|
|
inline void
|
2019-03-27 00:53:31 +01:00
|
|
|
ircd::ios::asio_handler_invoke(callable &f,
|
|
|
|
handle<function> *const h)
|
|
|
|
try
|
|
|
|
{
|
|
|
|
handler::enter(h);
|
|
|
|
boost::asio::asio_handler_invoke(f, &h);
|
|
|
|
handler::leave(h);
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
if(handler::fault(h))
|
|
|
|
handler::leave(h);
|
|
|
|
else
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
2019-03-29 00:56:55 +01:00
|
|
|
template<class function>
|
2020-12-09 02:54:34 +01:00
|
|
|
inline bool
|
2019-03-29 00:56:55 +01:00
|
|
|
ircd::ios::asio_handler_is_continuation(handle<function> *const h)
|
|
|
|
{
|
|
|
|
return handler::continuation(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class function>
|
2020-12-09 02:54:34 +01:00
|
|
|
inline void *
|
2019-03-29 00:56:55 +01:00
|
|
|
__attribute__((malloc, returns_nonnull, warn_unused_result, alloc_size(1)))
|
|
|
|
ircd::ios::asio_handler_allocate(size_t size,
|
|
|
|
handle<function> *const h)
|
|
|
|
{
|
|
|
|
return handler::allocate(h, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class function>
|
2020-12-09 02:54:34 +01:00
|
|
|
inline void
|
2019-03-29 00:56:55 +01:00
|
|
|
ircd::ios::asio_handler_deallocate(void *const ptr,
|
|
|
|
size_t size,
|
|
|
|
handle<function> *const h)
|
|
|
|
{
|
|
|
|
handler::deallocate(h, ptr, size);
|
|
|
|
}
|
|
|
|
|
2020-12-09 02:54:34 +01:00
|
|
|
//
|
|
|
|
// ircd::ios::handle
|
|
|
|
//
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
inline
|
|
|
|
ircd::ios::handle<function>::handle(ios::descriptor &d,
|
|
|
|
function f)
|
|
|
|
noexcept
|
|
|
|
:handler{&d, prof::cycles()}
|
|
|
|
,f(std::move(f))
|
|
|
|
{
|
|
|
|
handler::enqueue(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
template<class... args>
|
|
|
|
inline void
|
|
|
|
ircd::ios::handle<function>::operator()(args&&... a)
|
|
|
|
const
|
|
|
|
{
|
|
|
|
assert(descriptor && descriptor->stats);
|
|
|
|
assert(descriptor->stats->queued > 0);
|
|
|
|
descriptor->stats->queued--;
|
|
|
|
f(std::forward<args>(a)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// ircd::ios::handler
|
|
|
|
//
|
|
|
|
|
|
|
|
inline void
|
|
|
|
ircd::ios::handler::enqueue(handler *const &handler)
|
|
|
|
noexcept
|
|
|
|
{
|
|
|
|
assert(handler && handler->descriptor);
|
|
|
|
auto &descriptor(*handler->descriptor);
|
|
|
|
|
|
|
|
assert(descriptor.stats);
|
|
|
|
auto &stats(*descriptor.stats);
|
|
|
|
++stats.queued;
|
|
|
|
|
|
|
|
if constexpr(profile_logging)
|
|
|
|
log::logf
|
|
|
|
{
|
|
|
|
log, log::level::DEBUG,
|
|
|
|
"QUEUE %5u %-20s [%11lu] ------[%9lu] q:%-4lu",
|
|
|
|
descriptor.id,
|
|
|
|
trunc(descriptor.name, 20),
|
|
|
|
stats.calls,
|
|
|
|
0UL,
|
|
|
|
stats.queued,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
ircd::ios::handler::continuation(handler *const &handler)
|
|
|
|
noexcept
|
|
|
|
{
|
|
|
|
assert(handler && handler->descriptor);
|
|
|
|
auto &descriptor(*handler->descriptor);
|
|
|
|
return descriptor.continuation;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// ircd::ios
|
|
|
|
//
|
|
|
|
|
2019-04-11 00:01:11 +02:00
|
|
|
inline const ircd::string_view &
|
|
|
|
ircd::ios::name(const handler &handler)
|
|
|
|
{
|
|
|
|
assert(handler.descriptor);
|
|
|
|
return name(*handler.descriptor);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const ircd::string_view &
|
|
|
|
ircd::ios::name(const descriptor &descriptor)
|
|
|
|
{
|
|
|
|
return descriptor.name;
|
|
|
|
}
|
|
|
|
|
2019-08-14 04:37:30 +02:00
|
|
|
inline const uint64_t &
|
|
|
|
__attribute__((always_inline))
|
|
|
|
ircd::ios::epoch()
|
|
|
|
noexcept
|
|
|
|
{
|
|
|
|
return handler::epoch;
|
|
|
|
}
|