QDateTime: Fix clearing the ShortData flag in setMSecsSinceEpoch
Unlike setTimeSpec, this forgot to clear the bit when detaching. So it's possible that some further use of the flags could incorrectly conclude that the data was short and then proceed to corrupt the pointer. The example from QTBUG-59061 caused this because toUTC() -> toTimeSpec() calls setMSecsSinceEpoch which left the bit set; then addDays() calls setDateTime(), which calls checkValidDateTime() and that corrupted the pointer. This problem was more visible on 32-bit systems because no QDateTime was short (except for default constructed ones), but it can happen on 64-bit with sufficiently large dates. Task-number: QTBUG-59061 Change-Id: Ibc5c715fda334a75bd2efffd14a562a375a4e69b Reviewed-by: Edward Welbourne <edward.welbourne@qt.io>
This commit is contained in:
parent
4d3781b640
commit
07fffa6010
@ -2842,6 +2842,9 @@ inline bool QDateTime::Data::isShort() const
|
|||||||
{
|
{
|
||||||
bool b = quintptr(d) & QDateTimePrivate::ShortData;
|
bool b = quintptr(d) & QDateTimePrivate::ShortData;
|
||||||
|
|
||||||
|
// sanity check:
|
||||||
|
Q_ASSERT(b || (d->m_status & QDateTimePrivate::ShortData) == 0);
|
||||||
|
|
||||||
// even if CanBeSmall = false, we have short data for a default-constructed
|
// even if CanBeSmall = false, we have short data for a default-constructed
|
||||||
// QDateTime object. But it's unlikely.
|
// QDateTime object. But it's unlikely.
|
||||||
if (CanBeSmall)
|
if (CanBeSmall)
|
||||||
@ -3658,7 +3661,7 @@ void QDateTime::setMSecsSinceEpoch(qint64 msecs)
|
|||||||
d.data.status = status;
|
d.data.status = status;
|
||||||
} else {
|
} else {
|
||||||
d.detach();
|
d.detach();
|
||||||
d->m_status = status;
|
d->m_status = status & ~QDateTimePrivate::ShortData;
|
||||||
d->m_msecs = msecs;
|
d->m_msecs = msecs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -667,6 +667,29 @@ void tst_QDateTime::setMSecsSinceEpoch()
|
|||||||
QCOMPARE(dt.time(), utc.time());
|
QCOMPARE(dt.time(), utc.time());
|
||||||
QCOMPARE(dt.timeSpec(), Qt::UTC);
|
QCOMPARE(dt.timeSpec(), Qt::UTC);
|
||||||
|
|
||||||
|
{
|
||||||
|
QDateTime dt1 = QDateTime::fromMSecsSinceEpoch(msecs, Qt::UTC);
|
||||||
|
QCOMPARE(dt1, utc);
|
||||||
|
QCOMPARE(dt1.date(), utc.date());
|
||||||
|
QCOMPARE(dt1.time(), utc.time());
|
||||||
|
QCOMPARE(dt1.timeSpec(), Qt::UTC);
|
||||||
|
}
|
||||||
|
{
|
||||||
|
QDateTime dt1(utc.date(), utc.time(), Qt::UTC);
|
||||||
|
QCOMPARE(dt1, utc);
|
||||||
|
QCOMPARE(dt1.date(), utc.date());
|
||||||
|
QCOMPARE(dt1.time(), utc.time());
|
||||||
|
QCOMPARE(dt1.timeSpec(), Qt::UTC);
|
||||||
|
}
|
||||||
|
{
|
||||||
|
// used to fail to clear the ShortData bit, causing corruption
|
||||||
|
QDateTime dt1 = dt.addDays(0);
|
||||||
|
QCOMPARE(dt1, utc);
|
||||||
|
QCOMPARE(dt1.date(), utc.date());
|
||||||
|
QCOMPARE(dt1.time(), utc.time());
|
||||||
|
QCOMPARE(dt1.timeSpec(), Qt::UTC);
|
||||||
|
}
|
||||||
|
|
||||||
if (zoneIsCET) {
|
if (zoneIsCET) {
|
||||||
QCOMPARE(dt.toLocalTime(), cet);
|
QCOMPARE(dt.toLocalTime(), cet);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user