Commit 5b87292c authored by Allan Sandfeld Jensen's avatar Allan Sandfeld Jensen
Browse files

Fix DateTime with recent versions of tzdata

An backport of http://trac.webkit.org/changeset/150833

 needed for
correct time KRAT, YAKT and MOS timezones.

Change-Id: I3b5369d1427757c0d638865324a36e43dcaa60bf
Task-number: QTBUG-41422
Reviewed-by: default avatarJocelyn Turcotte <jocelyn.turcotte@digia.com>
Showing with 108 additions and 115 deletions
...@@ -129,10 +129,8 @@ ConstructType DateConstructor::getConstructData(ConstructData& constructData) ...@@ -129,10 +129,8 @@ ConstructType DateConstructor::getConstructData(ConstructData& constructData)
// ECMA 15.9.2 // ECMA 15.9.2
static JSValue JSC_HOST_CALL callDate(ExecState* exec, JSObject*, JSValue, const ArgList&) static JSValue JSC_HOST_CALL callDate(ExecState* exec, JSObject*, JSValue, const ArgList&)
{ {
time_t localTime = time(0); GregorianDateTime ts;
tm localTM; msToGregorianDateTime(exec, currentTimeMS(), false, ts);
getLocalTime(&localTime, &localTM);
GregorianDateTime ts(exec, localTM);
DateConversionBuffer date; DateConversionBuffer date;
DateConversionBuffer time; DateConversionBuffer time;
formatDate(ts, date); formatDate(ts, date);
......
...@@ -143,7 +143,6 @@ JSGlobalData::JSGlobalData(bool isShared) ...@@ -143,7 +143,6 @@ JSGlobalData::JSGlobalData(bool isShared)
, functionCodeBlockBeingReparsed(0) , functionCodeBlockBeingReparsed(0)
, firstStringifierToMark(0) , firstStringifierToMark(0)
, markStack(jsArrayVPtr) , markStack(jsArrayVPtr)
, cachedUTCOffset(NaN)
#ifndef NDEBUG #ifndef NDEBUG
, mainThreadOnly(false) , mainThreadOnly(false)
#endif #endif
...@@ -259,8 +258,7 @@ JSGlobalData::ClientData::~ClientData() ...@@ -259,8 +258,7 @@ JSGlobalData::ClientData::~ClientData()
void JSGlobalData::resetDateCache() void JSGlobalData::resetDateCache()
{ {
cachedUTCOffset = NaN; localTimeOffsetCache.reset();
dstOffsetCache.reset();
cachedDateString = UString(); cachedDateString = UString();
dateInstanceCache.reset(); dateInstanceCache.reset();
} }
......
...@@ -39,6 +39,7 @@ ...@@ -39,6 +39,7 @@
#include "SmallStrings.h" #include "SmallStrings.h"
#include "TimeoutChecker.h" #include "TimeoutChecker.h"
#include "WeakRandom.h" #include "WeakRandom.h"
#include <wtf/DateMath.h>
#include <wtf/Forward.h> #include <wtf/Forward.h>
#include <wtf/HashMap.h> #include <wtf/HashMap.h>
#include <wtf/RefCounted.h> #include <wtf/RefCounted.h>
...@@ -63,21 +64,23 @@ namespace JSC { ...@@ -63,21 +64,23 @@ namespace JSC {
struct HashTable; struct HashTable;
struct Instruction; struct Instruction;
struct DSTOffsetCache { struct LocalTimeOffsetCache {
DSTOffsetCache() LocalTimeOffsetCache()
: start(0.0)
, end(-1.0)
, increment(0.0)
{ {
reset();
} }
void reset() void reset()
{ {
offset = 0.0; offset = LocalTimeOffset();
start = 0.0; start = 0.0;
end = -1.0; end = -1.0;
increment = 0.0; increment = 0.0;
} }
double offset; LocalTimeOffset offset;
double start; double start;
double end; double end;
double increment; double increment;
...@@ -180,8 +183,7 @@ namespace JSC { ...@@ -180,8 +183,7 @@ namespace JSC {
MarkStack markStack; MarkStack markStack;
double cachedUTCOffset; LocalTimeOffsetCache localTimeOffsetCache;
DSTOffsetCache dstOffsetCache;
UString cachedDateString; UString cachedDateString;
double cachedDateStringValue; double cachedDateStringValue;
......
...@@ -377,6 +377,8 @@ int equivalentYearForDST(int year) ...@@ -377,6 +377,8 @@ int equivalentYearForDST(int year)
return year; return year;
} }
#if !HAVE(TM_GMTOFF)
static int32_t calculateUTCOffset() static int32_t calculateUTCOffset()
{ {
#if PLATFORM(BREWMP) #if PLATFORM(BREWMP)
...@@ -418,23 +420,15 @@ static int32_t calculateUTCOffset() ...@@ -418,23 +420,15 @@ static int32_t calculateUTCOffset()
/* /*
* Get the DST offset for the time passed in. * Get the DST offset for the time passed in.
*/ */
static double calculateDSTOffsetSimple(double localTimeSeconds, double utcOffset) static double calculateDSTOffset(time_t localTime, double utcOffset)
{ {
if (localTimeSeconds > maxUnixTime)
localTimeSeconds = maxUnixTime;
else if (localTimeSeconds < 0) // Go ahead a day to make localtime work (does not work with 0)
localTimeSeconds += secondsPerDay;
//input is UTC so we have to shift back to local time to determine DST thus the + getUTCOffset() //input is UTC so we have to shift back to local time to determine DST thus the + getUTCOffset()
double offsetTime = (localTimeSeconds * msPerSecond) + utcOffset; double offsetTime = (localTime * msPerSecond) + utcOffset;
// Offset from UTC but doesn't include DST obviously // Offset from UTC but doesn't include DST obviously
int offsetHour = msToHours(offsetTime); int offsetHour = msToHours(offsetTime);
int offsetMinute = msToMinutes(offsetTime); int offsetMinute = msToMinutes(offsetTime);
// FIXME: time_t has a potential problem in 2038
time_t localTime = static_cast<time_t>(localTimeSeconds);
tm localTM; tm localTM;
getLocalTime(&localTime, &localTM); getLocalTime(&localTime, &localTM);
...@@ -446,10 +440,12 @@ static double calculateDSTOffsetSimple(double localTimeSeconds, double utcOffset ...@@ -446,10 +440,12 @@ static double calculateDSTOffsetSimple(double localTimeSeconds, double utcOffset
return (diff * msPerSecond); return (diff * msPerSecond);
} }
// Get the DST offset, given a time in UTC #endif
static double calculateDSTOffset(double ms, double utcOffset)
// Returns combined offset in millisecond (UTC + DST).
LocalTimeOffset calculateLocalTimeOffset(double ms)
{ {
// On Mac OS X, the call to localtime (see calculateDSTOffsetSimple) will return historically accurate // On Mac OS X, the call to localtime (see calculateDSTOffset) will return historically accurate
// DST information (e.g. New Zealand did not have DST from 1946 to 1974) however the JavaScript // DST information (e.g. New Zealand did not have DST from 1946 to 1974) however the JavaScript
// standard explicitly dictates that historical information should not be considered when // standard explicitly dictates that historical information should not be considered when
// determining DST. For this reason we shift away from years that localtime can handle but would // determining DST. For this reason we shift away from years that localtime can handle but would
...@@ -465,7 +461,23 @@ static double calculateDSTOffset(double ms, double utcOffset) ...@@ -465,7 +461,23 @@ static double calculateDSTOffset(double ms, double utcOffset)
ms = (day * msPerDay) + msToMilliseconds(ms); ms = (day * msPerDay) + msToMilliseconds(ms);
} }
return calculateDSTOffsetSimple(ms / msPerSecond, utcOffset); double localTimeSeconds = ms / msPerSecond;
if (localTimeSeconds > maxUnixTime)
localTimeSeconds = maxUnixTime;
else if (localTimeSeconds < 0) // Go ahead a day to make localtime work (does not work with 0).
localTimeSeconds += secondsPerDay;
// FIXME: time_t has a potential problem in 2038.
time_t localTime = static_cast<time_t>(localTimeSeconds);
#if HAVE(TM_GMTOFF)
tm localTM;
getLocalTime(&localTime, &localTM);
return LocalTimeOffset(localTM.tm_isdst, localTM.tm_gmtoff * msPerSecond);
#else
double utcOffset = calculateUTCOffset();
double dstOffset = calculateDSTOffset(localTime, utcOffset);
return LocalTimeOffset(dstOffset, utcOffset + dstOffset);
#endif
} }
void initializeDates() void initializeDates()
...@@ -838,11 +850,9 @@ double parseDateFromNullTerminatedCharacters(const char* dateString) ...@@ -838,11 +850,9 @@ double parseDateFromNullTerminatedCharacters(const char* dateString)
return NaN; return NaN;
// fall back to local timezone // fall back to local timezone
if (!haveTZ) { if (!haveTZ)
double utcOffset = calculateUTCOffset(); offset = calculateLocalTimeOffset(ms).offset / msPerMinute;
double dstOffset = calculateDSTOffset(ms, utcOffset);
offset = static_cast<int>((utcOffset + dstOffset) / msPerMinute);
}
return ms - (offset * msPerMinute); return ms - (offset * msPerMinute);
} }
...@@ -859,14 +869,14 @@ double timeClip(double t) ...@@ -859,14 +869,14 @@ double timeClip(double t)
#if USE(JSC) #if USE(JSC)
namespace JSC { namespace JSC {
// Get the DST offset for the time passed in. // Get the combined UTC + DST offset for the time passed in.
// //
// NOTE: The implementation relies on the fact that no time zones have // NOTE: The implementation relies on the fact that no time zones have
// more than one daylight savings offset change per month. // more than one daylight savings offset change per month.
// If this function is called with NaN it returns NaN. // If this function is called with NaN it returns NaN.
static double getDSTOffset(ExecState* exec, double ms, double utcOffset) static LocalTimeOffset localTimeOffset(ExecState* exec, double ms)
{ {
DSTOffsetCache& cache = exec->globalData().dstOffsetCache; LocalTimeOffsetCache& cache = exec->globalData().localTimeOffsetCache;
double start = cache.start; double start = cache.start;
double end = cache.end; double end = cache.end;
...@@ -878,7 +888,7 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset) ...@@ -878,7 +888,7 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset)
double newEnd = end + cache.increment; double newEnd = end + cache.increment;
if (ms <= newEnd) { if (ms <= newEnd) {
double endOffset = calculateDSTOffset(newEnd, utcOffset); LocalTimeOffset endOffset = calculateLocalTimeOffset(newEnd);
if (cache.offset == endOffset) { if (cache.offset == endOffset) {
// If the offset at the end of the new interval still matches // If the offset at the end of the new interval still matches
// the offset in the cache, we grow the cached time interval // the offset in the cache, we grow the cached time interval
...@@ -886,34 +896,33 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset) ...@@ -886,34 +896,33 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset)
cache.end = newEnd; cache.end = newEnd;
cache.increment = msPerMonth; cache.increment = msPerMonth;
return endOffset; return endOffset;
}
LocalTimeOffset offset = calculateLocalTimeOffset(ms);
if (offset == endOffset) {
// The offset at the given time is equal to the offset at the
// new end of the interval, so that means that we've just skipped
// the point in time where the DST offset change occurred. Updated
// the interval to reflect this and reset the increment.
cache.start = ms;
cache.end = newEnd;
cache.increment = msPerMonth;
} else { } else {
double offset = calculateDSTOffset(ms, utcOffset); // The interval contains a DST offset change and the given time is
if (offset == endOffset) { // before it. Adjust the increment to avoid a linear search for
// The offset at the given time is equal to the offset at the // the offset change point and change the end of the interval.
// new end of the interval, so that means that we've just skipped cache.increment /= 3;
// the point in time where the DST offset change occurred. Updated cache.end = ms;
// the interval to reflect this and reset the increment.
cache.start = ms;
cache.end = newEnd;
cache.increment = msPerMonth;
} else {
// The interval contains a DST offset change and the given time is
// before it. Adjust the increment to avoid a linear search for
// the offset change point and change the end of the interval.
cache.increment /= 3;
cache.end = ms;
}
// Update the offset in the cache and return it.
cache.offset = offset;
return offset;
} }
// Update the offset in the cache and return it.
cache.offset = offset;
return offset;
} }
} }
// Compute the DST offset for the time and shrink the cache interval // Compute the DST offset for the time and shrink the cache interval
// to only contain the time. This allows fast repeated DST offset // to only contain the time. This allows fast repeated DST offset
// computations for the same time. // computations for the same time.
double offset = calculateDSTOffset(ms, utcOffset); LocalTimeOffset offset = calculateLocalTimeOffset(ms);
cache.offset = offset; cache.offset = offset;
cache.start = ms; cache.start = ms;
cache.end = ms; cache.end = ms;
...@@ -921,30 +930,14 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset) ...@@ -921,30 +930,14 @@ static double getDSTOffset(ExecState* exec, double ms, double utcOffset)
return offset; return offset;
} }
/*
* Get the difference in milliseconds between this time zone and UTC (GMT)
* NOT including DST.
*/
double getUTCOffset(ExecState* exec)
{
double utcOffset = exec->globalData().cachedUTCOffset;
if (!isnan(utcOffset))
return utcOffset;
exec->globalData().cachedUTCOffset = calculateUTCOffset();
return exec->globalData().cachedUTCOffset;
}
double gregorianDateTimeToMS(ExecState* exec, const GregorianDateTime& t, double milliSeconds, bool inputIsUTC) double gregorianDateTimeToMS(ExecState* exec, const GregorianDateTime& t, double milliSeconds, bool inputIsUTC)
{ {
double day = dateToDaysFrom1970(t.year + 1900, t.month, t.monthDay); double day = dateToDaysFrom1970(t.year + 1900, t.month, t.monthDay);
double ms = timeToMS(t.hour, t.minute, t.second, milliSeconds); double ms = timeToMS(t.hour, t.minute, t.second, milliSeconds);
double result = (day * WTF::msPerDay) + ms; double result = (day * WTF::msPerDay) + ms;
if (!inputIsUTC) { // convert to UTC if (!inputIsUTC)
double utcOffset = getUTCOffset(exec); result -= localTimeOffset(exec, result).offset;
result -= utcOffset;
result -= getDSTOffset(exec, result, utcOffset);
}
return result; return result;
} }
...@@ -952,12 +945,10 @@ double gregorianDateTimeToMS(ExecState* exec, const GregorianDateTime& t, double ...@@ -952,12 +945,10 @@ double gregorianDateTimeToMS(ExecState* exec, const GregorianDateTime& t, double
// input is UTC // input is UTC
void msToGregorianDateTime(ExecState* exec, double ms, bool outputIsUTC, GregorianDateTime& tm) void msToGregorianDateTime(ExecState* exec, double ms, bool outputIsUTC, GregorianDateTime& tm)
{ {
double dstOff = 0.0; LocalTimeOffset localTime(false, 0);
double utcOff = 0.0;
if (!outputIsUTC) { if (!outputIsUTC) {
utcOff = getUTCOffset(exec); localTime = localTimeOffset(exec, ms);
dstOff = getDSTOffset(exec, ms, utcOff); ms += localTime.offset;
ms += dstOff + utcOff;
} }
const int year = msToYear(ms); const int year = msToYear(ms);
...@@ -969,8 +960,8 @@ void msToGregorianDateTime(ExecState* exec, double ms, bool outputIsUTC, Gregori ...@@ -969,8 +960,8 @@ void msToGregorianDateTime(ExecState* exec, double ms, bool outputIsUTC, Gregori
tm.monthDay = dayInMonthFromDayInYear(tm.yearDay, isLeapYear(year)); tm.monthDay = dayInMonthFromDayInYear(tm.yearDay, isLeapYear(year));
tm.month = monthFromDayInYear(tm.yearDay, isLeapYear(year)); tm.month = monthFromDayInYear(tm.yearDay, isLeapYear(year));
tm.year = year - 1900; tm.year = year - 1900;
tm.isDST = dstOff != 0.0; tm.isDST = localTime.isDST;
tm.utcOffset = static_cast<long>((dstOff + utcOff) / WTF::msPerSecond); tm.utcOffset = localTime.offset / WTF::msPerSecond;
tm.timeZone = NULL; tm.timeZone = NULL;
} }
...@@ -984,11 +975,9 @@ double parseDateFromNullTerminatedCharacters(ExecState* exec, const char* dateSt ...@@ -984,11 +975,9 @@ double parseDateFromNullTerminatedCharacters(ExecState* exec, const char* dateSt
return NaN; return NaN;
// fall back to local timezone // fall back to local timezone
if (!haveTZ) { if (!haveTZ)
double utcOffset = getUTCOffset(exec); offset = calculateLocalTimeOffset(ms).offset / msPerMinute;
double dstOffset = getDSTOffset(exec, ms, utcOffset);
offset = static_cast<int>((utcOffset + dstOffset) / WTF::msPerMinute);
}
return ms - (offset * WTF::msPerMinute); return ms - (offset * WTF::msPerMinute);
} }
......
...@@ -50,6 +50,34 @@ ...@@ -50,6 +50,34 @@
#include <wtf/UnusedParam.h> #include <wtf/UnusedParam.h>
namespace WTF { namespace WTF {
struct LocalTimeOffset {
LocalTimeOffset()
: isDST(false)
, offset(0)
{
}
LocalTimeOffset(bool isDST, int offset)
: isDST(isDST)
, offset(offset)
{
}
bool operator==(const LocalTimeOffset& other)
{
return isDST == other.isDST && offset == other.offset;
}
bool operator!=(const LocalTimeOffset& other)
{
return isDST != other.isDST || offset != other.offset;
}
bool isDST;
int offset;
};
void initializeDates(); void initializeDates();
int equivalentYearForDST(int year); int equivalentYearForDST(int year);
...@@ -83,6 +111,9 @@ int dayInYear(double ms, int year); ...@@ -83,6 +111,9 @@ int dayInYear(double ms, int year);
int monthFromDayInYear(int dayInYear, bool leapYear); int monthFromDayInYear(int dayInYear, bool leapYear);
int dayInMonthFromDayInYear(int dayInYear, bool leapYear); int dayInMonthFromDayInYear(int dayInYear, bool leapYear);
// Returns combined offset in millisecond (UTC + DST).
LocalTimeOffset calculateLocalTimeOffset(double utcInMilliseconds);
} // namespace WTF } // namespace WTF
using WTF::dateToDaysFrom1970; using WTF::dateToDaysFrom1970;
...@@ -94,6 +125,8 @@ using WTF::msPerDay; ...@@ -94,6 +125,8 @@ using WTF::msPerDay;
using WTF::msPerSecond; using WTF::msPerSecond;
using WTF::msToYear; using WTF::msToYear;
using WTF::secondsPerMinute; using WTF::secondsPerMinute;
using WTF::LocalTimeOffset;
using WTF::calculateLocalTimeOffset;
#if USE(JSC) #if USE(JSC)
namespace JSC { namespace JSC {
...@@ -128,33 +161,6 @@ struct GregorianDateTime : Noncopyable { ...@@ -128,33 +161,6 @@ struct GregorianDateTime : Noncopyable {
delete [] timeZone; delete [] timeZone;
} }
GregorianDateTime(ExecState* exec, const tm& inTm)
: second(inTm.tm_sec)
, minute(inTm.tm_min)
, hour(inTm.tm_hour)
, weekDay(inTm.tm_wday)
, monthDay(inTm.tm_mday)
, yearDay(inTm.tm_yday)
, month(inTm.tm_mon)
, year(inTm.tm_year)
, isDST(inTm.tm_isdst)
{
UNUSED_PARAM(exec);
#if HAVE(TM_GMTOFF)
utcOffset = static_cast<int>(inTm.tm_gmtoff);
#else
utcOffset = static_cast<int>(getUTCOffset(exec) / WTF::msPerSecond + (isDST ? WTF::secondsPerHour : 0));
#endif
#if HAVE(TM_ZONE)
int inZoneSize = strlen(inTm.tm_zone) + 1;
timeZone = new char[inZoneSize];
strncpy(timeZone, inTm.tm_zone, inZoneSize);
#else
timeZone = 0;
#endif
}
operator tm() const operator tm() const
{ {
tm ret; tm ret;
......
Supports Markdown
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment