emailrelay/src/gnet/gtimer.cpp
Graeme Walker 2911440f23 v0.9.6
2001-12-18 12:00:00 +00:00

231 lines
4.4 KiB
C++

//
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// 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., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// ===
//
// gtimer.cpp
//
#include "gdef.h"
#include "gtimer.h"
#include "gevent.h"
#include "gdebug.h"
namespace GNet
{
class TimerUpdate ;
} ;
// Class: GNet::TimerUpdate
// Description: A private implementation class used by GNet::Timer.
//
class GNet::TimerUpdate
{
public:
TimerUpdate( Timer & , const std::string & ) ;
~TimerUpdate() ;
private:
TimerUpdate( const TimerUpdate & ) ; // not implemented
void operator=( const TimerUpdate & ) ; // not implemented
private:
Timer & m_timer ;
std::string m_type ;
G::DateTime::EpochTime m_soonest ;
} ;
// ===
GNet::TimeoutHandler::~TimeoutHandler()
{
}
// ===
GNet::Timer::Timer( TimeoutHandler & handler ) :
m_time(0UL) ,
m_handler(&handler)
{
TimerUpdate update( *this , "ctor" ) ;
TimerList::instance().add( *this ) ;
}
GNet::Timer::Timer() :
m_time(0UL) ,
m_handler(NULL)
{
TimerUpdate update( *this , "ctor" ) ;
TimerList::instance().add( *this ) ;
}
GNet::Timer::~Timer()
{
try
{
TimerUpdate update( *this , "dtor" ) ;
TimerList::instance().remove( *this ) ;
}
catch(...)
{
}
}
void GNet::Timer::startTimer( unsigned int time )
{
TimerUpdate update( *this , "start" ) ;
m_time = G::DateTime::now() + time ;
}
void GNet::Timer::cancelTimer()
{
TimerUpdate update( *this , "cancel" ) ;
m_time = 0U ;
}
void GNet::Timer::doTimeout()
{
if( m_time != 0U )
{
m_time = 0U ;
G_DEBUG( "GNet::Timer::doTimeout" ) ;
onTimeout() ;
if( m_handler != NULL )
m_handler->onTimeout(*this) ;
}
}
void GNet::Timer::onTimeout()
{
// no-op
}
G::DateTime::EpochTime GNet::Timer::t() const
{
return m_time ;
}
// ===
GNet::TimerList * GNet::TimerList::m_this = NULL ;
GNet::TimerList::TimerList()
{
if( m_this == NULL )
m_this = this ;
}
GNet::TimerList::~TimerList()
{
if( m_this == this )
m_this = NULL ;
}
void GNet::TimerList::add( Timer & t )
{
m_set.insert( &t ) ;
}
void GNet::TimerList::remove( Timer & t )
{
m_set.erase( &t ) ;
}
void GNet::TimerList::update( G::DateTime::EpochTime t_old ,
const std::string & op )
{
G::DateTime::EpochTime t_new = soonest() ;
G_DEBUG( "GNet::TimerList::update: " << op << ": " << t_old << " -> " << t_new ) ;
(void) op.length() ; // pacify the compiler
if( t_old != t_new )
{
EventSources::instance().setTimeout( t_new ) ;
}
}
G::DateTime::EpochTime GNet::TimerList::soonest() const
{
G::DateTime::EpochTime result = 0U ;
const Set::const_iterator end = m_set.end() ;
for( Set::const_iterator p = m_set.begin() ; p != end ; ++p )
{
if( (*p)->t() != 0UL && ( result == 0U || (*p)->t() < result ) )
result = (*p)->t() ;
}
return result ;
}
unsigned int GNet::TimerList::interval( bool & infinite ) const
{
G::DateTime::EpochTime then = soonest() ;
infinite = then == 0U ;
if( infinite )
{
return 0U ;
}
else
{
G::DateTime::EpochTime now = G::DateTime::now() ;
return now >= then ? 0U : (then-now) ;
}
}
GNet::TimerList * GNet::TimerList::instance( const NoThrow & )
{
return m_this ;
}
GNet::TimerList & GNet::TimerList::instance()
{
if( m_this == NULL )
throw NoInstance() ;
return * m_this ;
}
void GNet::TimerList::doTimeouts()
{
G_DEBUG( "GNet::TimerList::doTimeouts" ) ;
G::DateTime::EpochTime now = G::DateTime::now() ;
for( Set::iterator p = m_set.begin() ; p != m_set.end() ; ++p )
{
if( now >= (*p)->t() )
(*p)->doTimeout() ;
}
EventSources::instance().setTimeout( soonest() ) ;
}
// ===
GNet::TimerUpdate::TimerUpdate( Timer & timer , const std::string & type ) :
m_timer(timer) ,
m_type(type)
{
m_soonest = TimerList::instance().soonest() ;
}
GNet::TimerUpdate::~TimerUpdate()
{
try
{
TimerList::instance().update( m_soonest , m_type ) ;
}
catch(...)
{
}
}