Rework setMSecsSinceEpoch() to avoid local->UTC conversions
The function takes a UTC time, which it converts to local time when needed, but its (two) calls to refresh a local time instance were doing the (more expensive) reverse conversion, which we don't need (because we knew UTC to start with) and, in this case, it can't land on an invalid time, so we don't need to cope with that. Change-Id: I49b42bfa9f6a5fde12810f5a0da9ff4466ca86a4 Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: Thiago Macieira <thiago.macieira@intel.com> Reviewed-by: Andrei Golubev <andrei.golubev@qt.io>
This commit is contained in:
parent
fd5b92c2ba
commit
74eddd5bf0
@ -4005,63 +4005,60 @@ void QDateTime::setMSecsSinceEpoch(qint64 msecs)
|
|||||||
{
|
{
|
||||||
auto status = getStatus(d);
|
auto status = getStatus(d);
|
||||||
const auto spec = extractSpec(status);
|
const auto spec = extractSpec(status);
|
||||||
|
Q_ASSERT(spec == Qt::UTC || spec == Qt::LocalTime || !d.isShort());
|
||||||
|
qint64 local = msecs;
|
||||||
|
int offsetFromUtc = 0;
|
||||||
|
|
||||||
status &= ~QDateTimePrivate::ValidityMask;
|
status &= ~QDateTimePrivate::ValidityMask;
|
||||||
switch (spec) {
|
if (spec == Qt::UTC || spec == Qt::OffsetFromUTC) {
|
||||||
case Qt::UTC:
|
if (spec == Qt::OffsetFromUTC)
|
||||||
status |= QDateTimePrivate::ValidWhenMask;
|
offsetFromUtc = d->m_offsetFromUtc;
|
||||||
break;
|
if (!offsetFromUtc || !add_overflow(msecs, offsetFromUtc * MSECS_PER_SEC, &local))
|
||||||
case Qt::OffsetFromUTC:
|
|
||||||
if (!add_overflow(msecs, d->m_offsetFromUtc * MSECS_PER_SEC, &msecs))
|
|
||||||
status |= QDateTimePrivate::ValidWhenMask;
|
status |= QDateTimePrivate::ValidWhenMask;
|
||||||
break;
|
} else {
|
||||||
case Qt::TimeZone:
|
auto dst = extractDaylightStatus(status);
|
||||||
Q_ASSERT(!d.isShort());
|
if (spec == Qt::LocalTime) {
|
||||||
#if QT_CONFIG(timezone)
|
QDate dt;
|
||||||
d.detach();
|
QTime tm;
|
||||||
if (!d->m_timeZone.isValid())
|
if (QDateTimePrivate::epochMSecsToLocalTime(msecs, &dt, &tm, &dst))
|
||||||
break;
|
setDateTime(d, dt, tm);
|
||||||
status = mergeDaylightStatus(status,
|
|
||||||
d->m_timeZone.d->isDaylightTime(msecs)
|
|
||||||
? QDateTimePrivate::DaylightTime
|
|
||||||
: QDateTimePrivate::StandardTime);
|
|
||||||
d->m_offsetFromUtc = d->m_timeZone.d->offsetFromUtc(msecs);
|
|
||||||
// NB: cast to qint64 here is important to make sure a matching
|
|
||||||
// add_overflow is found, GCC 7.5.0 fails without this cast
|
|
||||||
if (!add_overflow(msecs, qint64(d->m_offsetFromUtc * MSECS_PER_SEC), &msecs))
|
|
||||||
status |= QDateTimePrivate::ValidWhenMask;
|
|
||||||
#endif // timezone
|
|
||||||
break;
|
|
||||||
case Qt::LocalTime: {
|
|
||||||
QDate dt;
|
|
||||||
QTime tm;
|
|
||||||
QDateTimePrivate::DaylightStatus dstStatus;
|
|
||||||
if (QDateTimePrivate::epochMSecsToLocalTime(msecs, &dt, &tm, &dstStatus)) {
|
|
||||||
setDateTime(d, dt, tm);
|
|
||||||
status = getStatus(d);
|
status = getStatus(d);
|
||||||
|
if ((status & QDateTimePrivate::ValidDate) && (status & QDateTimePrivate::ValidTime)) {
|
||||||
|
local = getMSecs(d);
|
||||||
|
offsetFromUtc = (local - msecs) / MSECS_PER_SEC;
|
||||||
|
status |= QDateTimePrivate::ValidWhenMask;
|
||||||
|
}
|
||||||
|
#if QT_CONFIG(timezone)
|
||||||
|
} else if (spec == Qt::TimeZone && (d.detach(), d->m_timeZone.isValid())) {
|
||||||
|
const auto data = d->m_timeZone.d->data(msecs);
|
||||||
|
if (data.offsetFromUtc != QTimeZonePrivate::invalidSeconds()) {
|
||||||
|
dst = data.daylightTimeOffset
|
||||||
|
? QDateTimePrivate::DaylightTime
|
||||||
|
: QDateTimePrivate::StandardTime;
|
||||||
|
offsetFromUtc = data.offsetFromUtc;
|
||||||
|
if (!offsetFromUtc
|
||||||
|
// NB: cast to qint64 here is important to make sure a matching
|
||||||
|
// add_overflow is found, GCC 7.5.0 fails without this cast
|
||||||
|
|| !add_overflow(msecs, qint64(offsetFromUtc * MSECS_PER_SEC), &local)) {
|
||||||
|
status |= QDateTimePrivate::ValidWhenMask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // timezone
|
||||||
}
|
}
|
||||||
if ((status & QDateTimePrivate::ValidDate) && (status & QDateTimePrivate::ValidTime)) {
|
status = mergeDaylightStatus(status, dst);
|
||||||
refreshZonedDateTime(d, spec); // FIXME: we do this again, below
|
|
||||||
msecs = getMSecs(d);
|
|
||||||
status = mergeDaylightStatus(getStatus(d), dstStatus);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Q_ASSERT(!(status & QDateTimePrivate::ValidDateTime)
|
||||||
|
|| (offsetFromUtc >= -SECS_PER_DAY && offsetFromUtc <= SECS_PER_DAY));
|
||||||
|
|
||||||
if (msecsCanBeSmall(msecs) && d.isShort()) {
|
if (msecsCanBeSmall(local) && d.isShort()) {
|
||||||
// we can keep short
|
// we can keep short
|
||||||
d.data.msecs = qintptr(msecs);
|
d.data.msecs = qintptr(local);
|
||||||
d.data.status = status.toInt();
|
d.data.status = status.toInt();
|
||||||
} else {
|
} else {
|
||||||
d.detach();
|
d.detach();
|
||||||
d->m_status = status & ~QDateTimePrivate::ShortData;
|
d->m_status = status & ~QDateTimePrivate::ShortData;
|
||||||
d->m_msecs = msecs;
|
d->m_msecs = local;
|
||||||
}
|
d->m_offsetFromUtc = offsetFromUtc;
|
||||||
|
|
||||||
if (spec == Qt::LocalTime || spec == Qt::TimeZone) {
|
|
||||||
refreshZonedDateTime(d, spec);
|
|
||||||
Q_ASSERT((d.isShort() ? d.data.msecs : d->m_msecs) == msecs);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user