2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2012-05-10 12:06:41 +02:00
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/timer.hpp"
|
|
|
|
#include "base/debug.hpp"
|
2018-07-19 13:34:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/utility.hpp"
|
2013-03-18 11:02:18 +01:00
|
|
|
#include <boost/thread/mutex.hpp>
|
|
|
|
#include <boost/thread/condition_variable.hpp>
|
|
|
|
#include <boost/multi_index_container.hpp>
|
|
|
|
#include <boost/multi_index/ordered_index.hpp>
|
|
|
|
#include <boost/multi_index/key_extractors.hpp>
|
2017-11-21 12:12:58 +01:00
|
|
|
#include <thread>
|
2012-03-28 13:24:49 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2017-11-21 13:03:51 +01:00
|
|
|
namespace icinga {
|
|
|
|
|
|
|
|
class TimerHolder {
|
|
|
|
public:
|
|
|
|
TimerHolder(Timer *timer)
|
|
|
|
: m_Timer(timer)
|
|
|
|
{ }
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
inline Timer *GetObject() const
|
2017-11-21 13:03:51 +01:00
|
|
|
{
|
|
|
|
return m_Timer;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
inline double GetNextUnlocked() const
|
2017-11-21 13:03:51 +01:00
|
|
|
{
|
|
|
|
return m_Timer->m_Next;
|
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
operator Timer *() const
|
2017-11-21 13:03:51 +01:00
|
|
|
{
|
|
|
|
return m_Timer;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Timer *m_Timer;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-03-18 11:02:18 +01:00
|
|
|
typedef boost::multi_index_container<
|
2017-11-21 13:03:51 +01:00
|
|
|
TimerHolder,
|
2013-03-18 11:02:18 +01:00
|
|
|
boost::multi_index::indexed_by<
|
2017-11-21 13:03:51 +01:00
|
|
|
boost::multi_index::ordered_unique<boost::multi_index::const_mem_fun<TimerHolder, Timer *, &TimerHolder::GetObject> >,
|
|
|
|
boost::multi_index::ordered_non_unique<boost::multi_index::const_mem_fun<TimerHolder, double, &TimerHolder::GetNextUnlocked> >
|
2013-03-18 11:02:18 +01:00
|
|
|
>
|
|
|
|
> TimerSet;
|
|
|
|
|
2014-12-05 15:55:20 +01:00
|
|
|
static boost::mutex l_TimerMutex;
|
|
|
|
static boost::condition_variable l_TimerCV;
|
2017-11-21 12:12:58 +01:00
|
|
|
static std::thread l_TimerThread;
|
2014-12-05 15:55:20 +01:00
|
|
|
static bool l_StopTimerThread;
|
2013-03-18 11:02:18 +01:00
|
|
|
static TimerSet l_Timers;
|
2018-07-20 15:36:15 +02:00
|
|
|
static int l_AliveTimers = 0;
|
2012-03-28 13:24:49 +02:00
|
|
|
|
2014-05-28 13:46:39 +02:00
|
|
|
/**
|
|
|
|
* Destructor for the Timer class.
|
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
Timer::~Timer()
|
2014-05-28 13:46:39 +02:00
|
|
|
{
|
2015-03-01 22:25:14 +01:00
|
|
|
Stop(true);
|
2014-05-28 13:46:39 +02:00
|
|
|
}
|
|
|
|
|
2018-07-19 12:49:27 +02:00
|
|
|
void Timer::Initialize()
|
|
|
|
{
|
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2018-07-20 15:36:15 +02:00
|
|
|
|
|
|
|
if (l_AliveTimers > 0) {
|
|
|
|
InitializeThread();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timer::Uninitialize()
|
|
|
|
{
|
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
|
|
|
|
|
|
|
if (l_AliveTimers > 0) {
|
|
|
|
UninitializeThread();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timer::InitializeThread()
|
|
|
|
{
|
2018-07-19 12:49:27 +02:00
|
|
|
l_StopTimerThread = false;
|
|
|
|
l_TimerThread = std::thread(&Timer::TimerThreadProc);
|
|
|
|
}
|
|
|
|
|
2018-07-20 15:36:15 +02:00
|
|
|
void Timer::UninitializeThread()
|
2013-02-19 23:02:08 +01:00
|
|
|
{
|
2017-12-19 15:50:05 +01:00
|
|
|
{
|
|
|
|
l_StopTimerThread = true;
|
|
|
|
l_TimerCV.notify_all();
|
|
|
|
}
|
|
|
|
|
2018-07-20 15:36:15 +02:00
|
|
|
l_TimerMutex.unlock();
|
|
|
|
|
2017-12-19 15:50:05 +01:00
|
|
|
if (l_TimerThread.joinable())
|
|
|
|
l_TimerThread.join();
|
2018-07-20 15:36:15 +02:00
|
|
|
|
|
|
|
l_TimerMutex.lock();
|
2012-03-28 13:24:49 +02:00
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
2013-02-17 19:14:34 +01:00
|
|
|
* Calls this timer.
|
2012-05-08 15:14:20 +02:00
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
void Timer::Call()
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2013-03-19 16:12:28 +01:00
|
|
|
try {
|
2019-04-17 22:31:42 +02:00
|
|
|
OnTimerExpired(this);
|
2013-03-19 16:12:28 +01:00
|
|
|
} catch (...) {
|
2015-02-28 08:43:49 +01:00
|
|
|
InternalReschedule(true);
|
2013-03-19 16:12:28 +01:00
|
|
|
|
|
|
|
throw;
|
|
|
|
}
|
2012-06-18 17:23:48 +02:00
|
|
|
|
2015-02-28 08:43:49 +01:00
|
|
|
InternalReschedule(true);
|
2012-03-28 13:24:49 +02:00
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
|
|
|
* Sets the interval for this timer.
|
|
|
|
*
|
|
|
|
* @param interval The new interval.
|
|
|
|
*/
|
2012-07-25 12:59:17 +02:00
|
|
|
void Timer::SetInterval(double interval)
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2012-03-28 13:24:49 +02:00
|
|
|
m_Interval = interval;
|
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
|
|
|
* Retrieves the interval for this timer.
|
|
|
|
*
|
|
|
|
* @returns The interval.
|
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
double Timer::GetInterval() const
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2012-03-28 13:24:49 +02:00
|
|
|
return m_Interval;
|
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
|
|
|
* Registers the timer and starts processing events for it.
|
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
void Timer::Start()
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2013-03-01 12:07:52 +01:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-03-01 12:07:52 +01:00
|
|
|
m_Started = true;
|
2018-07-20 15:36:15 +02:00
|
|
|
|
|
|
|
if (++l_AliveTimers == 1) {
|
|
|
|
InitializeThread();
|
|
|
|
}
|
2013-03-01 12:07:52 +01:00
|
|
|
}
|
2013-02-18 14:40:24 +01:00
|
|
|
|
2015-02-28 08:43:49 +01:00
|
|
|
InternalReschedule(false);
|
2012-03-28 13:24:49 +02:00
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
|
|
|
* Unregisters the timer and stops processing events for it.
|
|
|
|
*/
|
2015-02-28 08:43:49 +01:00
|
|
|
void Timer::Stop(bool wait)
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
if (l_StopTimerThread)
|
2014-11-12 09:33:13 +01:00
|
|
|
return;
|
|
|
|
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-02-18 14:40:24 +01:00
|
|
|
|
2018-07-20 15:36:15 +02:00
|
|
|
if (m_Started && --l_AliveTimers == 0) {
|
|
|
|
UninitializeThread();
|
|
|
|
}
|
|
|
|
|
2013-02-18 14:40:24 +01:00
|
|
|
m_Started = false;
|
2014-05-28 13:46:39 +02:00
|
|
|
l_Timers.erase(this);
|
2012-07-10 13:00:53 +02:00
|
|
|
|
2013-02-17 19:14:34 +01:00
|
|
|
/* Notify the worker thread that we've disabled a timer. */
|
2014-12-05 15:55:20 +01:00
|
|
|
l_TimerCV.notify_all();
|
2015-02-28 08:43:49 +01:00
|
|
|
|
|
|
|
while (wait && m_Running)
|
|
|
|
l_TimerCV.wait(lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timer::Reschedule(double next)
|
|
|
|
{
|
|
|
|
InternalReschedule(false, next);
|
2012-03-28 13:24:49 +02:00
|
|
|
}
|
|
|
|
|
2012-05-08 15:14:20 +02:00
|
|
|
/**
|
|
|
|
* Reschedules this timer.
|
|
|
|
*
|
2015-02-28 08:43:49 +01:00
|
|
|
* @param completed Whether the timer has just completed its callback.
|
2013-02-17 19:14:34 +01:00
|
|
|
* @param next The time when this timer should be called again. Use -1 to let
|
2017-12-19 15:50:05 +01:00
|
|
|
* the timer figure out a suitable time based on the interval.
|
2012-05-08 15:14:20 +02:00
|
|
|
*/
|
2015-02-28 08:43:49 +01:00
|
|
|
void Timer::InternalReschedule(bool completed, double next)
|
2012-03-28 13:24:49 +02:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2015-02-28 08:43:49 +01:00
|
|
|
if (completed)
|
|
|
|
m_Running = false;
|
|
|
|
|
2013-03-20 11:11:46 +01:00
|
|
|
if (next < 0) {
|
|
|
|
/* Don't schedule the next call if this is not a periodic timer. */
|
|
|
|
if (m_Interval <= 0)
|
|
|
|
return;
|
|
|
|
|
2013-03-06 13:01:51 +01:00
|
|
|
next = Utility::GetTime() + m_Interval;
|
2013-03-20 11:11:46 +01:00
|
|
|
}
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2012-03-28 13:24:49 +02:00
|
|
|
m_Next = next;
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2015-02-27 16:44:33 +01:00
|
|
|
if (m_Started && !m_Running) {
|
2013-02-18 14:40:24 +01:00
|
|
|
/* Remove and re-add the timer to update the index. */
|
2014-05-28 13:46:39 +02:00
|
|
|
l_Timers.erase(this);
|
|
|
|
l_Timers.insert(this);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2013-02-18 14:40:24 +01:00
|
|
|
/* Notify the worker that we've rescheduled a timer. */
|
2014-12-05 15:55:20 +01:00
|
|
|
l_TimerCV.notify_all();
|
2013-02-18 14:40:24 +01:00
|
|
|
}
|
2013-02-17 19:14:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieves when the timer is next due.
|
|
|
|
*
|
|
|
|
* @returns The timestamp.
|
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
double Timer::GetNext() const
|
2013-02-17 19:14:34 +01:00
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-02-17 19:14:34 +01:00
|
|
|
return m_Next;
|
2012-03-28 13:24:49 +02:00
|
|
|
}
|
2012-09-25 14:03:41 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Adjusts all timers by adding the specified amount of time to their
|
|
|
|
* next scheduled timestamp.
|
|
|
|
*
|
|
|
|
* @param adjustment The adjustment.
|
|
|
|
*/
|
|
|
|
void Timer::AdjustTimers(double adjustment)
|
|
|
|
{
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2012-09-25 15:24:14 +02:00
|
|
|
double now = Utility::GetTime();
|
|
|
|
|
2013-03-16 21:18:53 +01:00
|
|
|
typedef boost::multi_index::nth_index<TimerSet, 1>::type TimerView;
|
2013-03-18 11:02:18 +01:00
|
|
|
TimerView& idx = boost::get<1>(l_Timers);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2014-05-28 13:46:39 +02:00
|
|
|
std::vector<Timer *> timers;
|
2014-03-17 08:41:53 +01:00
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (Timer *timer : idx) {
|
2015-01-12 12:36:49 +01:00
|
|
|
if (std::fabs(now - (timer->m_Next + adjustment)) <
|
2017-12-19 15:50:05 +01:00
|
|
|
std::fabs(now - timer->m_Next)) {
|
2012-09-25 15:24:14 +02:00
|
|
|
timer->m_Next += adjustment;
|
2014-04-20 07:21:38 +02:00
|
|
|
timers.push_back(timer);
|
2013-03-18 11:15:46 +01:00
|
|
|
}
|
2013-02-17 19:14:34 +01:00
|
|
|
}
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (Timer *timer : timers) {
|
2014-04-20 07:21:38 +02:00
|
|
|
l_Timers.erase(timer);
|
2014-05-28 13:46:39 +02:00
|
|
|
l_Timers.insert(timer);
|
2014-04-20 07:21:38 +02:00
|
|
|
}
|
2014-04-20 11:22:15 +02:00
|
|
|
|
2013-02-17 19:14:34 +01:00
|
|
|
/* Notify the worker that we've rescheduled some timers. */
|
2014-12-05 15:55:20 +01:00
|
|
|
l_TimerCV.notify_all();
|
2013-02-17 19:14:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Worker thread proc for Timer objects.
|
|
|
|
*/
|
2018-01-04 04:25:35 +01:00
|
|
|
void Timer::TimerThreadProc()
|
2013-02-17 19:14:34 +01:00
|
|
|
{
|
2018-07-19 13:34:12 +02:00
|
|
|
Log(LogDebug, "Timer", "TimerThreadProc started.");
|
|
|
|
|
2013-08-30 10:19:32 +02:00
|
|
|
Utility::SetThreadName("Timer Thread");
|
|
|
|
|
2013-02-17 19:14:34 +01:00
|
|
|
for (;;) {
|
2014-12-05 15:55:20 +01:00
|
|
|
boost::mutex::scoped_lock lock(l_TimerMutex);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2013-03-16 21:18:53 +01:00
|
|
|
typedef boost::multi_index::nth_index<TimerSet, 1>::type NextTimerView;
|
2013-03-18 11:02:18 +01:00
|
|
|
NextTimerView& idx = boost::get<1>(l_Timers);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
|
|
|
/* Wait until there is at least one timer. */
|
2014-12-05 15:55:20 +01:00
|
|
|
while (idx.empty() && !l_StopTimerThread)
|
|
|
|
l_TimerCV.wait(lock);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2014-12-05 15:55:20 +01:00
|
|
|
if (l_StopTimerThread)
|
2013-02-19 23:02:08 +01:00
|
|
|
break;
|
|
|
|
|
2016-08-27 19:56:12 +02:00
|
|
|
auto it = idx.begin();
|
2014-05-28 13:46:39 +02:00
|
|
|
Timer *timer = *it;
|
2013-02-17 19:14:34 +01:00
|
|
|
|
|
|
|
double wait = timer->m_Next - Utility::GetTime();
|
|
|
|
|
2014-05-03 20:02:22 +02:00
|
|
|
if (wait > 0.01) {
|
2013-02-17 19:14:34 +01:00
|
|
|
/* Wait for the next timer. */
|
2018-04-15 06:02:42 +02:00
|
|
|
l_TimerCV.timed_wait(lock, boost::posix_time::milliseconds(long(wait * 1000)));
|
2013-02-17 19:14:34 +01:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove the timer from the list so it doesn't get called again
|
|
|
|
* until the current call is completed. */
|
2013-03-18 11:02:18 +01:00
|
|
|
l_Timers.erase(timer);
|
2013-02-17 19:14:34 +01:00
|
|
|
|
2015-02-27 14:07:12 +01:00
|
|
|
timer->m_Running = true;
|
|
|
|
|
2013-02-26 10:13:54 +01:00
|
|
|
lock.unlock();
|
|
|
|
|
2013-02-17 19:14:34 +01:00
|
|
|
/* Asynchronously call the timer. */
|
2019-03-11 11:17:26 +01:00
|
|
|
Utility::QueueAsyncCallback([timer]() { timer->Call(); });
|
2012-09-25 14:03:41 +02:00
|
|
|
}
|
|
|
|
}
|