blob: 98b421053e0440b08f2226e1623baca7e2a043df [file] [log] [blame]
/*
* Copyright (C) 1999-2000 Harri Porten (porten@kde.org)
* Copyright (C) 2006-2023 Apple Inc. All rights reserved.
* Copyright (C) 2009 Google Inc. All rights reserved.
* Copyright (C) 2012 the V8 project authors. All rights reserved.
* Copyright (C) 2010 Research In Motion Limited. All rights reserved.
*
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is Mozilla Communicator client code, released
* March 31, 1998.
*
* The Initial Developer of the Original Code is
* Netscape Communications Corporation.
* Portions created by the Initial Developer are Copyright (C) 1998
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
*/
#pragma once
#include "DateInstanceCache.h"
#include <wtf/DateMath.h>
#include <wtf/GregorianDateTime.h>
#include <wtf/SaturatedArithmetic.h>
namespace JSC {
class JSGlobalObject;
class OpaqueICUTimeZone;
class VM;
static constexpr double minECMAScriptTime = -8.64E15;
#if PLATFORM(COCOA)
extern JS_EXPORT_PRIVATE std::atomic<uint64_t> lastTimeZoneID;
#endif
// We do not expose icu::TimeZone in this header file. And we cannot use icu::TimeZone forward declaration
// because icu namespace can be an alias to icu$verNum namespace.
struct OpaqueICUTimeZoneDeleter {
JS_EXPORT_PRIVATE void operator()(OpaqueICUTimeZone*);
};
struct LocalTimeOffsetCache {
LocalTimeOffsetCache() = default;
bool isEmpty() { return start > end; }
LocalTimeOffset offset { };
int64_t start { WTF::Int64Milliseconds::maxECMAScriptTime };
int64_t end { WTF::Int64Milliseconds::minECMAScriptTime };
uint64_t epoch { 0 };
};
class DateCache {
WTF_MAKE_FAST_ALLOCATED;
WTF_MAKE_NONCOPYABLE(DateCache);
public:
DateCache();
~DateCache();
bool hasTimeZoneChange()
{
#if PLATFORM(COCOA)
return m_cachedTimezoneID != lastTimeZoneID;
#else
return true; // always force a time zone check.
#endif
}
void resetIfNecessary()
{
#if PLATFORM(COCOA)
if (LIKELY(!hasTimeZoneChange()))
return;
m_cachedTimezoneID = lastTimeZoneID;
#endif
resetIfNecessarySlow();
}
JS_EXPORT_PRIVATE void resetIfNecessarySlow();
String defaultTimeZone();
String timeZoneDisplayName(bool isDST);
Ref<DateInstanceData> cachedDateInstanceData(double millisecondsFromEpoch);
void msToGregorianDateTime(double millisecondsFromEpoch, WTF::TimeType outputTimeType, GregorianDateTime&);
double gregorianDateTimeToMS(const GregorianDateTime&, double milliseconds, WTF::TimeType);
double localTimeToMS(double milliseconds, WTF::TimeType);
JS_EXPORT_PRIVATE double parseDate(JSGlobalObject*, VM&, const WTF::String&);
static void timeZoneChanged();
private:
class DSTCache {
public:
static constexpr unsigned cacheSize = 32;
// The implementation relies on the fact that no time zones have
// more than one daylight savings offset change per 19 days.
// In Egypt in 2010 they decided to suspend DST during Ramadan. This
// led to a short interval where DST is in effect from September 10 to
// September 30.
static constexpr int64_t defaultDSTDeltaInMilliseconds = 19 * WTF::Int64Milliseconds::secondsPerDay * 1000;
DSTCache()
: m_before(m_entries.data())
, m_after(m_entries.data() + 1)
{
}
uint64_t bumpEpoch()
{
++m_epoch;
return m_epoch;
}
void reset()
{
m_entries.fill(LocalTimeOffsetCache { });
m_before = m_entries.data();
m_after = m_entries.data() + 1;
m_epoch = 0;
}
LocalTimeOffset localTimeOffset(DateCache&, int64_t millisecondsFromEpoch, WTF::TimeType);
private:
LocalTimeOffsetCache* leastRecentlyUsed(LocalTimeOffsetCache* exclude);
std::tuple<LocalTimeOffsetCache*, LocalTimeOffsetCache*> probe(int64_t millisecondsFromEpoch);
void extendTheAfterCache(int64_t millisecondsFromEpoch, LocalTimeOffset);
uint64_t m_epoch { 0 };
std::array<LocalTimeOffsetCache, cacheSize> m_entries { };
LocalTimeOffsetCache* m_before;
LocalTimeOffsetCache* m_after;
};
struct YearMonthDayCache {
int32_t m_days { 0 };
int32_t m_year { 0 };
int32_t m_month { 0 };
int32_t m_day { 0 };
};
void timeZoneCacheSlow();
LocalTimeOffset localTimeOffset(int64_t millisecondsFromEpoch, WTF::TimeType inputTimeType = WTF::UTCTime)
{
static_assert(!WTF::UTCTime);
static_assert(WTF::LocalTime == 1);
return m_caches[static_cast<unsigned>(inputTimeType)].localTimeOffset(*this, millisecondsFromEpoch, inputTimeType);
}
LocalTimeOffset calculateLocalTimeOffset(double millisecondsFromEpoch, WTF::TimeType inputTimeType);
OpaqueICUTimeZone* timeZoneCache()
{
if (!m_timeZoneCache)
timeZoneCacheSlow();
return m_timeZoneCache.get();
}
std::tuple<int32_t, int32_t, int32_t> yearMonthDayFromDaysWithCache(int32_t days);
std::unique_ptr<OpaqueICUTimeZone, OpaqueICUTimeZoneDeleter> m_timeZoneCache;
std::array<DSTCache, 2> m_caches;
std::optional<YearMonthDayCache> m_yearMonthDayCache;
String m_cachedDateString;
double m_cachedDateStringValue;
DateInstanceCache m_dateInstanceCache;
uint64_t m_cachedTimezoneID { 0 };
String m_timeZoneStandardDisplayNameCache;
String m_timeZoneDSTDisplayNameCache;
};
ALWAYS_INLINE bool isUTCEquivalent(StringView timeZone)
{
return timeZone == "Etc/UTC"_s || timeZone == "Etc/GMT"_s;
}
} // namespace JSC