2011-10-25 07:51:19 +00:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
2012-01-05 04:03:39 +00:00
|
|
|
** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies).
|
2011-10-25 07:51:19 +00:00
|
|
|
** All rights reserved.
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
**
|
|
|
|
** 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 <qcoreapplication.h>
|
|
|
|
|
|
|
|
#include "private/qcore_unix_p.h"
|
|
|
|
#include "private/qtimerinfo_unix_p.h"
|
|
|
|
#include "private/qobject_p.h"
|
|
|
|
#include "private/qabstracteventdispatcher_p.h"
|
|
|
|
|
|
|
|
#include <sys/times.h>
|
|
|
|
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
Q_CORE_EXPORT bool qt_disable_lowpriority_timers=false;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Internal functions for manipulating timer data structures. The
|
|
|
|
* timerBitVec array is used for keeping track of timer identifiers.
|
|
|
|
*/
|
|
|
|
|
|
|
|
QTimerInfoList::QTimerInfoList()
|
|
|
|
{
|
|
|
|
#if (_POSIX_MONOTONIC_CLOCK-0 <= 0) && !defined(Q_OS_MAC) && !defined(Q_OS_NACL)
|
|
|
|
if (!QElapsedTimer::isMonotonic()) {
|
|
|
|
// not using monotonic timers, initialize the timeChanged() machinery
|
|
|
|
previousTime = qt_gettime();
|
|
|
|
|
|
|
|
tms unused;
|
|
|
|
previousTicks = times(&unused);
|
|
|
|
|
|
|
|
ticksPerSecond = sysconf(_SC_CLK_TCK);
|
|
|
|
msPerTick = 1000/ticksPerSecond;
|
|
|
|
} else {
|
|
|
|
// detected monotonic timers
|
|
|
|
previousTime.tv_sec = previousTime.tv_usec = 0;
|
|
|
|
previousTicks = 0;
|
|
|
|
ticksPerSecond = 0;
|
|
|
|
msPerTick = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
firstTimerInfo = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
timeval QTimerInfoList::updateCurrentTime()
|
|
|
|
{
|
|
|
|
return (currentTime = qt_gettime());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ((_POSIX_MONOTONIC_CLOCK-0 <= 0) && !defined(Q_OS_MAC) && !defined(Q_OS_INTEGRITY)) || defined(QT_BOOTSTRAPPED)
|
|
|
|
|
|
|
|
template <>
|
|
|
|
timeval qAbs(const timeval &t)
|
|
|
|
{
|
|
|
|
timeval tmp = t;
|
|
|
|
if (tmp.tv_sec < 0) {
|
|
|
|
tmp.tv_sec = -tmp.tv_sec - 1;
|
|
|
|
tmp.tv_usec -= 1000000;
|
|
|
|
}
|
|
|
|
if (tmp.tv_sec == 0 && tmp.tv_usec < 0) {
|
|
|
|
tmp.tv_usec = -tmp.tv_usec;
|
|
|
|
}
|
|
|
|
return normalizedTimeval(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Returns true if the real time clock has changed by more than 10%
|
|
|
|
relative to the processor time since the last time this function was
|
|
|
|
called. This presumably means that the system time has been changed.
|
|
|
|
|
|
|
|
If /a delta is nonzero, delta is set to our best guess at how much the system clock was changed.
|
|
|
|
*/
|
|
|
|
bool QTimerInfoList::timeChanged(timeval *delta)
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_NACL
|
|
|
|
Q_UNUSED(delta)
|
|
|
|
return false; // Calling "times" crashes.
|
|
|
|
#endif
|
|
|
|
struct tms unused;
|
|
|
|
clock_t currentTicks = times(&unused);
|
|
|
|
|
|
|
|
clock_t elapsedTicks = currentTicks - previousTicks;
|
|
|
|
timeval elapsedTime = currentTime - previousTime;
|
|
|
|
|
|
|
|
timeval elapsedTimeTicks;
|
|
|
|
elapsedTimeTicks.tv_sec = elapsedTicks / ticksPerSecond;
|
|
|
|
elapsedTimeTicks.tv_usec = (((elapsedTicks * 1000) / ticksPerSecond) % 1000) * 1000;
|
|
|
|
|
|
|
|
timeval dummy;
|
|
|
|
if (!delta)
|
|
|
|
delta = &dummy;
|
|
|
|
*delta = elapsedTime - elapsedTimeTicks;
|
|
|
|
|
|
|
|
previousTicks = currentTicks;
|
|
|
|
previousTime = currentTime;
|
|
|
|
|
|
|
|
// If tick drift is more than 10% off compared to realtime, we assume that the clock has
|
|
|
|
// been set. Of course, we have to allow for the tick granularity as well.
|
|
|
|
timeval tickGranularity;
|
|
|
|
tickGranularity.tv_sec = 0;
|
|
|
|
tickGranularity.tv_usec = msPerTick * 1000;
|
|
|
|
return elapsedTimeTicks < ((qAbs(*delta) - tickGranularity) * 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QTimerInfoList::repairTimersIfNeeded()
|
|
|
|
{
|
|
|
|
if (QElapsedTimer::isMonotonic())
|
|
|
|
return;
|
|
|
|
timeval delta;
|
|
|
|
if (timeChanged(&delta))
|
|
|
|
timerRepair(delta);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // !(_POSIX_MONOTONIC_CLOCK-0 <= 0) && !defined(QT_BOOTSTRAPPED)
|
|
|
|
|
|
|
|
void QTimerInfoList::repairTimersIfNeeded()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
insert timer info into list
|
|
|
|
*/
|
|
|
|
void QTimerInfoList::timerInsert(QTimerInfo *ti)
|
|
|
|
{
|
|
|
|
int index = size();
|
|
|
|
while (index--) {
|
|
|
|
register const QTimerInfo * const t = at(index);
|
|
|
|
if (!(ti->timeout < t->timeout))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
insert(index+1, ti);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
repair broken timer
|
|
|
|
*/
|
|
|
|
void QTimerInfoList::timerRepair(const timeval &diff)
|
|
|
|
{
|
|
|
|
// repair all timers
|
|
|
|
for (int i = 0; i < size(); ++i) {
|
|
|
|
register QTimerInfo *t = at(i);
|
|
|
|
t->timeout = t->timeout + diff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-21 10:33:56 +00:00
|
|
|
inline timeval &operator+=(timeval &t1, int ms)
|
|
|
|
{
|
|
|
|
t1.tv_sec += ms / 1000;
|
|
|
|
t1.tv_usec += ms % 1000 * 1000;
|
|
|
|
return normalizedTimeval(t1);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline timeval operator+(const timeval &t1, int ms)
|
|
|
|
{
|
|
|
|
timeval t2 = t1;
|
|
|
|
return t2 += ms;
|
|
|
|
}
|
|
|
|
|
2011-10-25 07:51:19 +00:00
|
|
|
static timeval roundToMillisecond(timeval val)
|
|
|
|
{
|
|
|
|
// always round up
|
|
|
|
// worst case scenario is that the first trigger of a 1-ms timer is 0.999 ms late
|
|
|
|
|
|
|
|
int us = val.tv_usec % 1000;
|
|
|
|
val.tv_usec += 1000 - us;
|
|
|
|
return normalizedTimeval(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Returns the time to wait for the next timer, or null if no timers
|
|
|
|
are waiting.
|
|
|
|
*/
|
|
|
|
bool QTimerInfoList::timerWait(timeval &tm)
|
|
|
|
{
|
|
|
|
timeval currentTime = updateCurrentTime();
|
|
|
|
repairTimersIfNeeded();
|
|
|
|
|
|
|
|
// Find first waiting timer not already active
|
|
|
|
QTimerInfo *t = 0;
|
|
|
|
for (QTimerInfoList::const_iterator it = constBegin(); it != constEnd(); ++it) {
|
|
|
|
if (!(*it)->activateRef) {
|
|
|
|
t = *it;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!t)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (currentTime < t->timeout) {
|
|
|
|
// time to wait
|
|
|
|
tm = roundToMillisecond(t->timeout - currentTime);
|
|
|
|
} else {
|
|
|
|
// no time to wait
|
|
|
|
tm.tv_sec = 0;
|
|
|
|
tm.tv_usec = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-12-21 10:33:33 +00:00
|
|
|
void QTimerInfoList::registerTimer(int timerId, int interval, Qt::TimerType timerType, QObject *object)
|
2011-10-25 07:51:19 +00:00
|
|
|
{
|
|
|
|
QTimerInfo *t = new QTimerInfo;
|
|
|
|
t->id = timerId;
|
2011-12-21 10:33:56 +00:00
|
|
|
t->interval = interval;
|
2011-12-21 10:33:33 +00:00
|
|
|
t->timerType = timerType;
|
2011-10-25 07:51:19 +00:00
|
|
|
t->timeout = updateCurrentTime() + t->interval;
|
|
|
|
t->obj = object;
|
|
|
|
t->activateRef = 0;
|
|
|
|
|
|
|
|
timerInsert(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QTimerInfoList::unregisterTimer(int timerId)
|
|
|
|
{
|
|
|
|
// set timer inactive
|
|
|
|
for (int i = 0; i < count(); ++i) {
|
|
|
|
register QTimerInfo *t = at(i);
|
|
|
|
if (t->id == timerId) {
|
|
|
|
// found it
|
|
|
|
removeAt(i);
|
|
|
|
if (t == firstTimerInfo)
|
|
|
|
firstTimerInfo = 0;
|
|
|
|
if (t->activateRef)
|
|
|
|
*(t->activateRef) = 0;
|
|
|
|
delete t;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// id not found
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QTimerInfoList::unregisterTimers(QObject *object)
|
|
|
|
{
|
|
|
|
if (isEmpty())
|
|
|
|
return false;
|
|
|
|
for (int i = 0; i < count(); ++i) {
|
|
|
|
register QTimerInfo *t = at(i);
|
|
|
|
if (t->obj == object) {
|
|
|
|
// object found
|
|
|
|
removeAt(i);
|
|
|
|
if (t == firstTimerInfo)
|
|
|
|
firstTimerInfo = 0;
|
|
|
|
if (t->activateRef)
|
|
|
|
*(t->activateRef) = 0;
|
|
|
|
delete t;
|
|
|
|
// move back one so that we don't skip the new current item
|
|
|
|
--i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-12-21 10:33:33 +00:00
|
|
|
QList<QAbstractEventDispatcher::TimerInfo> QTimerInfoList::registeredTimers(QObject *object) const
|
2011-10-25 07:51:19 +00:00
|
|
|
{
|
2011-12-21 10:33:33 +00:00
|
|
|
QList<QAbstractEventDispatcher::TimerInfo> list;
|
2011-10-25 07:51:19 +00:00
|
|
|
for (int i = 0; i < count(); ++i) {
|
|
|
|
register const QTimerInfo * const t = at(i);
|
|
|
|
if (t->obj == object)
|
2011-12-21 10:33:56 +00:00
|
|
|
list << QAbstractEventDispatcher::TimerInfo(t->id, t->interval, t->timerType);
|
2011-10-25 07:51:19 +00:00
|
|
|
}
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Activate pending timers, returning how many where activated.
|
|
|
|
*/
|
|
|
|
int QTimerInfoList::activateTimers()
|
|
|
|
{
|
|
|
|
if (qt_disable_lowpriority_timers || isEmpty())
|
|
|
|
return 0; // nothing to do
|
|
|
|
|
|
|
|
int n_act = 0, maxCount = 0;
|
|
|
|
firstTimerInfo = 0;
|
|
|
|
|
|
|
|
timeval currentTime = updateCurrentTime();
|
|
|
|
repairTimersIfNeeded();
|
|
|
|
|
|
|
|
|
|
|
|
// Find out how many timer have expired
|
|
|
|
for (QTimerInfoList::const_iterator it = constBegin(); it != constEnd(); ++it) {
|
|
|
|
if (currentTime < (*it)->timeout)
|
|
|
|
break;
|
|
|
|
maxCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
//fire the timers.
|
|
|
|
while (maxCount--) {
|
|
|
|
if (isEmpty())
|
|
|
|
break;
|
|
|
|
|
|
|
|
QTimerInfo *currentTimerInfo = first();
|
|
|
|
if (currentTime < currentTimerInfo->timeout)
|
|
|
|
break; // no timer has expired
|
|
|
|
|
|
|
|
if (!firstTimerInfo) {
|
|
|
|
firstTimerInfo = currentTimerInfo;
|
|
|
|
} else if (firstTimerInfo == currentTimerInfo) {
|
|
|
|
// avoid sending the same timer multiple times
|
|
|
|
break;
|
|
|
|
} else if (currentTimerInfo->interval < firstTimerInfo->interval
|
|
|
|
|| currentTimerInfo->interval == firstTimerInfo->interval) {
|
|
|
|
firstTimerInfo = currentTimerInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove from list
|
|
|
|
removeFirst();
|
|
|
|
|
|
|
|
// determine next timeout time
|
|
|
|
currentTimerInfo->timeout += currentTimerInfo->interval;
|
|
|
|
if (currentTimerInfo->timeout < currentTime)
|
|
|
|
currentTimerInfo->timeout = currentTime + currentTimerInfo->interval;
|
|
|
|
|
|
|
|
// reinsert timer
|
|
|
|
timerInsert(currentTimerInfo);
|
2011-12-21 10:33:56 +00:00
|
|
|
if (currentTimerInfo->interval > 0)
|
2011-10-25 07:51:19 +00:00
|
|
|
n_act++;
|
|
|
|
|
|
|
|
if (!currentTimerInfo->activateRef) {
|
|
|
|
// send event, but don't allow it to recurse
|
|
|
|
currentTimerInfo->activateRef = ¤tTimerInfo;
|
|
|
|
|
|
|
|
QTimerEvent e(currentTimerInfo->id);
|
|
|
|
QCoreApplication::sendEvent(currentTimerInfo->obj, &e);
|
|
|
|
|
|
|
|
if (currentTimerInfo)
|
|
|
|
currentTimerInfo->activateRef = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
firstTimerInfo = 0;
|
|
|
|
return n_act;
|
|
|
|
}
|
|
|
|
|
|
|
|
QT_END_NAMESPACE
|