2019-02-08 10:05:24 +01:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
|
|
|
* Copyright (C) 2012-2018 Icinga Development Team (https://icinga.com/) *
|
|
|
|
* *
|
|
|
|
* This program is free software; you can redistribute it and/or *
|
|
|
|
* modify it under the terms of the GNU General Public License *
|
|
|
|
* as published by the Free Software Foundation; either version 2 *
|
|
|
|
* of the License, or (at your option) any later version. *
|
|
|
|
* *
|
|
|
|
* This program is distributed in the hope that it will be useful, *
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
|
|
* GNU General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU General Public License *
|
|
|
|
* along with this program; if not, write to the Free Software Foundation *
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
#include "base/exception.hpp"
|
|
|
|
#include "base/io-engine.hpp"
|
|
|
|
#include "base/lazy-init.hpp"
|
|
|
|
#include "base/logger.hpp"
|
|
|
|
#include <exception>
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
|
|
|
#include <boost/asio/io_service.hpp>
|
|
|
|
#include <boost/asio/spawn.hpp>
|
|
|
|
#include <boost/date_time/posix_time/ptime.hpp>
|
2019-02-22 16:13:28 +01:00
|
|
|
#include <boost/system/error_code.hpp>
|
2019-02-08 10:05:24 +01:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
|
|
|
CpuBoundWork::CpuBoundWork(boost::asio::yield_context yc)
|
2019-02-14 13:10:04 +01:00
|
|
|
: m_Done(false)
|
2019-02-08 10:05:24 +01:00
|
|
|
{
|
|
|
|
auto& ioEngine (IoEngine::Get());
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
auto availableSlots (ioEngine.m_CpuBoundSemaphore.fetch_sub(1));
|
|
|
|
|
|
|
|
if (availableSlots < 1) {
|
|
|
|
ioEngine.m_CpuBoundSemaphore.fetch_add(1);
|
|
|
|
ioEngine.m_AlreadyExpiredTimer.async_wait(yc);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CpuBoundWork::~CpuBoundWork()
|
|
|
|
{
|
2019-02-14 13:10:04 +01:00
|
|
|
if (!m_Done) {
|
|
|
|
IoEngine::Get().m_CpuBoundSemaphore.fetch_add(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CpuBoundWork::Done()
|
|
|
|
{
|
|
|
|
if (!m_Done) {
|
|
|
|
IoEngine::Get().m_CpuBoundSemaphore.fetch_add(1);
|
|
|
|
|
|
|
|
m_Done = true;
|
|
|
|
}
|
2019-02-08 10:05:24 +01:00
|
|
|
}
|
|
|
|
|
2019-02-15 12:25:25 +01:00
|
|
|
IoBoundWorkSlot::IoBoundWorkSlot(boost::asio::yield_context yc)
|
|
|
|
: yc(yc)
|
|
|
|
{
|
|
|
|
IoEngine::Get().m_CpuBoundSemaphore.fetch_add(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
IoBoundWorkSlot::~IoBoundWorkSlot()
|
|
|
|
{
|
|
|
|
auto& ioEngine (IoEngine::Get());
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
auto availableSlots (ioEngine.m_CpuBoundSemaphore.fetch_sub(1));
|
|
|
|
|
|
|
|
if (availableSlots < 1) {
|
|
|
|
ioEngine.m_CpuBoundSemaphore.fetch_add(1);
|
|
|
|
ioEngine.m_AlreadyExpiredTimer.async_wait(yc);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-08 10:05:24 +01:00
|
|
|
LazyInit<std::unique_ptr<IoEngine>> IoEngine::m_Instance ([]() { return std::unique_ptr<IoEngine>(new IoEngine()); });
|
|
|
|
|
|
|
|
IoEngine& IoEngine::Get()
|
|
|
|
{
|
|
|
|
return *m_Instance.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
boost::asio::io_service& IoEngine::GetIoService()
|
|
|
|
{
|
|
|
|
return m_IoService;
|
|
|
|
}
|
|
|
|
|
2019-02-21 10:08:38 +01:00
|
|
|
IoEngine::IoEngine() : m_IoService(), m_KeepAlive(m_IoService), m_Threads(decltype(m_Threads)::size_type(std::thread::hardware_concurrency() * 2u)), m_AlreadyExpiredTimer(m_IoService)
|
2019-02-08 10:05:24 +01:00
|
|
|
{
|
|
|
|
m_AlreadyExpiredTimer.expires_at(boost::posix_time::neg_infin);
|
2019-02-21 10:08:38 +01:00
|
|
|
m_CpuBoundSemaphore.store(std::thread::hardware_concurrency() * 3u / 2u);
|
2019-02-08 10:05:24 +01:00
|
|
|
|
2019-02-15 15:43:58 +01:00
|
|
|
for (auto& thread : m_Threads) {
|
|
|
|
thread = std::thread(&IoEngine::RunEventLoop, this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IoEngine::~IoEngine()
|
|
|
|
{
|
|
|
|
for (auto& thread : m_Threads) {
|
|
|
|
m_IoService.post([]() {
|
|
|
|
throw TerminateIoThread();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& thread : m_Threads) {
|
|
|
|
thread.join();
|
2019-02-08 10:05:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IoEngine::RunEventLoop()
|
|
|
|
{
|
|
|
|
for (;;) {
|
|
|
|
try {
|
|
|
|
m_IoService.run();
|
|
|
|
|
2019-02-15 15:43:58 +01:00
|
|
|
break;
|
|
|
|
} catch (const TerminateIoThread&) {
|
2019-02-08 10:05:24 +01:00
|
|
|
break;
|
|
|
|
} catch (const std::exception& e) {
|
|
|
|
Log(LogCritical, "IoEngine", "Exception during I/O operation!");
|
|
|
|
Log(LogDebug, "IoEngine") << "Exception during I/O operation: " << DiagnosticInformation(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-02-22 16:13:28 +01:00
|
|
|
|
|
|
|
AsioConditionVariable::AsioConditionVariable(boost::asio::io_service& io, bool init)
|
|
|
|
: m_Timer(io)
|
|
|
|
{
|
|
|
|
m_Timer.expires_at(init ? boost::posix_time::neg_infin : boost::posix_time::pos_infin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsioConditionVariable::Set()
|
|
|
|
{
|
|
|
|
m_Timer.expires_at(boost::posix_time::neg_infin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsioConditionVariable::Clear()
|
|
|
|
{
|
|
|
|
m_Timer.expires_at(boost::posix_time::pos_infin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsioConditionVariable::Wait(boost::asio::yield_context yc)
|
|
|
|
{
|
|
|
|
boost::system::error_code ec;
|
|
|
|
m_Timer.async_wait(yc[ec]);
|
|
|
|
}
|