2019-05-24 16:25:32 +02:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2019-02-08 10:05:24 +01:00
|
|
|
|
|
|
|
#ifndef IO_ENGINE_H
|
|
|
|
#define IO_ENGINE_H
|
|
|
|
|
2020-02-13 15:50:42 +01:00
|
|
|
#include "base/exception.hpp"
|
2019-02-08 10:05:24 +01:00
|
|
|
#include "base/lazy-init.hpp"
|
2020-02-13 15:50:42 +01:00
|
|
|
#include "base/logger.hpp"
|
2020-02-05 17:17:41 +01:00
|
|
|
#include "base/shared-object.hpp"
|
2019-02-08 10:05:24 +01:00
|
|
|
#include <atomic>
|
2019-02-15 15:43:58 +01:00
|
|
|
#include <exception>
|
2019-02-08 10:05:24 +01:00
|
|
|
#include <memory>
|
2019-02-15 15:43:58 +01:00
|
|
|
#include <thread>
|
2020-02-05 17:17:41 +01:00
|
|
|
#include <utility>
|
2019-02-15 15:43:58 +01:00
|
|
|
#include <vector>
|
2019-09-06 15:11:55 +02:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <boost/exception/all.hpp>
|
2019-02-08 10:05:24 +01:00
|
|
|
#include <boost/asio/deadline_timer.hpp>
|
2019-09-09 15:11:38 +02:00
|
|
|
#include <boost/asio/io_context.hpp>
|
2019-02-08 10:05:24 +01:00
|
|
|
#include <boost/asio/spawn.hpp>
|
|
|
|
|
2019-02-22 16:16:59 +01:00
|
|
|
namespace icinga
|
|
|
|
{
|
|
|
|
|
2019-02-08 10:05:24 +01:00
|
|
|
/**
|
|
|
|
* Scope lock for CPU-bound work done in an I/O thread
|
|
|
|
*
|
|
|
|
* @ingroup base
|
|
|
|
*/
|
|
|
|
class CpuBoundWork
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CpuBoundWork(boost::asio::yield_context yc);
|
|
|
|
CpuBoundWork(const CpuBoundWork&) = delete;
|
|
|
|
CpuBoundWork(CpuBoundWork&&) = delete;
|
|
|
|
CpuBoundWork& operator=(const CpuBoundWork&) = delete;
|
|
|
|
CpuBoundWork& operator=(CpuBoundWork&&) = delete;
|
|
|
|
~CpuBoundWork();
|
2019-02-14 13:10:04 +01:00
|
|
|
|
|
|
|
void Done();
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool m_Done;
|
2019-02-08 10:05:24 +01:00
|
|
|
};
|
|
|
|
|
2019-02-15 12:25:25 +01:00
|
|
|
/**
|
|
|
|
* Scope break for CPU-bound work done in an I/O thread
|
|
|
|
*
|
|
|
|
* @ingroup base
|
|
|
|
*/
|
|
|
|
class IoBoundWorkSlot
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
IoBoundWorkSlot(boost::asio::yield_context yc);
|
|
|
|
IoBoundWorkSlot(const IoBoundWorkSlot&) = delete;
|
|
|
|
IoBoundWorkSlot(IoBoundWorkSlot&&) = delete;
|
|
|
|
IoBoundWorkSlot& operator=(const IoBoundWorkSlot&) = delete;
|
|
|
|
IoBoundWorkSlot& operator=(IoBoundWorkSlot&&) = delete;
|
|
|
|
~IoBoundWorkSlot();
|
|
|
|
|
|
|
|
private:
|
|
|
|
boost::asio::yield_context yc;
|
|
|
|
};
|
|
|
|
|
2019-02-08 10:05:24 +01:00
|
|
|
/**
|
|
|
|
* Async I/O engine
|
|
|
|
*
|
|
|
|
* @ingroup base
|
|
|
|
*/
|
|
|
|
class IoEngine
|
|
|
|
{
|
|
|
|
friend CpuBoundWork;
|
2019-02-15 12:25:25 +01:00
|
|
|
friend IoBoundWorkSlot;
|
2019-02-08 10:05:24 +01:00
|
|
|
|
|
|
|
public:
|
|
|
|
IoEngine(const IoEngine&) = delete;
|
|
|
|
IoEngine(IoEngine&&) = delete;
|
|
|
|
IoEngine& operator=(const IoEngine&) = delete;
|
|
|
|
IoEngine& operator=(IoEngine&&) = delete;
|
2019-02-15 15:43:58 +01:00
|
|
|
~IoEngine();
|
2019-02-08 10:05:24 +01:00
|
|
|
|
|
|
|
static IoEngine& Get();
|
|
|
|
|
2019-09-09 15:11:38 +02:00
|
|
|
boost::asio::io_context& GetIoContext();
|
2019-02-08 10:05:24 +01:00
|
|
|
|
2019-09-06 15:11:55 +02:00
|
|
|
static inline size_t GetCoroutineStackSize() {
|
|
|
|
#ifdef _WIN32
|
|
|
|
// Increase the stack size for Windows coroutines to prevent exception corruption.
|
|
|
|
// Rationale: Low cost Windows agent only & https://github.com/Icinga/icinga2/issues/7431
|
|
|
|
return 8 * 1024 * 1024;
|
|
|
|
#else /* _WIN32 */
|
2019-12-13 17:20:06 +01:00
|
|
|
// Increase the stack size for Linux/Unix coroutines for many JSON objects on the stack.
|
|
|
|
// This may help mitigate possible stack overflows. https://github.com/Icinga/icinga2/issues/7532
|
|
|
|
return 256 * 1024;
|
|
|
|
//return boost::coroutines::stack_allocator::traits_type::default_size(); // Default 64 KB
|
2019-09-06 15:11:55 +02:00
|
|
|
#endif /* _WIN32 */
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Handler, typename Function>
|
2020-02-05 15:30:03 +01:00
|
|
|
static void SpawnCoroutine(Handler& h, Function f) {
|
2019-09-06 15:11:55 +02:00
|
|
|
|
2020-02-05 15:30:03 +01:00
|
|
|
boost::asio::spawn(h,
|
2019-09-06 15:11:55 +02:00
|
|
|
[f](boost::asio::yield_context yc) {
|
|
|
|
|
|
|
|
try {
|
|
|
|
f(yc);
|
|
|
|
} catch (const boost::coroutines::detail::forced_unwind &) {
|
|
|
|
// Required for proper stack unwinding when coroutines are destroyed.
|
|
|
|
// https://github.com/boostorg/coroutine/issues/39
|
|
|
|
throw;
|
2020-02-13 15:50:42 +01:00
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogCritical, "IoEngine", "Exception in coroutine!");
|
|
|
|
Log(LogDebug, "IoEngine") << "Exception in coroutine: " << DiagnosticInformation(ex);
|
2019-09-06 15:11:55 +02:00
|
|
|
} catch (...) {
|
2020-02-13 15:50:42 +01:00
|
|
|
Log(LogCritical, "IoEngine", "Exception in coroutine!");
|
2019-09-06 15:11:55 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
boost::coroutines::attributes(GetCoroutineStackSize()) // Set a pre-defined stack size.
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-02-09 15:58:44 +01:00
|
|
|
static inline
|
|
|
|
void YieldCurrentCoroutine(boost::asio::yield_context yc)
|
|
|
|
{
|
|
|
|
Get().m_AlreadyExpiredTimer.async_wait(yc);
|
|
|
|
}
|
|
|
|
|
2019-02-08 10:05:24 +01:00
|
|
|
private:
|
|
|
|
IoEngine();
|
|
|
|
|
|
|
|
void RunEventLoop();
|
|
|
|
|
|
|
|
static LazyInit<std::unique_ptr<IoEngine>> m_Instance;
|
|
|
|
|
2019-09-09 15:11:38 +02:00
|
|
|
boost::asio::io_context m_IoContext;
|
|
|
|
boost::asio::executor_work_guard<boost::asio::io_context::executor_type> m_KeepAlive;
|
2019-02-15 15:43:58 +01:00
|
|
|
std::vector<std::thread> m_Threads;
|
2019-02-08 10:05:24 +01:00
|
|
|
boost::asio::deadline_timer m_AlreadyExpiredTimer;
|
2019-02-22 15:07:59 +01:00
|
|
|
std::atomic_int_fast32_t m_CpuBoundSemaphore;
|
2019-02-08 10:05:24 +01:00
|
|
|
};
|
|
|
|
|
2019-02-15 15:43:58 +01:00
|
|
|
class TerminateIoThread : public std::exception
|
|
|
|
{
|
|
|
|
};
|
|
|
|
|
2019-02-22 16:13:28 +01:00
|
|
|
/**
|
|
|
|
* Condition variable which doesn't block I/O threads
|
|
|
|
*
|
|
|
|
* @ingroup base
|
|
|
|
*/
|
|
|
|
class AsioConditionVariable
|
|
|
|
{
|
|
|
|
public:
|
2019-09-09 15:11:38 +02:00
|
|
|
AsioConditionVariable(boost::asio::io_context& io, bool init = false);
|
2019-02-22 16:13:28 +01:00
|
|
|
|
|
|
|
void Set();
|
|
|
|
void Clear();
|
|
|
|
void Wait(boost::asio::yield_context yc);
|
|
|
|
|
|
|
|
private:
|
|
|
|
boost::asio::deadline_timer m_Timer;
|
|
|
|
};
|
|
|
|
|
2020-02-05 17:17:41 +01:00
|
|
|
/**
|
|
|
|
* I/O timeout emulator
|
|
|
|
*
|
|
|
|
* @ingroup base
|
|
|
|
*/
|
|
|
|
class Timeout : public SharedObject
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
DECLARE_PTR_TYPEDEFS(Timeout);
|
|
|
|
|
|
|
|
template<class Executor, class TimeoutFromNow, class OnTimeout>
|
|
|
|
Timeout(boost::asio::io_context& io, Executor& executor, TimeoutFromNow timeoutFromNow, OnTimeout onTimeout)
|
|
|
|
: m_Timer(io)
|
|
|
|
{
|
|
|
|
Ptr keepAlive (this);
|
|
|
|
|
|
|
|
m_Cancelled.store(false);
|
|
|
|
m_Timer.expires_from_now(std::move(timeoutFromNow));
|
|
|
|
|
|
|
|
IoEngine::SpawnCoroutine(executor, [this, keepAlive, onTimeout](boost::asio::yield_context yc) {
|
|
|
|
if (m_Cancelled.load()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
|
|
|
m_Timer.async_wait(yc[ec]);
|
|
|
|
|
|
|
|
if (ec) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_Cancelled.load()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto f (onTimeout);
|
|
|
|
f(std::move(yc));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void Cancel();
|
|
|
|
|
|
|
|
private:
|
|
|
|
boost::asio::deadline_timer m_Timer;
|
|
|
|
std::atomic<bool> m_Cancelled;
|
|
|
|
};
|
|
|
|
|
2019-02-22 16:16:59 +01:00
|
|
|
}
|
|
|
|
|
2019-02-08 10:05:24 +01:00
|
|
|
#endif /* IO_ENGINE_H */
|