summaryrefslogtreecommitdiffstats
path: root/js/src/vm/DateTime.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/vm/DateTime.h')
-rw-r--r--js/src/vm/DateTime.h388
1 files changed, 388 insertions, 0 deletions
diff --git a/js/src/vm/DateTime.h b/js/src/vm/DateTime.h
new file mode 100644
index 0000000000..20feae33a8
--- /dev/null
+++ b/js/src/vm/DateTime.h
@@ -0,0 +1,388 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef vm_DateTime_h
+#define vm_DateTime_h
+
+#include "mozilla/UniquePtr.h"
+
+#include <stdint.h>
+
+#include "js/Utility.h"
+#include "threading/ExclusiveData.h"
+
+#if JS_HAS_INTL_API
+# include "mozilla/intl/ICU4CGlue.h"
+# include "mozilla/intl/TimeZone.h"
+#endif
+
+namespace JS {
+class Realm;
+}
+
+namespace js {
+
+/* Constants defined by ES5 15.9.1.10. */
+constexpr double HoursPerDay = 24;
+constexpr double MinutesPerHour = 60;
+constexpr double SecondsPerMinute = 60;
+constexpr double msPerSecond = 1000;
+constexpr double msPerMinute = msPerSecond * SecondsPerMinute;
+constexpr double msPerHour = msPerMinute * MinutesPerHour;
+
+/* ES5 15.9.1.2. */
+constexpr double msPerDay = msPerHour * HoursPerDay;
+
+/*
+ * Additional quantities not mentioned in the spec. Be careful using these!
+ * They aren't doubles and aren't defined in terms of all the other constants.
+ * If you need constants that trigger floating point semantics, you'll have to
+ * manually cast to get it.
+ */
+constexpr unsigned SecondsPerHour = 60 * 60;
+constexpr unsigned SecondsPerDay = SecondsPerHour * 24;
+
+constexpr double StartOfTime = -8.64e15;
+constexpr double EndOfTime = 8.64e15;
+
+extern bool InitDateTimeState();
+
+extern void FinishDateTimeState();
+
+enum class ResetTimeZoneMode : bool {
+ DontResetIfOffsetUnchanged,
+ ResetEvenIfOffsetUnchanged,
+};
+
+/**
+ * Engine-internal variant of JS::ResetTimeZone with an additional flag to
+ * control whether to forcibly reset all time zone data (this is the default
+ * behavior when calling JS::ResetTimeZone) or to try to reuse the previous
+ * time zone data.
+ */
+extern void ResetTimeZoneInternal(ResetTimeZoneMode mode);
+
+/**
+ * Stores date/time information, particularly concerning the current local
+ * time zone, and implements a small cache for daylight saving time offset
+ * computation.
+ *
+ * The basic idea is premised upon this fact: the DST offset never changes more
+ * than once in any thirty-day period. If we know the offset at t_0 is o_0,
+ * the offset at [t_1, t_2] is also o_0, where t_1 + 3_0 days == t_2,
+ * t_1 <= t_0, and t0 <= t2. (In other words, t_0 is always somewhere within a
+ * thirty-day range where the DST offset is constant: DST changes never occur
+ * more than once in any thirty-day period.) Therefore, if we intelligently
+ * retain knowledge of the offset for a range of dates (which may vary over
+ * time), and if requests are usually for dates within that range, we can often
+ * provide a response without repeated offset calculation.
+ *
+ * Our caching strategy is as follows: on the first request at date t_0 compute
+ * the requested offset o_0. Save { start: t_0, end: t_0, offset: o_0 } as the
+ * cache's state. Subsequent requests within that range are straightforwardly
+ * handled. If a request for t_i is far outside the range (more than thirty
+ * days), compute o_i = dstOffset(t_i) and save { start: t_i, end: t_i,
+ * offset: t_i }. Otherwise attempt to *overextend* the range to either
+ * [start - 30d, end] or [start, end + 30d] as appropriate to encompass
+ * t_i. If the offset o_i30 is the same as the cached offset, extend the
+ * range. Otherwise the over-guess crossed a DST change -- compute
+ * o_i = dstOffset(t_i) and either extend the original range (if o_i == offset)
+ * or start a new one beneath/above the current one with o_i30 as the offset.
+ *
+ * This cache strategy results in 0 to 2 DST offset computations. The naive
+ * always-compute strategy is 1 computation, and since cache maintenance is a
+ * handful of integer arithmetic instructions the speed difference between
+ * always-1 and 1-with-cache is negligible. Caching loses if two computations
+ * happen: when the date is within 30 days of the cached range and when that
+ * 30-day range crosses a DST change. This is relatively uncommon. Further,
+ * instances of such are often dominated by in-range hits, so caching is an
+ * overall slight win.
+ *
+ * Why 30 days? For correctness the duration must be smaller than any possible
+ * duration between DST changes. Past that, note that 1) a large duration
+ * increases the likelihood of crossing a DST change while reducing the number
+ * of cache misses, and 2) a small duration decreases the size of the cached
+ * range while producing more misses. Using a month as the interval change is
+ * a balance between these two that tries to optimize for the calendar month at
+ * a time that a site might display. (One could imagine an adaptive duration
+ * that accommodates near-DST-change dates better; we don't believe the
+ * potential win from better caching offsets the loss from extra complexity.)
+ */
+class DateTimeInfo {
+ public:
+ // Whether we should resist fingerprinting. For realms in RFP mode a separate
+ // DateTimeInfo instance is used that is always in the UTC time zone.
+ enum class ShouldRFP { No, Yes };
+
+ private:
+ static ExclusiveData<DateTimeInfo>* instance;
+ static ExclusiveData<DateTimeInfo>* instanceRFP;
+
+ friend class ExclusiveData<DateTimeInfo>;
+
+ friend bool InitDateTimeState();
+ friend void FinishDateTimeState();
+
+ explicit DateTimeInfo(bool shouldResistFingerprinting);
+ ~DateTimeInfo();
+
+ static auto acquireLockWithValidTimeZone(ShouldRFP shouldRFP) {
+ auto guard =
+ shouldRFP == ShouldRFP::Yes ? instanceRFP->lock() : instance->lock();
+ if (guard->timeZoneStatus_ != TimeZoneStatus::Valid) {
+ guard->updateTimeZone();
+ }
+ return guard;
+ }
+
+ public:
+ static ShouldRFP shouldRFP(JS::Realm* realm);
+
+ // The spec implicitly assumes DST and time zone adjustment information
+ // never change in the course of a function -- sometimes even across
+ // reentrancy. So make critical sections as narrow as possible.
+
+ /**
+ * Get the DST offset in milliseconds at a UTC time. This is usually
+ * either 0 or |msPerSecond * SecondsPerHour|, but at least one exotic time
+ * zone (Lord Howe Island, Australia) has a fractional-hour offset, just to
+ * keep things interesting.
+ */
+ static int32_t getDSTOffsetMilliseconds(ShouldRFP shouldRFP,
+ int64_t utcMilliseconds) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->internalGetDSTOffsetMilliseconds(utcMilliseconds);
+ }
+
+ /**
+ * The offset in seconds from the current UTC time to the current local
+ * standard time (i.e. not including any offset due to DST) as computed by the
+ * operating system.
+ */
+ static int32_t utcToLocalStandardOffsetSeconds(ShouldRFP shouldRFP) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->utcToLocalStandardOffsetSeconds_;
+ }
+
+#if JS_HAS_INTL_API
+ enum class TimeZoneOffset { UTC, Local };
+
+ /**
+ * Return the time zone offset, including DST, in milliseconds at the
+ * given time. The input time can be either at UTC or at local time.
+ */
+ static int32_t getOffsetMilliseconds(ShouldRFP shouldRFP,
+ int64_t milliseconds,
+ TimeZoneOffset offset) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->internalGetOffsetMilliseconds(milliseconds, offset);
+ }
+
+ /**
+ * Copy the display name for the current time zone at the given time,
+ * localized for the specified locale, into the supplied buffer. If the
+ * buffer is too small, an empty string is stored. The stored display name
+ * is null-terminated in any case.
+ */
+ static bool timeZoneDisplayName(ShouldRFP shouldRFP, char16_t* buf,
+ size_t buflen, int64_t utcMilliseconds,
+ const char* locale) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->internalTimeZoneDisplayName(buf, buflen, utcMilliseconds,
+ locale);
+ }
+
+ /**
+ * Copy the identifier for the current time zone to the provided resizable
+ * buffer.
+ */
+ template <typename B>
+ static mozilla::intl::ICUResult timeZoneId(ShouldRFP shouldRFP, B& buffer) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->timeZone()->GetId(buffer);
+ }
+
+ /**
+ * A number indicating the raw offset from GMT in milliseconds.
+ */
+ static mozilla::Result<int32_t, mozilla::intl::ICUError> getRawOffsetMs(
+ ShouldRFP shouldRFP) {
+ auto guard = acquireLockWithValidTimeZone(shouldRFP);
+ return guard->timeZone()->GetRawOffsetMs();
+ }
+#else
+ /**
+ * Return the local time zone adjustment (ES2019 20.3.1.7) as computed by
+ * the operating system.
+ */
+ static int32_t localTZA(ShouldRFP shouldRFP) {
+ return utcToLocalStandardOffsetSeconds(shouldRFP) * msPerSecond;
+ }
+#endif /* JS_HAS_INTL_API */
+
+ private:
+ // The method below should only be called via js::ResetTimeZoneInternal().
+ friend void js::ResetTimeZoneInternal(ResetTimeZoneMode);
+
+ static void resetTimeZone(ResetTimeZoneMode mode) {
+ {
+ auto guard = instance->lock();
+ guard->internalResetTimeZone(mode);
+ }
+ {
+ // Only needed to initialize the default state and any later call will
+ // perform an unnecessary reset.
+ auto guard = instanceRFP->lock();
+ guard->internalResetTimeZone(mode);
+ }
+ }
+
+ struct RangeCache {
+ // Start and end offsets in seconds describing the current and the
+ // last cached range.
+ int64_t startSeconds, endSeconds;
+ int64_t oldStartSeconds, oldEndSeconds;
+
+ // The current and the last cached offset in milliseconds.
+ int32_t offsetMilliseconds;
+ int32_t oldOffsetMilliseconds;
+
+ void reset();
+
+ void sanityCheck();
+ };
+
+ bool shouldResistFingerprinting_;
+
+ enum class TimeZoneStatus : uint8_t { Valid, NeedsUpdate, UpdateIfChanged };
+
+ TimeZoneStatus timeZoneStatus_;
+
+ /**
+ * The offset in seconds from the current UTC time to the current local
+ * standard time (i.e. not including any offset due to DST) as computed by the
+ * operating system.
+ *
+ * Cached because retrieving this dynamically is Slow, and a certain venerable
+ * benchmark which shall not be named depends on it being fast.
+ *
+ * SpiderMonkey occasionally and arbitrarily updates this value from the
+ * system time zone to attempt to keep this reasonably up-to-date. If
+ * temporary inaccuracy can't be tolerated, JSAPI clients may call
+ * JS::ResetTimeZone to forcibly sync this with the system time zone.
+ *
+ * In most cases this value is consistent with the raw time zone offset as
+ * returned by the ICU default time zone (`icu::TimeZone::getRawOffset()`),
+ * but it is possible to create cases where the operating system default time
+ * zone differs from the ICU default time zone. For example ICU doesn't
+ * support the full range of TZ environment variable settings, which can
+ * result in <ctime> returning a different time zone than what's returned by
+ * ICU. One example is "TZ=WGT3WGST,M3.5.0/-2,M10.5.0/-1", where <ctime>
+ * returns -3 hours as the local offset, but ICU flat out rejects the TZ value
+ * and instead infers the default time zone via "/etc/localtime" (on Unix).
+ * This offset can also differ from ICU when the operating system and ICU use
+ * different tzdata versions and the time zone rules of the current system
+ * time zone have changed. Or, on Windows, when the Windows default time zone
+ * can't be mapped to a IANA time zone, see for example
+ * <https://unicode-org.atlassian.net/browse/ICU-13845>.
+ *
+ * When ICU is exclusively used for time zone computations, that means when
+ * |JS_HAS_INTL_API| is true, this field is only used to detect system default
+ * time zone changes. It must not be used to convert between local and UTC
+ * time, because, as outlined above, this could lead to different results when
+ * compared to ICU.
+ */
+ int32_t utcToLocalStandardOffsetSeconds_;
+
+ RangeCache dstRange_; // UTC-based ranges
+
+#if JS_HAS_INTL_API
+ // Use the full date-time range when we can use mozilla::intl::TimeZone.
+ static constexpr int64_t MinTimeT =
+ static_cast<int64_t>(StartOfTime / msPerSecond);
+ static constexpr int64_t MaxTimeT =
+ static_cast<int64_t>(EndOfTime / msPerSecond);
+
+ RangeCache utcRange_; // localtime-based ranges
+ RangeCache localRange_; // UTC-based ranges
+
+ /**
+ * The current time zone. Lazily constructed to avoid potential I/O access
+ * when initializing this class.
+ */
+ mozilla::UniquePtr<mozilla::intl::TimeZone> timeZone_;
+
+ /**
+ * Cached names of the standard and daylight savings display names of the
+ * current time zone for the default locale.
+ */
+ JS::UniqueChars locale_;
+ JS::UniqueTwoByteChars standardName_;
+ JS::UniqueTwoByteChars daylightSavingsName_;
+#else
+ // Restrict the data-time range to the minimum required time_t range as
+ // specified in POSIX. Most operating systems support 64-bit time_t
+ // values, but we currently still have some configurations which use
+ // 32-bit time_t, e.g. the ARM simulator on 32-bit Linux (bug 1406993).
+ // Bug 1406992 explores to use 64-bit time_t when supported by the
+ // underlying operating system.
+ static constexpr int64_t MinTimeT = 0; /* time_t 01/01/1970 */
+ static constexpr int64_t MaxTimeT = 2145830400; /* time_t 12/31/2037 */
+#endif /* JS_HAS_INTL_API */
+
+ static constexpr int64_t RangeExpansionAmount = 30 * SecondsPerDay;
+
+ void internalResetTimeZone(ResetTimeZoneMode mode);
+
+ void updateTimeZone();
+
+ void internalResyncICUDefaultTimeZone();
+
+ int64_t toClampedSeconds(int64_t milliseconds);
+
+ using ComputeFn = int32_t (DateTimeInfo::*)(int64_t);
+
+ /**
+ * Get or compute an offset value for the requested seconds value.
+ */
+ int32_t getOrComputeValue(RangeCache& range, int64_t seconds,
+ ComputeFn compute);
+
+ /**
+ * Compute the DST offset at the given UTC time in seconds from the epoch.
+ * (getDSTOffsetMilliseconds attempts to return a cached value from the
+ * dstRange_ member, but in case of a cache miss it calls this method.)
+ */
+ int32_t computeDSTOffsetMilliseconds(int64_t utcSeconds);
+
+ int32_t internalGetDSTOffsetMilliseconds(int64_t utcMilliseconds);
+
+#if JS_HAS_INTL_API
+ /**
+ * Compute the UTC offset in milliseconds for the given local time. Called
+ * by internalGetOffsetMilliseconds on a cache miss.
+ */
+ int32_t computeUTCOffsetMilliseconds(int64_t localSeconds);
+
+ /**
+ * Compute the local time offset in milliseconds for the given UTC time.
+ * Called by internalGetOffsetMilliseconds on a cache miss.
+ */
+ int32_t computeLocalOffsetMilliseconds(int64_t utcSeconds);
+
+ int32_t internalGetOffsetMilliseconds(int64_t milliseconds,
+ TimeZoneOffset offset);
+
+ bool internalTimeZoneDisplayName(char16_t* buf, size_t buflen,
+ int64_t utcMilliseconds, const char* locale);
+
+ mozilla::intl::TimeZone* timeZone();
+#endif /* JS_HAS_INTL_API */
+};
+
+} /* namespace js */
+
+#endif /* vm_DateTime_h */