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.
|
2017-03-24 16:45:26 -07:00
|
|
|
|
2017-09-29 23:04:41 -07:00
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_IOS_H
|
|
|
|
|
2018-01-08 03:01:33 -08: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-29 23:04:41 -07:00
|
|
|
/// Forward declarations for boost::asio because it is not included here.
|
|
|
|
namespace boost::asio
|
2017-08-28 14:51:22 -07:00
|
|
|
{
|
2017-12-29 15:53:39 -07:00
|
|
|
struct io_context;
|
2018-12-11 14:55:10 -08:00
|
|
|
struct signal_set;
|
2019-03-26 16:53:31 -07:00
|
|
|
|
|
|
|
template<class function>
|
|
|
|
void asio_handler_invoke(function&, ...);
|
2017-08-28 14:51:22 -07:00
|
|
|
}
|
2017-03-24 16:45:26 -07:00
|
|
|
|
2017-09-29 23:04:41 -07:00
|
|
|
namespace ircd
|
2017-03-24 16:45:26 -07:00
|
|
|
{
|
2018-10-17 05:12:10 -07:00
|
|
|
namespace asio = boost::asio; ///< Alias so that asio:: can be used.
|
2018-11-01 20:14:00 -07:00
|
|
|
|
2019-05-31 16:06:55 -07:00
|
|
|
extern const info::versions boost_version_api, boost_version_abi;
|
2018-10-17 05:12:10 -07:00
|
|
|
}
|
2017-03-24 16:45:26 -07:00
|
|
|
|
2018-10-17 05:12:10 -07:00
|
|
|
namespace ircd::ios
|
|
|
|
{
|
2019-03-26 16:53:31 -07:00
|
|
|
struct handler;
|
|
|
|
struct descriptor;
|
|
|
|
template<class function> struct handle;
|
|
|
|
|
2019-05-15 23:20:13 -07:00
|
|
|
IRCD_OVERLOAD(synchronous)
|
2019-04-15 23:46:12 -07:00
|
|
|
struct dispatch;
|
|
|
|
struct defer;
|
|
|
|
struct post;
|
|
|
|
|
2017-10-11 17:38:24 -07:00
|
|
|
extern const std::thread::id static_thread_id;
|
2018-10-17 05:12:10 -07:00
|
|
|
extern std::thread::id main_thread_id;
|
|
|
|
extern asio::io_context *user;
|
2017-10-11 17:38:24 -07:00
|
|
|
|
2019-05-06 13:48:49 -07:00
|
|
|
const string_view &name(const descriptor &);
|
|
|
|
const string_view &name(const handler &);
|
|
|
|
|
2017-10-11 17:38:24 -07:00
|
|
|
bool is_main_thread();
|
2018-12-28 13:17:28 -08:00
|
|
|
bool is_static_thread();
|
2017-10-11 17:38:24 -07:00
|
|
|
void assert_main_thread();
|
|
|
|
|
2019-08-05 16:15:56 -07:00
|
|
|
bool available() noexcept;
|
|
|
|
asio::io_context &get() noexcept;
|
2019-08-13 19:37:30 -07:00
|
|
|
const uint64_t &epoch() noexcept;
|
2018-12-28 13:05:03 -08:00
|
|
|
|
2019-05-06 13:48:49 -07:00
|
|
|
void forked_parent();
|
|
|
|
void forked_child();
|
|
|
|
void forking();
|
2019-04-10 15:01:11 -07:00
|
|
|
|
2018-10-17 05:12:10 -07:00
|
|
|
void init(asio::io_context &user);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace ircd
|
|
|
|
{
|
|
|
|
using ios::assert_main_thread;
|
|
|
|
using ios::is_main_thread;
|
|
|
|
using ios::dispatch;
|
2019-04-12 16:26:34 -07:00
|
|
|
using ios::defer;
|
2018-10-17 05:12:10 -07:00
|
|
|
using ios::post;
|
2017-10-11 17:38:24 -07:00
|
|
|
}
|
|
|
|
|
2019-04-15 23:46:12 -07:00
|
|
|
struct ircd::ios::dispatch
|
|
|
|
{
|
|
|
|
dispatch(descriptor &, std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
dispatch(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-05 17:34:42 -07:00
|
|
|
dispatch(descriptor &, synchronous_t);
|
2019-04-15 23:46:12 -07:00
|
|
|
dispatch(std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
dispatch(synchronous_t, const std::function<void ()> &);
|
2019-04-15 23:46:12 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::ios::defer
|
|
|
|
{
|
|
|
|
defer(descriptor &, std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
defer(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-05 17:34:42 -07:00
|
|
|
defer(descriptor &, synchronous_t);
|
2019-04-15 23:46:12 -07:00
|
|
|
defer(std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
defer(synchronous_t, const std::function<void ()> &);
|
2019-04-15 23:46:12 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::ios::post
|
|
|
|
{
|
|
|
|
post(descriptor &, std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
post(descriptor &, synchronous_t, const std::function<void ()> &);
|
2019-07-05 17:34:42 -07:00
|
|
|
post(descriptor &, synchronous_t);
|
2019-04-15 23:46:12 -07:00
|
|
|
post(std::function<void ()>);
|
2019-05-15 23:20:13 -07:00
|
|
|
post(synchronous_t, const std::function<void ()> &);
|
2019-04-15 23:46:12 -07:00
|
|
|
};
|
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
struct ircd::ios::descriptor
|
|
|
|
:instance_list<descriptor>
|
|
|
|
{
|
2019-03-28 17:09:46 -07:00
|
|
|
struct stats;
|
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
static uint64_t ids;
|
|
|
|
|
2019-03-27 02:49:28 -07:00
|
|
|
static void *default_allocator(handler &, const size_t &);
|
2019-09-10 15:11:25 -07:00
|
|
|
static void default_deallocator(handler &, void *const &, const size_t &) noexcept;
|
2019-03-27 02:49:28 -07:00
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
string_view name;
|
|
|
|
uint64_t id {++ids};
|
2019-03-28 17:09:46 -07:00
|
|
|
std::unique_ptr<struct stats> stats;
|
2019-03-27 02:49:28 -07:00
|
|
|
std::function<void *(handler &, const size_t &)> allocator;
|
|
|
|
std::function<void (handler &, void *const &, const size_t &)> deallocator;
|
2019-03-28 16:56:55 -07:00
|
|
|
bool continuation;
|
2019-03-27 02:49:28 -07:00
|
|
|
|
|
|
|
descriptor(const string_view &name,
|
|
|
|
const decltype(allocator) & = default_allocator,
|
2019-03-28 16:56:55 -07:00
|
|
|
const decltype(deallocator) & = default_deallocator,
|
|
|
|
const bool &continuation = false);
|
2019-03-27 02:49:28 -07:00
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
descriptor(descriptor &&) = delete;
|
|
|
|
descriptor(const descriptor &) = delete;
|
|
|
|
~descriptor() noexcept;
|
|
|
|
};
|
|
|
|
|
2019-03-28 17:09:46 -07:00
|
|
|
struct ircd::ios::descriptor::stats
|
|
|
|
{
|
2019-04-10 22:52:33 -07:00
|
|
|
uint64_t queued {0};
|
2019-03-28 17:09:46 -07: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};
|
|
|
|
|
|
|
|
stats &operator+=(const stats &) &;
|
|
|
|
|
|
|
|
stats();
|
|
|
|
~stats() noexcept;
|
|
|
|
};
|
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
struct ircd::ios::handler
|
|
|
|
{
|
2019-03-27 02:06:55 -07:00
|
|
|
static thread_local handler *current;
|
2019-07-15 16:02:40 -07:00
|
|
|
static thread_local uint64_t epoch;
|
2019-03-27 02:06:55 -07:00
|
|
|
|
2019-03-26 21:22:22 -07:00
|
|
|
static void *allocate(handler *const &, const size_t &);
|
2019-07-15 16:01:35 -07: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-26 16:53:31 -07:00
|
|
|
|
2019-04-16 17:41:13 -07:00
|
|
|
ios::descriptor *descriptor {nullptr};
|
2019-03-26 22:03:48 -07:00
|
|
|
uint64_t slice_start {0};
|
2019-03-26 16:53:31 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
struct ircd::ios::handle
|
|
|
|
:handler
|
|
|
|
{
|
|
|
|
function f;
|
|
|
|
|
|
|
|
template<class... args>
|
|
|
|
void operator()(args&&... a) const;
|
|
|
|
|
2019-04-16 17:41:13 -07:00
|
|
|
handle(ios::descriptor &, function);
|
2019-03-26 16:53:31 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
namespace ircd::ios
|
|
|
|
{
|
2019-03-26 21:22:22 -07: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-28 16:56:55 -07:00
|
|
|
template<class function>
|
|
|
|
bool asio_handler_is_continuation(handle<function> *);
|
|
|
|
|
2019-03-26 16:53:31 -07:00
|
|
|
template<class callable,
|
|
|
|
class function>
|
|
|
|
void asio_handler_invoke(callable &f, handle<function> *);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
ircd::ios::handle<function>::handle(ios::descriptor &d,
|
2019-04-16 17:41:13 -07:00
|
|
|
function f)
|
2019-03-26 16:53:31 -07:00
|
|
|
:handler{&d}
|
2019-04-16 17:41:13 -07:00
|
|
|
,f{std::move(f)}
|
2019-04-10 22:52:33 -07:00
|
|
|
{
|
|
|
|
assert(d.stats);
|
|
|
|
d.stats->queued++;
|
|
|
|
}
|
2019-03-26 16:53:31 -07:00
|
|
|
|
|
|
|
template<class function>
|
|
|
|
template<class... args>
|
|
|
|
void
|
|
|
|
ircd::ios::handle<function>::operator()(args&&... a)
|
|
|
|
const
|
|
|
|
{
|
2019-04-10 22:52:33 -07:00
|
|
|
assert(descriptor && descriptor->stats);
|
|
|
|
assert(descriptor->stats->queued > 0);
|
|
|
|
descriptor->stats->queued--;
|
2019-03-26 16:53:31 -07:00
|
|
|
f(std::forward<args>(a)...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class callable,
|
|
|
|
class function>
|
|
|
|
void
|
|
|
|
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-28 16:56:55 -07:00
|
|
|
template<class function>
|
|
|
|
bool
|
|
|
|
ircd::ios::asio_handler_is_continuation(handle<function> *const h)
|
|
|
|
{
|
|
|
|
return handler::continuation(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class function>
|
|
|
|
void *
|
|
|
|
__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>
|
|
|
|
void
|
|
|
|
ircd::ios::asio_handler_deallocate(void *const ptr,
|
|
|
|
size_t size,
|
|
|
|
handle<function> *const h)
|
|
|
|
{
|
|
|
|
handler::deallocate(h, ptr, size);
|
|
|
|
}
|
|
|
|
|
2019-04-10 15:01:11 -07: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-13 19:37:30 -07:00
|
|
|
inline const uint64_t &
|
|
|
|
__attribute__((always_inline))
|
|
|
|
ircd::ios::epoch()
|
|
|
|
noexcept
|
|
|
|
{
|
|
|
|
return handler::epoch;
|
|
|
|
}
|
|
|
|
|
2017-10-11 17:38:24 -07:00
|
|
|
inline void
|
2019-05-13 23:42:56 -07:00
|
|
|
__attribute__((always_inline))
|
2018-10-17 05:12:10 -07:00
|
|
|
ircd::ios::assert_main_thread()
|
2017-10-11 17:38:24 -07:00
|
|
|
{
|
|
|
|
assert(is_main_thread());
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
2019-05-13 23:42:56 -07:00
|
|
|
__attribute__((always_inline))
|
2018-10-17 05:12:10 -07:00
|
|
|
ircd::ios::is_main_thread()
|
2017-10-11 17:38:24 -07:00
|
|
|
{
|
2019-06-24 03:49:18 -07:00
|
|
|
return std::this_thread::get_id() == main_thread_id ||
|
|
|
|
main_thread_id == std::thread::id();
|
2017-09-29 23:04:41 -07:00
|
|
|
}
|
2018-12-28 13:17:28 -08:00
|
|
|
|
|
|
|
inline bool
|
2019-05-13 23:42:56 -07:00
|
|
|
__attribute__((always_inline))
|
2018-12-28 13:17:28 -08:00
|
|
|
ircd::ios::is_static_thread()
|
|
|
|
{
|
|
|
|
return std::this_thread::get_id() == static_thread_id;
|
|
|
|
}
|