// -*- mode: C++; indent-tabs-mode: nil; -*- #ifdef HAVE_CONFIG_H # include "../config.h" #endif // HAVE_CONFIG_H #include "timer.hh" #include "timerqueuemanager.hh" namespace otk { static timeval normalizeTimeval(const timeval &tm) { timeval ret = tm; while (ret.tv_usec < 0) { if (ret.tv_sec > 0) { --ret.tv_sec; ret.tv_usec += 1000000; } else { ret.tv_usec = 0; } } if (ret.tv_usec >= 1000000) { ret.tv_sec += ret.tv_usec / 1000000; ret.tv_usec %= 1000000; } if (ret.tv_sec < 0) ret.tv_sec = 0; return ret; } OBTimer::OBTimer(OBTimerQueueManager *m, OBTimeoutHandler h, OBTimeoutData d) { manager = m; handler = h; data = d; recur = timing = false; } OBTimer::~OBTimer(void) { if (timing) stop(); } void OBTimer::setTimeout(long t) { _timeout.tv_sec = t / 1000; _timeout.tv_usec = t % 1000; _timeout.tv_usec *= 1000; } void OBTimer::setTimeout(const timeval &t) { _timeout.tv_sec = t.tv_sec; _timeout.tv_usec = t.tv_usec; } void OBTimer::start(void) { gettimeofday(&_start, 0); if (! timing) { timing = true; manager->addTimer(this); } } void OBTimer::stop(void) { if (timing) { timing = false; manager->removeTimer(this); } } void OBTimer::fireTimeout(void) { if (handler) handler(data); } timeval OBTimer::timeRemaining(const timeval &tm) const { timeval ret = endpoint(); ret.tv_sec -= tm.tv_sec; ret.tv_usec -= tm.tv_usec; return normalizeTimeval(ret); } timeval OBTimer::endpoint(void) const { timeval ret; ret.tv_sec = _start.tv_sec + _timeout.tv_sec; ret.tv_usec = _start.tv_usec + _timeout.tv_usec; return normalizeTimeval(ret); } bool OBTimer::shouldFire(const timeval &tm) const { timeval end = endpoint(); return ! ((tm.tv_sec < end.tv_sec) || (tm.tv_sec == end.tv_sec && tm.tv_usec < end.tv_usec)); } }