/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/ ** ** This file is part of the QtCore module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this ** file. Please review the following information to ensure the GNU Lesser ** General Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU General ** Public License version 3.0 as published by the Free Software Foundation ** and appearing in the file LICENSE.GPL included in the packaging of this ** file. Please review the following information to ensure the GNU General ** Public License version 3.0 requirements will be met: ** http://www.gnu.org/copyleft/gpl.html. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qelapsedtimer.h" #include typedef ULONGLONG (WINAPI *PtrGetTickCount64)(void); static PtrGetTickCount64 ptrGetTickCount64 = 0; QT_BEGIN_NAMESPACE // Result of QueryPerformanceFrequency, 0 indicates that the high resolution timer is unavailable static quint64 counterFrequency = 0; static void resolveLibs() { static bool done = false; if (done) return; // try to get GetTickCount64 from the system HMODULE kernel32 = GetModuleHandleW(L"kernel32"); if (!kernel32) return; #if defined(Q_OS_WINCE) // does this function exist on WinCE, or will ever exist? ptrGetTickCount64 = (PtrGetTickCount64)GetProcAddress(kernel32, L"GetTickCount64"); #else ptrGetTickCount64 = (PtrGetTickCount64)GetProcAddress(kernel32, "GetTickCount64"); #endif // Retrieve the number of high-resolution performance counter ticks per second LARGE_INTEGER frequency; if (!QueryPerformanceFrequency(&frequency)) { counterFrequency = 0; } else { counterFrequency = frequency.QuadPart; } done = true; } static inline qint64 ticksToNanoseconds(qint64 ticks) { if (counterFrequency > 0) { // QueryPerformanceCounter uses an arbitrary frequency qint64 seconds = ticks / counterFrequency; qint64 nanoSeconds = (ticks - seconds * counterFrequency) * 1000000000 / counterFrequency; return seconds * 1000000000 + nanoSeconds; } else { // GetTickCount(64) return milliseconds return ticks * 1000000; } } static quint64 getTickCount() { resolveLibs(); // This avoids a division by zero and disables the high performance counter if it's not available if (counterFrequency > 0) { LARGE_INTEGER counter; if (QueryPerformanceCounter(&counter)) { return counter.QuadPart; } else { qWarning("QueryPerformanceCounter failed, although QueryPerformanceFrequency succeeded."); return 0; } } if (ptrGetTickCount64) return ptrGetTickCount64(); static quint32 highdword = 0; static quint32 lastval = 0; quint32 val = GetTickCount(); if (val < lastval) ++highdword; lastval = val; return val | (quint64(highdword) << 32); } int qt_msectime() { return ticksToNanoseconds(getTickCount()) / 1000000; } QElapsedTimer::ClockType QElapsedTimer::clockType() { resolveLibs(); if (counterFrequency > 0) return PerformanceCounter; else return TickCounter; } bool QElapsedTimer::isMonotonic() { return true; } void QElapsedTimer::start() { t1 = getTickCount(); t2 = 0; } qint64 QElapsedTimer::restart() { qint64 oldt1 = t1; t1 = getTickCount(); t2 = 0; return ticksToNanoseconds(t1 - oldt1) / 1000000; } qint64 QElapsedTimer::nsecsElapsed() const { qint64 elapsed = getTickCount() - t1; return ticksToNanoseconds(elapsed); } qint64 QElapsedTimer::elapsed() const { qint64 elapsed = getTickCount() - t1; return ticksToNanoseconds(elapsed) / 1000000; } qint64 QElapsedTimer::msecsSinceReference() const { return ticksToNanoseconds(t1) / 1000000; } qint64 QElapsedTimer::msecsTo(const QElapsedTimer &other) const { qint64 difference = other.t1 - t1; return ticksToNanoseconds(difference) / 1000000; } qint64 QElapsedTimer::secsTo(const QElapsedTimer &other) const { return msecsTo(other) / 1000; } bool operator<(const QElapsedTimer &v1, const QElapsedTimer &v2) { return (v1.t1 - v2.t1) < 0; } QT_END_NAMESPACE