mirror of https://github.com/Icinga/icinga2.git
308 lines
7.0 KiB
C++
308 lines
7.0 KiB
C++
/******************************************************************************
|
|
* Icinga 2 *
|
|
* Copyright (C) 2012 Icinga Development Team (http://www.icinga.org/) *
|
|
* *
|
|
* 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/timer.h"
|
|
#include "base/application.h"
|
|
#include "base/debug.h"
|
|
#include "base/utility.h"
|
|
#include "base/logger_fwd.h"
|
|
#include <boost/bind.hpp>
|
|
#include <boost/thread/thread.hpp>
|
|
#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>
|
|
|
|
using namespace icinga;
|
|
|
|
/**
|
|
* @ingroup base
|
|
*/
|
|
struct icinga::TimerNextExtractor
|
|
{
|
|
typedef double result_type;
|
|
|
|
/**
|
|
* Extracts the next timestamp from a Timer.
|
|
*
|
|
* Note: Caller must hold l_Mutex.
|
|
*
|
|
* @param wtimer Weak pointer to the timer.
|
|
* @returns The next timestamp
|
|
*/
|
|
double operator()(const weak_ptr<Timer>& wtimer)
|
|
{
|
|
Timer::Ptr timer = wtimer.lock();
|
|
|
|
if (!timer)
|
|
return 0;
|
|
|
|
return timer->m_Next;
|
|
}
|
|
};
|
|
|
|
typedef boost::multi_index_container<
|
|
Timer::WeakPtr,
|
|
boost::multi_index::indexed_by<
|
|
boost::multi_index::ordered_unique<boost::multi_index::identity<Timer::WeakPtr> >,
|
|
boost::multi_index::ordered_non_unique<TimerNextExtractor>
|
|
>
|
|
> TimerSet;
|
|
|
|
static boost::mutex l_Mutex;
|
|
static boost::condition_variable l_CV;
|
|
static boost::thread l_Thread;
|
|
static bool l_StopThread;
|
|
static TimerSet l_Timers;
|
|
|
|
/**
|
|
* Constructor for the Timer class.
|
|
*/
|
|
Timer::Timer(void)
|
|
: m_Interval(0), m_Next(0)
|
|
{ }
|
|
|
|
/**
|
|
* Initializes the timer sub-system.
|
|
*/
|
|
void Timer::Initialize(void)
|
|
{
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
l_StopThread = false;
|
|
l_Thread = boost::thread(boost::bind(&Timer::TimerThreadProc));
|
|
}
|
|
|
|
/**
|
|
* Disables the timer sub-system.
|
|
*/
|
|
void Timer::Uninitialize(void)
|
|
{
|
|
{
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
l_StopThread = true;
|
|
l_CV.notify_all();
|
|
}
|
|
|
|
l_Thread.join();
|
|
}
|
|
|
|
/**
|
|
* Calls this timer.
|
|
*/
|
|
void Timer::Call(void)
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
Timer::Ptr self = GetSelf();
|
|
|
|
try {
|
|
OnTimerExpired(self);
|
|
} catch (...) {
|
|
Reschedule();
|
|
|
|
throw;
|
|
}
|
|
|
|
Reschedule();
|
|
}
|
|
|
|
/**
|
|
* Sets the interval for this timer.
|
|
*
|
|
* @param interval The new interval.
|
|
*/
|
|
void Timer::SetInterval(double interval)
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
m_Interval = interval;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the interval for this timer.
|
|
*
|
|
* @returns The interval.
|
|
*/
|
|
double Timer::GetInterval(void) const
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
return m_Interval;
|
|
}
|
|
|
|
/**
|
|
* Registers the timer and starts processing events for it.
|
|
*/
|
|
void Timer::Start(void)
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
{
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
m_Started = true;
|
|
}
|
|
|
|
Reschedule();
|
|
}
|
|
|
|
/**
|
|
* Unregisters the timer and stops processing events for it.
|
|
*/
|
|
void Timer::Stop(void)
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
|
|
m_Started = false;
|
|
l_Timers.erase(GetSelf());
|
|
|
|
/* Notify the worker thread that we've disabled a timer. */
|
|
l_CV.notify_all();
|
|
}
|
|
|
|
/**
|
|
* Reschedules this timer.
|
|
*
|
|
* @param next The time when this timer should be called again. Use -1 to let
|
|
* the timer figure out a suitable time based on the interval.
|
|
*/
|
|
void Timer::Reschedule(double next)
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
|
|
if (next < 0) {
|
|
/* Don't schedule the next call if this is not a periodic timer. */
|
|
if (m_Interval <= 0)
|
|
return;
|
|
|
|
next = Utility::GetTime() + m_Interval;
|
|
}
|
|
|
|
m_Next = next;
|
|
|
|
if (m_Started) {
|
|
/* Remove and re-add the timer to update the index. */
|
|
l_Timers.erase(GetSelf());
|
|
l_Timers.insert(GetSelf());
|
|
|
|
/* Notify the worker that we've rescheduled a timer. */
|
|
l_CV.notify_all();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieves when the timer is next due.
|
|
*
|
|
* @returns The timestamp.
|
|
*/
|
|
double Timer::GetNext(void) const
|
|
{
|
|
ASSERT(!OwnsLock());
|
|
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
return m_Next;
|
|
}
|
|
|
|
/**
|
|
* Adjusts all timers by adding the specified amount of time to their
|
|
* next scheduled timestamp.
|
|
*
|
|
* @param adjustment The adjustment.
|
|
*/
|
|
void Timer::AdjustTimers(double adjustment)
|
|
{
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
|
|
double now = Utility::GetTime();
|
|
|
|
typedef boost::multi_index::nth_index<TimerSet, 1>::type TimerView;
|
|
TimerView& idx = boost::get<1>(l_Timers);
|
|
|
|
TimerView::iterator it;
|
|
for (it = idx.begin(); it != idx.end(); it++) {
|
|
Timer::Ptr timer = it->lock();
|
|
|
|
if (abs(now - (timer->m_Next + adjustment)) <
|
|
abs(now - timer->m_Next)) {
|
|
timer->m_Next += adjustment;
|
|
l_Timers.erase(timer);
|
|
l_Timers.insert(timer);
|
|
}
|
|
}
|
|
|
|
/* Notify the worker that we've rescheduled some timers. */
|
|
l_CV.notify_all();
|
|
}
|
|
|
|
/**
|
|
* Worker thread proc for Timer objects.
|
|
*/
|
|
void Timer::TimerThreadProc(void)
|
|
{
|
|
for (;;) {
|
|
boost::mutex::scoped_lock lock(l_Mutex);
|
|
|
|
typedef boost::multi_index::nth_index<TimerSet, 1>::type NextTimerView;
|
|
NextTimerView& idx = boost::get<1>(l_Timers);
|
|
|
|
/* Wait until there is at least one timer. */
|
|
while (idx.empty() && !l_StopThread)
|
|
l_CV.wait(lock);
|
|
|
|
if (l_StopThread)
|
|
break;
|
|
|
|
NextTimerView::iterator it = idx.begin();
|
|
Timer::Ptr timer = it->lock();
|
|
|
|
if (!timer) {
|
|
/* Remove the timer from the list if it's not alive anymore. */
|
|
idx.erase(it);
|
|
continue;
|
|
}
|
|
|
|
double wait = timer->m_Next - Utility::GetTime();
|
|
|
|
if (wait > 0) {
|
|
/* Make sure the timer we just examined can be destroyed while we're waiting. */
|
|
timer.reset();
|
|
|
|
/* Wait for the next timer. */
|
|
l_CV.timed_wait(lock, boost::posix_time::milliseconds(wait * 1000));
|
|
|
|
continue;
|
|
}
|
|
|
|
/* Remove the timer from the list so it doesn't get called again
|
|
* until the current call is completed. */
|
|
l_Timers.erase(timer);
|
|
|
|
lock.unlock();
|
|
|
|
/* Asynchronously call the timer. */
|
|
Utility::QueueAsyncCallback(boost::bind(&Timer::Call, timer));
|
|
}
|
|
}
|