xwords/xwords4/relay/timermgr.cpp
2005-09-02 06:35:25 +00:00

195 lines
4.8 KiB
C++

/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
/*
* Copyright 2005 by Eric House (fixin@peak.org). All rights reserved.
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include <signal.h>
#include <sys/time.h>
#include <errno.h>
#include <assert.h>
#include "timermgr.h"
#include "xwrelay_priv.h"
#include "configs.h"
#include "mlock.h"
TimerMgr::TimerMgr()
: m_nextFireTime(0)
, m_heartbeat(RelayConfigs::GetConfigs()->GetHeartbeatInterval() )
{
pthread_mutex_init( &m_timersMutex, NULL );
}
/* static */TimerMgr*
TimerMgr::getTimerMgr()
{
static TimerMgr* mgr = NULL;
if ( mgr == NULL ) {
mgr = new TimerMgr();
}
return mgr;
}
void
TimerMgr::setTimer( time_t inMillis, TimerProc proc, void* closure,
int interval )
{
logf( "setTimer: now = %d", now() );
TimerInfo ti;
ti.proc = proc;
ti.closure = closure;
ti.when = now() + inMillis;
ti.interval = interval;
MutexLock ml( &m_timersMutex );
if ( getTimer( proc, closure ) ) {
clearTimerImpl( proc, closure );
}
m_timers.push_back( ti );
figureNextFire();
logf( "setTimer done" );
}
time_t
TimerMgr::getPollTimeout()
{
MutexLock ml( &m_timersMutex );
time_t tout = m_nextFireTime;
if ( tout == 0 ) {
tout = m_heartbeat;
} else {
tout -= now();
if ( tout < 0 ) {
tout = 0;
}
}
logf( "getTimeout => %d", tout );
return tout * 1000;
}
int
TimerMgr::getTimer( TimerProc proc, void* closure )
{
list<TimerInfo>::iterator iter;
for ( iter = m_timers.begin(); iter != m_timers.end(); ++iter ) {
if ( (*iter).proc == proc
&& (*iter).closure == closure ) {
return 1;
}
}
return 0;
} /* getTimer */
void
TimerMgr::figureNextFire()
{
/* Don't call this unless have the lock!!! */
time_t t = 0x7FFFFFFF;
time_t cur = now();
list<TimerInfo>::iterator iter = m_timers.begin();
while ( iter != m_timers.end() ) {
time_t when = iter->when;
if ( when == 0 ) {
if ( iter->interval ) {
when = iter->when = cur + iter->interval;
} else {
m_timers.erase(iter);
iter = m_timers.begin();
continue;
}
}
if ( when < t ) {
t = when;
}
++iter;
}
m_nextFireTime = t == 0x7FFFFFFF? 0 : t;
logf( "figureNextFire: m_nextFireTime=%d (now=%d)",
m_nextFireTime, now() );
} /* figureNextFire */
void
TimerMgr::clearTimer( TimerProc proc, void* closure )
{
MutexLock ml( &m_timersMutex );
clearTimerImpl( proc, closure );
}
void
TimerMgr::fireElapsedTimers()
{
logf( "fireElapsedTimers" );
pthread_mutex_lock( &m_timersMutex );
time_t curTime = now();
vector<TimerProc> procs;
vector<void*> closures;
/* loop until we get through without firing a single one. Only fire one
per run, though, since calling erase invalidates the iterator.
PENDING: all this needs a mutex!!!! */
list<TimerInfo>::iterator iter;
for ( iter = m_timers.begin(); iter != m_timers.end(); ++iter ) {
TimerInfo* tip = &(*iter);
if ( tip->when <= curTime ) {
procs.push_back(tip->proc);
closures.push_back(tip->closure);
if ( tip->interval ) {
tip->when += tip->interval;
} else {
tip->when = 0; /* flag for removal */
}
}
}
pthread_mutex_unlock( &m_timersMutex );
vector<TimerProc>::iterator iter1 = procs.begin();
vector<void*>::iterator iter2 = closures.begin();
while ( iter1 != procs.end() ) {
(*iter1++)(*iter2++);
}
MutexLock ml( &m_timersMutex );
figureNextFire();
} /* fireElapsedTimers */
void
TimerMgr::clearTimerImpl( TimerProc proc, void* closure )
{
list<TimerInfo>::iterator iter;
for ( iter = m_timers.begin(); iter != m_timers.end(); ++iter ) {
TimerInfo* tip = &(*iter);
if ( tip->proc == proc && tip->closure == closure ) {
m_timers.erase(iter);
break;
}
}
}