timer.cxx

来自「linux可以运行程序源码」· CXX 代码 · 共 175 行

CXX
175
字号
/* Copyright (C) 2002-2005 RealVNC Ltd.  All Rights Reserved. *  * This 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 software 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 software; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, * USA. */// -=- Timer.cxx#include <stdio.h>#ifdef WIN32#include <windows.h>#ifndef _WIN32_WCE#include <sys/timeb.h>#endif#endif#include <rfb/Timer.h>#include <rfb/util.h>#include <rfb/LogWriter.h>using namespace rfb;#ifndef __NO_DEFINE_VLOG__static LogWriter vlog("Timer");#endif// Win32 does not provide gettimeofday, so we emulate it to simplify the// Timer code.#ifdef _WIN32static void gettimeofday(struct timeval* tv, void*){  LARGE_INTEGER counts, countsPerSec;  static double usecPerCount = 0.0;  if (QueryPerformanceCounter(&counts)) {    if (usecPerCount == 0.0) {      QueryPerformanceFrequency(&countsPerSec);      usecPerCount = 1000000.0 / countsPerSec.QuadPart;    }    LONGLONG usecs = (LONGLONG)(counts.QuadPart * usecPerCount);    tv->tv_usec = (long)(usecs % 1000000);    tv->tv_sec = (long)(usecs / 1000000);  } else {#ifndef _WIN32_WCE    struct timeb tb;    ftime(&tb);    tv->tv_sec = tb.time;    tv->tv_usec = tb.millitm * 1000;#else    throw SystemException("QueryPerformanceCounter", GetLastError());#endif  }}#endif// Millisecond timeout processing helper functionsinline static timeval addMillis(timeval inTime, int millis) {  int secs = millis / 1000;  millis = millis % 1000;  inTime.tv_sec += secs;  inTime.tv_usec += millis * 1000;  if (inTime.tv_usec >= 1000000) {    inTime.tv_sec++;    inTime.tv_usec -= 1000000;  }  return inTime;}inline static int diffTimeMillis(timeval later, timeval earlier) {  return ((later.tv_sec - earlier.tv_sec) * 1000) + ((later.tv_usec - earlier.tv_usec) / 1000);}std::list<Timer*> Timer::pending;int Timer::checkTimeouts() {  if (pending.empty())    return 0;  timeval now;  gettimeofday(&now, 0);  while (pending.front()->isBefore(now)) {    Timer* timer = pending.front();    pending.pop_front();    vlog.debug("handleTimeout(%p)", timer);    if (timer->cb->handleTimeout(timer)) {      timer->dueTime = addMillis(timer->dueTime, timer->timeoutMs);      if (timer->isBefore(now)) {        // Time has jumped forwards!	      vlog.info("time has moved forwards!");        timer->dueTime = addMillis(now, timer->timeoutMs);      }      insertTimer(timer);    } else if (pending.empty()) {      return 0;    }  }  return getNextTimeout();}int Timer::getNextTimeout() {  timeval now;  gettimeofday(&now, 0);  int toWait = __rfbmax(1, diffTimeMillis(pending.front()->dueTime, now));  if (toWait > pending.front()->timeoutMs) {    if (toWait - pending.front()->timeoutMs < 1000) {      vlog.info("gettimeofday is broken...");      return toWait;    }    // Time has jumped backwards!    vlog.info("time has moved backwards!");    pending.front()->dueTime = now;    toWait = 1;  }  return toWait;}void Timer::insertTimer(Timer* t) {  std::list<Timer*>::iterator i;  for (i=pending.begin(); i!=pending.end(); i++) {    if (t->isBefore((*i)->dueTime)) {      pending.insert(i, t);      return;    }  }  pending.push_back(t);}void Timer::start(int timeoutMs_) {  timeval now;  gettimeofday(&now, 0);  stop();  timeoutMs = timeoutMs_;  dueTime = addMillis(now, timeoutMs);  insertTimer(this);}void Timer::stop() {  pending.remove(this);}bool Timer::isStarted() {  std::list<Timer*>::iterator i;  for (i=pending.begin(); i!=pending.end(); i++) {    if (*i == this)      return true;  }  return false;}int Timer::getTimeoutMs() {  return timeoutMs;}bool Timer::isBefore(timeval other) {  return (dueTime.tv_sec < other.tv_sec) ||    ((dueTime.tv_sec == other.tv_sec) &&     (dueTime.tv_usec < other.tv_usec));}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?