qt 6.5.1 original

This commit is contained in:
kleuter
2023-10-29 23:33:08 +01:00
parent 71d22ab6b0
commit 85d238dfda
21202 changed files with 5499099 additions and 0 deletions

View File

@ -0,0 +1,2 @@
[stdchrono]
windows clang ci

View File

@ -0,0 +1,11 @@
# Copyright (C) 2022 The Qt Company Ltd.
# SPDX-License-Identifier: BSD-3-Clause
#####################################################################
## tst_qdeadlinetimer Test:
#####################################################################
qt_internal_add_test(tst_qdeadlinetimer
SOURCES
tst_qdeadlinetimer.cpp
)

View File

@ -0,0 +1,751 @@
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <QtCore/QString>
#include <QtCore/QTime>
#include <QtCore/QDeadlineTimer>
#include <QtCore/QElapsedTimer>
#include <QTest>
#include <QTimer>
#include <chrono>
#include <inttypes.h>
static const int minResolution = 400; // the minimum resolution for the tests
QT_BEGIN_NAMESPACE
namespace QTest {
template<> char *toString(const QDeadlineTimer &dt)
{
if (dt.isForever())
return qstrdup("QDeadlineTimer::Forever");
qint64 deadline = dt.deadlineNSecs();
char *buf = new char[256];
qsnprintf(buf, 256, "%lld.%09d%s",
deadline / 1000 / 1000 / 1000, qAbs(deadline) % (1000 * 1000 * 1000),
dt.hasExpired() ? " (expired)" : "");
return buf;
}
template <typename Rep, typename Period> char *toString(std::chrono::duration<Rep, Period> dur)
{
using namespace std::chrono;
static_assert(sizeof(double) == sizeof(qlonglong));
if constexpr (Period::num == 1 && sizeof(Rep) <= sizeof(qlonglong)) {
// typical case: second or sub-multiple of second, in a representation
// we can directly use
char *buf = new char[128];
if constexpr (std::is_integral_v<Rep>) {
char unit[] = "ss";
if constexpr (std::is_same_v<Period, std::atto>) { // from Norwegian "atten", 18
unit[0] = 'a';
} else if constexpr (std::is_same_v<Period, std::femto>) { // Norwegian "femten", 15
unit[0] = 'f';
} else if constexpr (std::is_same_v<Period, std::pico>) {
unit[0] = 'p';
} else if constexpr (std::is_same_v<Period, std::nano>) {
unit[0] = 'n';
} else if constexpr (std::is_same_v<Period, std::micro>) {
unit[0] = 'u'; // µ, really, but the output may not be UTF-8-safe
} else if constexpr (std::is_same_v<Period, std::milli>) {
unit[0] = 'm';
} else {
// deci, centi, cycles of something (60 Hz, 8000 Hz, etc.)
static_assert(Period::den == 1,
"Unsupported std::chrono::duration of a sub-multiple of second");
unit[1] = '\0';
}
// cast to qlonglong in case Rep is not int64_t
qsnprintf(buf, 128, "%lld %s", qlonglong(dur.count()), unit);
} else {
auto secs = duration_cast<duration<double>>(dur);
qsnprintf(buf, 128, "%g s", secs.count());
}
return buf;
} else if constexpr (std::is_integral_v<Rep> && Period::den == 1) {
// multiple of second, so just print it in seconds
return toString(std::chrono::seconds(dur));
} else {
// something else, use floating-point seconds
return toString(std::chrono::duration_cast<double>(dur));
}
}
}
QT_END_NAMESPACE
class tst_QDeadlineTimer : public QObject
{
Q_OBJECT
private Q_SLOTS:
void initTestCase_data();
void basics();
void foreverness();
void current();
void deadlines();
void setDeadline();
void overflow();
void expire();
void stdchrono();
};
void tst_QDeadlineTimer::initTestCase_data()
{
qRegisterMetaType<Qt::TimerType>();
QTest::addColumn<Qt::TimerType>("timerType");
QTest::newRow("precise") << Qt::PreciseTimer;
QTest::newRow("coarse") << Qt::CoarseTimer;
}
void tst_QDeadlineTimer::basics()
{
QDeadlineTimer deadline;
QCOMPARE(deadline.timerType(), Qt::CoarseTimer);
QFETCH_GLOBAL(Qt::TimerType, timerType);
deadline = QDeadlineTimer(timerType);
QCOMPARE(deadline.timerType(), timerType);
QVERIFY(!deadline.isForever());
QCOMPARE(deadline, QDeadlineTimer(timerType));
QVERIFY(!(deadline != QDeadlineTimer(timerType)));
QVERIFY(!(deadline < QDeadlineTimer()));
QCOMPARE_LE(deadline, QDeadlineTimer());
QCOMPARE_GE(deadline, QDeadlineTimer());
QVERIFY(!(deadline > QDeadlineTimer()));
QVERIFY(!(deadline < deadline));
QCOMPARE_LE(deadline, deadline);
QCOMPARE_GE(deadline, deadline);
QVERIFY(!(deadline > deadline));
// should have expired, but we may be running too early after boot
QTRY_VERIFY_WITH_TIMEOUT(deadline.hasExpired(), 100);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE(deadline.deadline(), qint64(0));
QCOMPARE(deadline.deadlineNSecs(), qint64(0));
deadline.setRemainingTime(0, timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseRemainingTime(0, 0, timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setDeadline(0, timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE(deadline.deadline(), qint64(0));
QCOMPARE(deadline.deadlineNSecs(), qint64(0));
deadline.setPreciseDeadline(0, 0, timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE(deadline.deadline(), qint64(0));
QCOMPARE(deadline.deadlineNSecs(), qint64(0));
}
void tst_QDeadlineTimer::foreverness()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
// we don't check whether timerType() is our type since it's possible it detects it's forever
QDeadlineTimer deadline = QDeadlineTimer::Forever;
QCOMPARE(deadline.timerType(), Qt::CoarseTimer);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline = QDeadlineTimer(-1, timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setRemainingTime(-1, timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseRemainingTime(-1, 0, timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseRemainingTime(-1, -1, timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setDeadline(std::numeric_limits<qint64>::max(), timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseDeadline(std::numeric_limits<qint64>::max(), 0, timerType);
QVERIFY(deadline.isForever());
QVERIFY(!deadline.hasExpired());
QCOMPARE(deadline.remainingTime(), qint64(-1));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline, deadline);
QVERIFY(!(deadline < deadline));
QCOMPARE_LE(deadline, deadline);
QCOMPARE_GE(deadline, deadline);
QVERIFY(!(deadline > deadline));
// adding to forever must still be forever
QDeadlineTimer deadline2 = deadline + 1;
QVERIFY(deadline2.isForever());
QVERIFY(!deadline2.hasExpired());
QCOMPARE(deadline2.remainingTime(), qint64(-1));
QCOMPARE(deadline2.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline2.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline2.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline2.timerType(), deadline.timerType());
QCOMPARE(deadline2 - deadline, qint64(0));
QCOMPARE(deadline2, deadline);
QVERIFY(!(deadline2 < deadline));
QCOMPARE_LE(deadline2, deadline);
QCOMPARE_GE(deadline2, deadline);
QVERIFY(!(deadline2 > deadline));
// subtracting from forever is *also* forever
deadline2 = deadline - 1;
QVERIFY(deadline2.isForever());
QVERIFY(!deadline2.hasExpired());
QCOMPARE(deadline2.remainingTime(), qint64(-1));
QCOMPARE(deadline2.remainingTimeNSecs(), qint64(-1));
QCOMPARE(deadline2.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline2.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline2.timerType(), deadline.timerType());
QCOMPARE(deadline2 - deadline, qint64(0));
QCOMPARE(deadline2, deadline);
QVERIFY(!(deadline2 < deadline));
QCOMPARE_LE(deadline2, deadline);
QCOMPARE_GE(deadline2, deadline);
QVERIFY(!(deadline2 > deadline));
// compare and order against a default-constructed object
QDeadlineTimer expired;
QVERIFY(!(deadline == expired));
QCOMPARE_NE(deadline, expired);
QVERIFY(!(deadline < expired));
QVERIFY(!(deadline <= expired));
QCOMPARE_GE(deadline, expired);
QCOMPARE_GT(deadline, expired);
}
void tst_QDeadlineTimer::current()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
auto deadline = QDeadlineTimer::current(timerType);
QVERIFY(deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
// subtracting from current should be "more expired"
QDeadlineTimer earlierDeadline = deadline - 1;
QVERIFY(earlierDeadline.hasExpired());
QVERIFY(!earlierDeadline.isForever());
QCOMPARE(earlierDeadline.timerType(), timerType);
QCOMPARE(earlierDeadline.remainingTime(), qint64(0));
QCOMPARE(earlierDeadline.remainingTimeNSecs(), qint64(0));
QCOMPARE_NE(earlierDeadline.deadline(), 0);
QCOMPARE_NE(earlierDeadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(earlierDeadline.deadlineNSecs(), 0);
QCOMPARE_NE(earlierDeadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(earlierDeadline.deadline(), deadline.deadline() - 1);
QCOMPARE(earlierDeadline.deadlineNSecs(), deadline.deadlineNSecs() - 1000*1000);
QCOMPARE(earlierDeadline - deadline, qint64(-1));
QCOMPARE_NE(earlierDeadline, deadline);
QCOMPARE_LT(earlierDeadline, deadline);
QCOMPARE_LE(earlierDeadline, deadline);
QVERIFY(!(earlierDeadline >= deadline));
QVERIFY(!(earlierDeadline > deadline));
}
void tst_QDeadlineTimer::deadlines()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
QDeadlineTimer deadline(4 * minResolution, timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (3 * minResolution));
QCOMPARE_LE(deadline.remainingTime(), (4 * minResolution));
QCOMPARE_GT(deadline.remainingTimeNSecs(), (3000000 * minResolution));
QCOMPARE_LE(deadline.remainingTimeNSecs(), (4000000 * minResolution));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setRemainingTime(4 * minResolution, timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (3 * minResolution));
QCOMPARE_LE(deadline.remainingTime(), (4 * minResolution));
QCOMPARE_GT(deadline.remainingTimeNSecs(), (3000000 * minResolution));
QCOMPARE_LE(deadline.remainingTimeNSecs(), (4000000 * minResolution));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseRemainingTime(0, 4000000 * minResolution, timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (3 * minResolution));
QCOMPARE_LE(deadline.remainingTime(), (4 * minResolution));
QCOMPARE_GT(deadline.remainingTimeNSecs(), (3000000 * minResolution));
QCOMPARE_LE(deadline.remainingTimeNSecs(), (4000000 * minResolution));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
deadline.setPreciseRemainingTime(1, 0, timerType); // 1 sec
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (1000 - minResolution));
QCOMPARE_LE(deadline.remainingTime(), 1000);
QCOMPARE_GT(deadline.remainingTimeNSecs(), (1000 - minResolution)*1000*1000);
QCOMPARE_LE(deadline.remainingTimeNSecs(), (1000*1000*1000));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
// adding to a future deadline must still be further in the future
QDeadlineTimer laterDeadline = deadline + 1;
QVERIFY(!laterDeadline.hasExpired());
QVERIFY(!laterDeadline.isForever());
QCOMPARE(laterDeadline.timerType(), timerType);
QCOMPARE_GT(laterDeadline.remainingTime(), (1000 - minResolution));
QCOMPARE_LE(laterDeadline.remainingTime(), 1001);
QCOMPARE_GT(laterDeadline.remainingTimeNSecs(), (1001 - minResolution)*1000*1000);
QCOMPARE_LE(laterDeadline.remainingTimeNSecs(), (1001*1000*1000));
QCOMPARE_NE(laterDeadline.deadline(), 0);
QCOMPARE_NE(laterDeadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(laterDeadline.deadlineNSecs(), 0);
QCOMPARE_NE(laterDeadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(laterDeadline.deadline(), deadline.deadline() + 1);
QCOMPARE(laterDeadline.deadlineNSecs(), deadline.deadlineNSecs() + 1000*1000);
QCOMPARE(laterDeadline - deadline, qint64(1));
QCOMPARE_NE(laterDeadline, deadline);
QVERIFY(!(laterDeadline < deadline));
QVERIFY(!(laterDeadline <= deadline));
QCOMPARE_GE(laterDeadline, deadline);
QCOMPARE_GT(laterDeadline, deadline);
// compare and order against a default-constructed object
QDeadlineTimer expired;
QVERIFY(!(deadline == expired));
QCOMPARE_NE(deadline, expired);
QVERIFY(!(deadline < expired));
QVERIFY(!(deadline <= expired));
QCOMPARE_GE(deadline, expired);
QCOMPARE_GT(deadline, expired);
// compare and order against a forever deadline
QDeadlineTimer forever_(QDeadlineTimer::Forever);
QVERIFY(!(deadline == forever_));
QCOMPARE_NE(deadline, forever_);
QCOMPARE_LT(deadline, forever_);
QCOMPARE_LE(deadline, forever_);
QVERIFY(!(deadline >= forever_));
QVERIFY(!(deadline > forever_));
}
void tst_QDeadlineTimer::setDeadline()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
auto now = QDeadlineTimer::current(timerType);
QDeadlineTimer deadline;
deadline.setDeadline(now.deadline(), timerType);
QVERIFY(deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE(deadline.deadline(), now.deadline());
// don't check deadlineNSecs!
deadline.setPreciseDeadline(now.deadlineNSecs() / (1000 * 1000 * 1000),
now.deadlineNSecs() % (1000 * 1000 * 1000), timerType);
QVERIFY(deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE(deadline.deadline(), now.deadline());
QCOMPARE(deadline.deadlineNSecs(), now.deadlineNSecs());
now = QDeadlineTimer::current(timerType);
deadline.setDeadline(now.deadline() + 4 * minResolution, timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (3 * minResolution));
QCOMPARE_LE(deadline.remainingTime(), (4 * minResolution));
QCOMPARE_GT(deadline.remainingTimeNSecs(), (3000000 * minResolution));
QCOMPARE_LE(deadline.remainingTimeNSecs(), (4000000 * minResolution));
QCOMPARE(deadline.deadline(), now.deadline() + 4 * minResolution); // yes, it's exact
// don't check deadlineNSecs!
now = QDeadlineTimer::current(timerType);
qint64 nsec = now.deadlineNSecs() + 4000000 * minResolution;
deadline.setPreciseDeadline(nsec / (1000 * 1000 * 1000),
nsec % (1000 * 1000 * 1000), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTime(), (3 * minResolution));
QCOMPARE_LE(deadline.remainingTime(), (4 * minResolution));
QCOMPARE_GT(deadline.remainingTimeNSecs(), (3000000 * minResolution));
QCOMPARE_LE(deadline.remainingTimeNSecs(), (4000000 * minResolution));
QCOMPARE(deadline.deadline(), nsec / (1000 * 1000));
QCOMPARE(deadline.deadlineNSecs(), nsec);
}
void tst_QDeadlineTimer::overflow()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
// Check the constructor for overflows (should also cover saturating the result of the deadline() method if overflowing)
QDeadlineTimer now = QDeadlineTimer::current(timerType), deadline(std::numeric_limits<qint64>::max() - 1, timerType);
QVERIFY(deadline.isForever() || deadline.deadline() >= now.deadline());
// Check the setDeadline with milliseconds (should also cover implicitly setting the nanoseconds as qint64 max)
deadline.setDeadline(std::numeric_limits<qint64>::max() - 1, timerType);
QVERIFY(deadline.isForever() || deadline.deadline() >= now.deadline());
// Check the setRemainingTime with milliseconds (should also cover implicitly setting the nanoseconds as qint64 max)
deadline.setRemainingTime(std::numeric_limits<qint64>::max() - 1, timerType);
QVERIFY(deadline.isForever() || deadline.deadline() >= now.deadline());
// Check that the deadline gets saturated when the arguments of setPreciseDeadline are large
deadline.setPreciseDeadline(std::numeric_limits<qint64>::max() - 1, std::numeric_limits<qint64>::max() - 1, timerType);
QCOMPARE(deadline.deadline(), std::numeric_limits<qint64>::max());
QVERIFY(deadline.isForever());
// Check that remainingTime gets saturated if we overflow
deadline.setPreciseRemainingTime(std::numeric_limits<qint64>::max() - 1, std::numeric_limits<qint64>::max() - 1, timerType);
QCOMPARE(deadline.remainingTime(), qint64(-1));
QVERIFY(deadline.isForever());
// Check that we saturate the getter for nanoseconds
deadline.setPreciseDeadline(std::numeric_limits<qint64>::max() - 1, 0, timerType);
QCOMPARE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
// Check that adding nanoseconds and overflowing is consistent and saturates the timer
deadline = QDeadlineTimer::addNSecs(deadline, std::numeric_limits<qint64>::max() - 1);
QVERIFY(deadline.isForever());
// Make sure forever is forever, regardless of us subtracting time from it
deadline = QDeadlineTimer(QDeadlineTimer::Forever, timerType);
deadline = QDeadlineTimer::addNSecs(deadline, -10000);
QVERIFY(deadline.isForever());
// Make sure we get the correct result when moving the deadline back and forth in time
QDeadlineTimer current = QDeadlineTimer::current(timerType);
QDeadlineTimer takenNSecs = QDeadlineTimer::addNSecs(current, -1000);
QVERIFY(takenNSecs.deadlineNSecs() - current.deadlineNSecs() == -1000);
QDeadlineTimer addedNSecs = QDeadlineTimer::addNSecs(current, 1000);
QVERIFY(addedNSecs.deadlineNSecs() - current.deadlineNSecs() == 1000);
// Make sure the calculation goes as expected when we need to subtract nanoseconds
// We make use of an additional timer to be certain that
// even when the environment is under load we can track the
// time needed to do the calls
static constexpr qint64 nsExpected = 1000 * 1000 * 1000 - 1000; // 1s - 1000ns, what we pass to setPreciseRemainingTime() later
QElapsedTimer callTimer;
callTimer.start();
deadline = QDeadlineTimer::current(timerType);
qint64 nsDeadline = deadline.deadlineNSecs();
// We adjust in relation to current() here, so we expect the difference to be a tad over the exact number.
// However we are tracking the elapsed time, so it shouldn't be a problem.
deadline.setPreciseRemainingTime(1, -1000, timerType);
qint64 difference = (deadline.deadlineNSecs() - nsDeadline) - nsExpected;
QCOMPARE_GE(difference, 0); // Should always be true, but just in case
QCOMPARE_LE(difference, callTimer.nsecsElapsed()); // Ideally difference should be 0 exactly
// Make sure setRemainingTime underflows gracefully
deadline.setPreciseRemainingTime(std::numeric_limits<qint64>::min() / 10, 0, timerType);
QVERIFY(!deadline.isForever()); // On Win/macOS the above underflows, make sure we don't saturate to Forever
QVERIFY(deadline.remainingTime() == 0);
// If the timer is saturated we don't want to get a valid number of milliseconds
QVERIFY(deadline.deadline() == std::numeric_limits<qint64>::min());
// Check that the conversion to milliseconds and nanoseconds underflows gracefully
deadline.setPreciseDeadline(std::numeric_limits<qint64>::min() / 10, 0, timerType);
QVERIFY(!deadline.isForever()); // On Win/macOS the above underflows, make sure we don't saturate to Forever
QVERIFY(deadline.deadline() == std::numeric_limits<qint64>::min());
QVERIFY(deadline.deadlineNSecs() == std::numeric_limits<qint64>::min());
}
void tst_QDeadlineTimer::expire()
{
QFETCH_GLOBAL(Qt::TimerType, timerType);
QDeadlineTimer deadline(minResolution, timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
qint64 previousDeadline = deadline.deadlineNSecs();
QTest::qSleep(2 * minResolution);
QCOMPARE(deadline.remainingTime(), qint64(0));
QCOMPARE(deadline.remainingTimeNSecs(), qint64(0));
QCOMPARE_NE(deadline.deadline(), 0);
QCOMPARE_NE(deadline.deadline(), std::numeric_limits<qint64>::max());
QCOMPARE_NE(deadline.deadlineNSecs(), 0);
QCOMPARE_NE(deadline.deadlineNSecs(), std::numeric_limits<qint64>::max());
QCOMPARE(deadline.deadlineNSecs(), previousDeadline);
}
void tst_QDeadlineTimer::stdchrono()
{
using namespace std::chrono;
QFETCH_GLOBAL(Qt::TimerType, timerType);
// create some forevers
QDeadlineTimer deadline = milliseconds::max();
QVERIFY(deadline.isForever());
deadline = milliseconds::max();
QVERIFY(deadline.isForever());
deadline.setRemainingTime(milliseconds::max(), timerType);
QVERIFY(deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
deadline = nanoseconds::max();
QVERIFY(deadline.isForever());
deadline.setRemainingTime(nanoseconds::max(), timerType);
QVERIFY(deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
deadline = hours::max();
QVERIFY(deadline.isForever());
deadline.setRemainingTime(hours::max(), timerType);
QVERIFY(deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
deadline = time_point<system_clock>::max();
QVERIFY(deadline.isForever());
deadline.setDeadline(time_point<system_clock>::max(), timerType);
QVERIFY(deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
deadline = time_point<steady_clock>::max();
QVERIFY(deadline.isForever());
deadline.setDeadline(time_point<steady_clock>::max(), timerType);
QVERIFY(deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QVERIFY(deadline == time_point<steady_clock>::max());
QVERIFY(deadline == time_point<system_clock>::max());
QCOMPARE(deadline.remainingTimeAsDuration(), nanoseconds::max());
// make it expired
deadline = time_point<system_clock>();
QVERIFY(deadline.hasExpired());
deadline.setDeadline(time_point<system_clock>(), timerType);
QVERIFY(deadline.hasExpired());
QCOMPARE(deadline.timerType(), timerType);
deadline = time_point<steady_clock>();
QVERIFY(deadline.hasExpired());
deadline.setDeadline(time_point<steady_clock>(), timerType);
QVERIFY(deadline.hasExpired());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE(deadline.remainingTimeAsDuration(), nanoseconds::zero());
QDeadlineTimer now;
bool timersExecuted = false;
auto steady_before = steady_clock::now();
auto system_before = system_clock::now();
decltype(steady_before) steady_after, steady_deadline;
decltype(system_before) system_after, system_deadline;
QTimer::singleShot(minResolution, Qt::PreciseTimer, [&]() {
now = QDeadlineTimer::current(timerType);
QTimer::singleShot(minResolution, Qt::PreciseTimer, [&]() {
steady_deadline = now.deadline<steady_clock>();
system_deadline = now.deadline<system_clock>();
steady_after = steady_clock::now();
system_after = system_clock::now();
timersExecuted = true;
});
});
QTRY_VERIFY2_WITH_TIMEOUT(timersExecuted,
"Looks like timers didn't fire on time.", 4 * minResolution);
#if defined(Q_OS_DARWIN) || defined(Q_OS_LINUX) || (defined(Q_CC_MSVC) && Q_CC_MSVC >= 1900)
{
// We know for these OS/compilers that the std::chrono::steady_clock uses the same
// reference time as QDeadlineTimer
qint64 before = duration_cast<nanoseconds>(steady_before.time_since_epoch()).count();
qint64 after = duration_cast<nanoseconds>(steady_after.time_since_epoch()).count();
QCOMPARE_GT(now.deadlineNSecs(), before);
QCOMPARE_LT(now.deadlineNSecs(), after);
}
#endif
{
auto diff = duration_cast<milliseconds>(steady_after - steady_deadline);
QCOMPARE_GT(diff.count(), minResolution / 2);
QCOMPARE_LT(diff.count(), 3 * minResolution / 2);
QDeadlineTimer dt_after(steady_after, timerType);
QCOMPARE_LT(now, dt_after);
diff = duration_cast<milliseconds>(steady_deadline - steady_before);
QCOMPARE_GT(diff.count(), minResolution / 2);
QCOMPARE_LT(diff.count(), 3 * minResolution / 2);
QDeadlineTimer dt_before(steady_before, timerType);
QCOMPARE_GT(now, dt_before);
}
{
auto diff = duration_cast<milliseconds>(system_after - system_deadline);
QCOMPARE_GT(diff.count(), minResolution / 2);
QCOMPARE_LT(diff.count(), 3 * minResolution / 2);
QDeadlineTimer dt_after(system_after, timerType);
QCOMPARE_LT(now, dt_after);
diff = duration_cast<milliseconds>(system_deadline - system_before);
QCOMPARE_GT(diff.count(), minResolution / 2);
QCOMPARE_LT(diff.count(), 3 * minResolution / 2);
QDeadlineTimer dt_before(system_before, timerType);
QCOMPARE_GT(now, dt_before);
}
// make it regular
now = QDeadlineTimer::current(timerType);
deadline.setRemainingTime(milliseconds(4 * minResolution), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTimeAsDuration(), milliseconds(3 * minResolution));
QCOMPARE_LT(deadline.remainingTimeAsDuration(), milliseconds(5 * minResolution));
QCOMPARE_GT(deadline.remainingTimeAsDuration(), nanoseconds(3000000 * minResolution));
QCOMPARE_LT(deadline.remainingTimeAsDuration(), nanoseconds(5000000 * minResolution));
QCOMPARE_GT(deadline.deadline<steady_clock>(), (steady_clock::now() + milliseconds(3 * minResolution)));
QCOMPARE_LT(deadline.deadline<steady_clock>(), (steady_clock::now() + milliseconds(5 * minResolution)));
QCOMPARE_GT(deadline.deadline<system_clock>(), (system_clock::now() + milliseconds(3 * minResolution)));
QCOMPARE_LT(deadline.deadline<system_clock>(), (system_clock::now() + milliseconds(5 * minResolution)));
if (timerType == Qt::CoarseTimer) {
QCOMPARE_GT(deadline, (now + milliseconds(3 * minResolution)));
QCOMPARE_LT(deadline, (now + milliseconds(5 * minResolution)));
QCOMPARE_GT(deadline, (now + nanoseconds(3000000 * minResolution)));
QCOMPARE_LT(deadline, (now + nanoseconds(5000000 * minResolution)));
QCOMPARE_GT(deadline, milliseconds(3 * minResolution));
QCOMPARE_LT(deadline, milliseconds(5 * minResolution));
QCOMPARE_GT(deadline, nanoseconds(3000000 * minResolution));
QCOMPARE_LT(deadline, nanoseconds(5000000 * minResolution));
QCOMPARE_GE(deadline, steady_clock::now());
QCOMPARE_GE(deadline, system_clock::now());
}
now = QDeadlineTimer::current(timerType);
deadline = QDeadlineTimer(seconds(1), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTimeAsDuration(), (seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.remainingTimeAsDuration(), seconds(1));
QCOMPARE_GT(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) + milliseconds(minResolution)));
QCOMPARE_GT(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) + milliseconds(minResolution)));
if (timerType == Qt::CoarseTimer) {
QCOMPARE_GT(deadline, (seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline, seconds(1));
}
now = QDeadlineTimer::current(timerType);
deadline.setRemainingTime(hours(1), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTimeAsDuration(), (hours(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.remainingTimeAsDuration(), hours(1));
QCOMPARE_GT(deadline.deadline<steady_clock>(), (steady_clock::now() + hours(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<steady_clock>(), (steady_clock::now() + hours(1) + milliseconds(minResolution)));
QCOMPARE_GT(deadline.deadline<system_clock>(), (system_clock::now() + hours(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<system_clock>(), (system_clock::now() + hours(1) + milliseconds(minResolution)));
now = QDeadlineTimer::current(timerType);
deadline.setDeadline(system_clock::now() + seconds(1), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTimeAsDuration(), (seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.remainingTimeAsDuration(), seconds(1));
QCOMPARE_GT(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) + milliseconds(minResolution)));
QCOMPARE_GT(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) + milliseconds(minResolution)));
now = QDeadlineTimer::current(timerType);
deadline.setDeadline(steady_clock::now() + seconds(1), timerType);
QVERIFY(!deadline.hasExpired());
QVERIFY(!deadline.isForever());
QCOMPARE(deadline.timerType(), timerType);
QCOMPARE_GT(deadline.remainingTimeAsDuration(), (seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.remainingTimeAsDuration(), seconds(1));
QCOMPARE_GT(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<steady_clock>(), (steady_clock::now() + seconds(1) + milliseconds(minResolution)));
QCOMPARE_GT(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) - milliseconds(minResolution)));
QCOMPARE_LE(deadline.deadline<system_clock>(), (system_clock::now() + seconds(1) + milliseconds(minResolution)));
}
QTEST_MAIN(tst_QDeadlineTimer)
#include "tst_qdeadlinetimer.moc"