| Index: src/date.js
|
| diff --git a/src/date.js b/src/date.js
|
| index 2e5d6e22c40d8b5bdec0e671c081d3aabef8bbda..75edf6d32d17c364023c39cb319c937c1421286d 100644
|
| --- a/src/date.js
|
| +++ b/src/date.js
|
| @@ -44,172 +44,6 @@ function ThrowDateTypeError() {
|
| throw new $TypeError('this is not a Date object.');
|
| }
|
|
|
| -// ECMA 262 - 5.2
|
| -function Modulo(value, remainder) {
|
| - var mod = value % remainder;
|
| - // All uses of this function for dates should produce a Smi.
|
| - return (mod >= 0 ? mod : mod + remainder) | 0;
|
| -}
|
| -
|
| -
|
| -function TimeWithinDay(time) {
|
| - return Modulo(time, msPerDay);
|
| -}
|
| -
|
| -
|
| -// ECMA 262 - 15.9.1.3
|
| -function DaysInYear(year) {
|
| - if (year % 4 != 0) return 365;
|
| - if ((year % 100 == 0) && (year % 400 != 0)) return 365;
|
| - return 366;
|
| -}
|
| -
|
| -
|
| -function DayFromYear(year) {
|
| - return 365 * (year-1970)
|
| - + FLOOR((year-1969)/4)
|
| - - FLOOR((year-1901)/100)
|
| - + FLOOR((year-1601)/400);
|
| -}
|
| -
|
| -
|
| -function TimeFromYear(year) {
|
| - return msPerDay * DayFromYear(year);
|
| -}
|
| -
|
| -
|
| -function InLeapYear(time) {
|
| - return DaysInYear(YearFromTime(time)) - 365; // Returns 1 or 0.
|
| -}
|
| -
|
| -
|
| -// ECMA 262 - 15.9.1.9
|
| -function EquivalentYear(year) {
|
| - // Returns an equivalent year in the range [2008-2035] matching
|
| - // - leap year.
|
| - // - week day of first day.
|
| - var time = TimeFromYear(year);
|
| - var recent_year = (InLeapYear(time) == 0 ? 1967 : 1956) +
|
| - (WeekDay(time) * 12) % 28;
|
| - // Find the year in the range 2008..2037 that is equivalent mod 28.
|
| - // Add 3*28 to give a positive argument to the modulus operator.
|
| - return 2008 + (recent_year + 3*28 - 2008) % 28;
|
| -}
|
| -
|
| -
|
| -function EquivalentTime(t) {
|
| - // The issue here is that some library calls don't work right for dates
|
| - // that cannot be represented using a non-negative signed 32 bit integer
|
| - // (measured in whole seconds based on the 1970 epoch).
|
| - // We solve this by mapping the time to a year with same leap-year-ness
|
| - // and same starting day for the year. The ECMAscript specification says
|
| - // we must do this, but for compatibility with other browsers, we use
|
| - // the actual year if it is in the range 1970..2037
|
| - if (t >= 0 && t <= 2.1e12) return t;
|
| -
|
| - var day = MakeDay(EquivalentYear(YearFromTime(t)),
|
| - MonthFromTime(t),
|
| - DateFromTime(t));
|
| - return MakeDate(day, TimeWithinDay(t));
|
| -}
|
| -
|
| -
|
| -// local_time_offset is initialized when the DST_offset_cache is missed.
|
| -// It must not be used until after a call to DaylightSavingsOffset().
|
| -// In this way, only one check, for a DST cache miss, is needed.
|
| -var local_time_offset;
|
| -
|
| -
|
| -// Because computing the DST offset is an expensive operation,
|
| -// we keep a cache of the last computed DST offset along with a time interval
|
| -// where we know the cache is valid.
|
| -// When the cache is valid, local_time_offset is also valid.
|
| -var DST_offset_cache = {
|
| - // Cached DST offset.
|
| - offset: 0,
|
| - // Time interval where the cached offset is valid.
|
| - start: 0, end: -1,
|
| - // Size of next interval expansion.
|
| - increment: 0,
|
| - initial_increment: 19 * msPerDay
|
| -};
|
| -
|
| -
|
| -// NOTE: 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.
|
| -//
|
| -// If this function is called with NaN it returns NaN.
|
| -function DaylightSavingsOffset(t) {
|
| - // Load the cache object from the builtins object.
|
| - var cache = DST_offset_cache;
|
| -
|
| - // Cache the start and the end in local variables for fast access.
|
| - var start = cache.start;
|
| - var end = cache.end;
|
| -
|
| - if (start <= t) {
|
| - // If the time fits in the cached interval, return the cached offset.
|
| - if (t <= end) return cache.offset;
|
| -
|
| - // If the cache misses, the local_time_offset may not be initialized.
|
| - if (IS_UNDEFINED(local_time_offset)) {
|
| - local_time_offset = %DateLocalTimeOffset();
|
| - }
|
| -
|
| - // Compute a possible new interval end.
|
| - var new_end = end + cache.increment;
|
| -
|
| - if (t <= new_end) {
|
| - var end_offset = %DateDaylightSavingsOffset(EquivalentTime(new_end));
|
| - if (cache.offset == end_offset) {
|
| - // If the offset at the end of the new interval still matches
|
| - // the offset in the cache, we grow the cached time interval
|
| - // and return the offset.
|
| - cache.end = new_end;
|
| - cache.increment = cache.initial_increment;
|
| - return end_offset;
|
| - } else {
|
| - var offset = %DateDaylightSavingsOffset(EquivalentTime(t));
|
| - if (offset == end_offset) {
|
| - // 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 = t;
|
| - cache.end = new_end;
|
| - cache.increment = cache.initial_increment;
|
| - } 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 = t;
|
| - }
|
| - // Update the offset in the cache and return it.
|
| - cache.offset = offset;
|
| - return offset;
|
| - }
|
| - }
|
| - }
|
| -
|
| - // If the cache misses, the local_time_offset may not be initialized.
|
| - if (IS_UNDEFINED(local_time_offset)) {
|
| - local_time_offset = %DateLocalTimeOffset();
|
| - }
|
| - // Compute the DST offset for the time and shrink the cache interval
|
| - // to only contain the time. This allows fast repeated DST offset
|
| - // computations for the same time.
|
| - var offset = %DateDaylightSavingsOffset(EquivalentTime(t));
|
| - cache.offset = offset;
|
| - cache.start = cache.end = t;
|
| - cache.increment = cache.initial_increment;
|
| - return offset;
|
| -}
|
| -
|
|
|
| var timezone_cache_time = $NaN;
|
| var timezone_cache_timezone;
|
| @@ -219,57 +53,18 @@ function LocalTimezone(t) {
|
| if (t == timezone_cache_time) {
|
| return timezone_cache_timezone;
|
| }
|
| - var timezone = %DateLocalTimezone(EquivalentTime(t));
|
| + var timezone = %DateLocalTimezone(t);
|
| timezone_cache_time = t;
|
| timezone_cache_timezone = timezone;
|
| return timezone;
|
| }
|
|
|
|
|
| -function WeekDay(time) {
|
| - return Modulo(DAY(time) + 4, 7);
|
| -}
|
| -
|
| -
|
| -function LocalTime(time) {
|
| - if (NUMBER_IS_NAN(time)) return time;
|
| - // DaylightSavingsOffset called before local_time_offset used.
|
| - return time + DaylightSavingsOffset(time) + local_time_offset;
|
| -}
|
| -
|
| -
|
| -var ltcache = {
|
| - key: null,
|
| - val: null
|
| -};
|
| -
|
| -function LocalTimeNoCheck(time) {
|
| - var ltc = ltcache;
|
| - if (%_ObjectEquals(time, ltc.key)) return ltc.val;
|
| -
|
| - // Inline the DST offset cache checks for speed.
|
| - // The cache is hit, or DaylightSavingsOffset is called,
|
| - // before local_time_offset is used.
|
| - var cache = DST_offset_cache;
|
| - if (cache.start <= time && time <= cache.end) {
|
| - var dst_offset = cache.offset;
|
| - } else {
|
| - var dst_offset = DaylightSavingsOffset(time);
|
| - }
|
| - ltc.key = time;
|
| - return (ltc.val = time + local_time_offset + dst_offset);
|
| -}
|
| -
|
| -
|
| function UTC(time) {
|
| if (NUMBER_IS_NAN(time)) return time;
|
| // local_time_offset is needed before the call to DaylightSavingsOffset,
|
| // so it may be uninitialized.
|
| - if (IS_UNDEFINED(local_time_offset)) {
|
| - local_time_offset = %DateLocalTimeOffset();
|
| - }
|
| - var tmp = time - local_time_offset;
|
| - return tmp - DaylightSavingsOffset(tmp);
|
| + return %DateToUTC(time);
|
| }
|
|
|
|
|
| @@ -292,48 +87,6 @@ function TimeInYear(year) {
|
| }
|
|
|
|
|
| -var ymd_from_time_cache = [1970, 0, 1];
|
| -var ymd_from_time_cached_time = 0;
|
| -
|
| -function YearFromTime(t) {
|
| - if (t !== ymd_from_time_cached_time) {
|
| - if (!$isFinite(t)) {
|
| - return $NaN;
|
| - }
|
| -
|
| - %DateYMDFromTime(t, ymd_from_time_cache);
|
| - ymd_from_time_cached_time = t;
|
| - }
|
| -
|
| - return ymd_from_time_cache[0];
|
| -}
|
| -
|
| -function MonthFromTime(t) {
|
| - if (t !== ymd_from_time_cached_time) {
|
| - if (!$isFinite(t)) {
|
| - return $NaN;
|
| - }
|
| - %DateYMDFromTime(t, ymd_from_time_cache);
|
| - ymd_from_time_cached_time = t;
|
| - }
|
| -
|
| - return ymd_from_time_cache[1];
|
| -}
|
| -
|
| -function DateFromTime(t) {
|
| - if (t !== ymd_from_time_cached_time) {
|
| - if (!$isFinite(t)) {
|
| - return $NaN;
|
| - }
|
| -
|
| - %DateYMDFromTime(t, ymd_from_time_cache);
|
| - ymd_from_time_cached_time = t;
|
| - }
|
| -
|
| - return ymd_from_time_cache[2];
|
| -}
|
| -
|
| -
|
| // Compute number of days given a year, month, date.
|
| // Note that month and date can lie outside the normal range.
|
| // For example:
|
| @@ -384,9 +137,6 @@ function TimeClip(time) {
|
| var Date_cache = {
|
| // Cached time value.
|
| time: $NaN,
|
| - // Cached year when interpreting the time as a local time. Only
|
| - // valid when the time matches cached time.
|
| - year: $NaN,
|
| // String input for which the cached time is valid.
|
| string: null
|
| };
|
| @@ -403,11 +153,10 @@ var Date_cache = {
|
| var value;
|
| if (argc == 0) {
|
| value = %DateCurrentTime();
|
| -
|
| + SET_UTC_DATE_VALUE(this, value);
|
| } else if (argc == 1) {
|
| if (IS_NUMBER(year)) {
|
| - value = TimeClip(year);
|
| -
|
| + value = year;
|
| } else if (IS_STRING(year)) {
|
| // Probe the Date cache. If we already have a time value for the
|
| // given time, we re-use that instead of parsing the string again.
|
| @@ -418,7 +167,6 @@ var Date_cache = {
|
| value = DateParse(year);
|
| if (!NUMBER_IS_NAN(value)) {
|
| cache.time = value;
|
| - cache.year = YearFromTime(LocalTimeNoCheck(value));
|
| cache.string = year;
|
| }
|
| }
|
| @@ -432,9 +180,9 @@ var Date_cache = {
|
| // which is the default for everything else than Date objects.
|
| // This makes us behave like KJS and SpiderMonkey.
|
| var time = ToPrimitive(year, NUMBER_HINT);
|
| - value = IS_STRING(time) ? DateParse(time) : TimeClip(ToNumber(time));
|
| + value = IS_STRING(time) ? DateParse(time) : ToNumber(time);
|
| }
|
| -
|
| + SET_UTC_DATE_VALUE(this, value);
|
| } else {
|
| year = ToNumber(year);
|
| month = ToNumber(month);
|
| @@ -448,40 +196,12 @@ var Date_cache = {
|
| TO_INTEGER(year) <= 99) ? 1900 + TO_INTEGER(year) : year;
|
| var day = MakeDay(year, month, date);
|
| var time = MakeTime(hours, minutes, seconds, ms);
|
| - value = TimeClip(UTC(MakeDate(day, time)));
|
| + value = MakeDate(day, time);
|
| + SET_LOCAL_DATE_VALUE(this, value);
|
| }
|
| - SET_DATE_VALUE(this, value);
|
| });
|
|
|
|
|
| -function ResetDate(date, time) {
|
| - SET_DATE_VALUE(date, time);
|
| - // Cache aggressively in case of a reset - we will typically use most fields.
|
| - if (NUMBER_IS_NAN(time)) {
|
| - SET_DATE_LOCAL(date, time);
|
| - SET_DATE_YEAR(date, time);
|
| - SET_DATE_MONTH(date, time);
|
| - SET_DATE_DAY(date, time);
|
| - SET_DATE_HOUR(date, time);
|
| - SET_DATE_MIN(date, time);
|
| - SET_DATE_SEC(date, time);
|
| - SET_DATE_WEEKDAY(date, time);
|
| - } else {
|
| - var local = LocalTimeNoCheck(time);
|
| - SET_DATE_LOCAL(date, local_time_offset);
|
| - SET_DATE_YEAR(date, YearFromTime(local));
|
| - SET_DATE_MONTH(date, MonthFromTime(local));
|
| - SET_DATE_DAY(date, DateFromTime(local));
|
| - SET_DATE_HOUR(date, HOUR_FROM_TIME(local));
|
| - SET_DATE_MIN(date, MIN_FROM_TIME(local));
|
| - SET_DATE_SEC(date, SEC_FROM_TIME(local));
|
| - SET_DATE_WEEKDAY(date, WeekDay(local));
|
| - }
|
| -
|
| - return time;
|
| -}
|
| -
|
| -
|
| %FunctionSetPrototype($Date, new $Date($NaN));
|
|
|
|
|
| @@ -495,11 +215,11 @@ function TwoDigitString(value) {
|
| }
|
|
|
|
|
| -function DateString(time) {
|
| - return WeekDays[WeekDay(time)] + ' '
|
| - + Months[MonthFromTime(time)] + ' '
|
| - + TwoDigitString(DateFromTime(time)) + ' '
|
| - + YearFromTime(time);
|
| +function DateString(date) {
|
| + return WeekDays[LOCAL_WEEKDAY(date)] + ' '
|
| + + Months[LOCAL_MONTH(date)] + ' '
|
| + + TwoDigitString(LOCAL_DAY(date)) + ' '
|
| + + LOCAL_YEAR(date);
|
| }
|
|
|
|
|
| @@ -509,38 +229,32 @@ var LongMonths = ['January', 'February', 'March', 'April', 'May', 'June',
|
| 'July', 'August', 'September', 'October', 'November', 'December'];
|
|
|
|
|
| -function LongDateString(time) {
|
| - return LongWeekDays[WeekDay(time)] + ', '
|
| - + LongMonths[MonthFromTime(time)] + ' '
|
| - + TwoDigitString(DateFromTime(time)) + ', '
|
| - + YearFromTime(time);
|
| +function LongDateString(date) {
|
| + return LongWeekDays[LOCAL_WEEKDAY(date)] + ', '
|
| + + LongMonths[LOCAL_MONTH(date)] + ' '
|
| + + TwoDigitString(LOCAL_DAY(date)) + ', '
|
| + + LOCAL_YEAR(date);
|
| }
|
|
|
|
|
| -function TimeString(time) {
|
| - return TwoDigitString(HOUR_FROM_TIME(time)) + ':'
|
| - + TwoDigitString(MIN_FROM_TIME(time)) + ':'
|
| - + TwoDigitString(SEC_FROM_TIME(time));
|
| +function TimeString(date) {
|
| + return TwoDigitString(LOCAL_HOUR(date)) + ':'
|
| + + TwoDigitString(LOCAL_MIN(date)) + ':'
|
| + + TwoDigitString(LOCAL_SEC(date));
|
| }
|
|
|
|
|
| -function LocalTimezoneString(time) {
|
| - var old_timezone = timezone_cache_timezone;
|
| - var timezone = LocalTimezone(time);
|
| - if (old_timezone && timezone != old_timezone) {
|
| - // If the timezone string has changed from the one that we cached,
|
| - // the local time offset may now be wrong. So we need to update it
|
| - // and try again.
|
| - local_time_offset = %DateLocalTimeOffset();
|
| - // We also need to invalidate the DST cache as the new timezone may have
|
| - // different DST times.
|
| - var dst_cache = DST_offset_cache;
|
| - dst_cache.start = 0;
|
| - dst_cache.end = -1;
|
| - }
|
| +function TimeStringUTC(date) {
|
| + return TwoDigitString(UTC_HOUR(date)) + ':'
|
| + + TwoDigitString(UTC_MIN(date)) + ':'
|
| + + TwoDigitString(UTC_SEC(date));
|
| +}
|
| +
|
|
|
| - var timezoneOffset =
|
| - (DaylightSavingsOffset(time) + local_time_offset) / msPerMinute;
|
| +function LocalTimezoneString(date) {
|
| + var timezone = LocalTimezone(UTC_DATE_VALUE(date));
|
| +
|
| + var timezoneOffset = -TIMEZONE_OFFSET(date);
|
| var sign = (timezoneOffset >= 0) ? 1 : -1;
|
| var hours = FLOOR((sign * timezoneOffset)/60);
|
| var min = FLOOR((sign * timezoneOffset)%60);
|
| @@ -550,8 +264,8 @@ function LocalTimezoneString(time) {
|
| }
|
|
|
|
|
| -function DatePrintString(time) {
|
| - return DateString(time) + ' ' + TimeString(time);
|
| +function DatePrintString(date) {
|
| + return DateString(date) + ' ' + TimeString(date);
|
| }
|
|
|
| // -------------------------------------------------------------------
|
| @@ -604,27 +318,30 @@ function DateNow() {
|
|
|
| // ECMA 262 - 15.9.5.2
|
| function DateToString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this)
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| - var time_zone_string = LocalTimezoneString(t); // May update local offset.
|
| - return DatePrintString(LocalTimeNoCheck(t)) + time_zone_string;
|
| + var time_zone_string = LocalTimezoneString(this)
|
| + return DatePrintString(this) + time_zone_string;
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.3
|
| function DateToDateString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| - return DateString(LocalTimeNoCheck(t));
|
| + return DateString(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.4
|
| function DateToTimeString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| - var time_zone_string = LocalTimezoneString(t); // May update local offset.
|
| - return TimeString(LocalTimeNoCheck(t)) + time_zone_string;
|
| + var time_zone_string = LocalTimezoneString(this);
|
| + return TimeString(this) + time_zone_string;
|
| }
|
|
|
|
|
| @@ -636,372 +353,388 @@ function DateToLocaleString() {
|
|
|
| // ECMA 262 - 15.9.5.6
|
| function DateToLocaleDateString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| - return LongDateString(LocalTimeNoCheck(t));
|
| + return LongDateString(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.7
|
| function DateToLocaleTimeString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| - var lt = LocalTimeNoCheck(t);
|
| - return TimeString(lt);
|
| + return TimeString(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.8
|
| function DateValueOf() {
|
| - return DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + return UTC_DATE_VALUE(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.9
|
| function DateGetTime() {
|
| - return DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + return UTC_DATE_VALUE(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.10
|
| function DateGetFullYear() {
|
| - var t = DATE_YEAR(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - t = DATE_VALUE_UNCHECKED(this);
|
| - var cache = Date_cache;
|
| - if (cache.time === t) return cache.year;
|
| - t = LocalTimeNoCheck(t);
|
| - if (!NUMBER_IS_NAN(t)) t = YearFromTime(t);
|
| - return t;
|
| + CHECK_DATE(this);
|
| + return LOCAL_YEAR(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.11
|
| function DateGetUTCFullYear() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return YearFromTime(t);
|
| + CHECK_DATE(this);
|
| + return UTC_YEAR(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.12
|
| function DateGetMonth() {
|
| - var t = DATE_MONTH(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_MONTH(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_MONTH(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.13
|
| function DateGetUTCMonth() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return MonthFromTime(t);
|
| + CHECK_DATE(this);
|
| + return UTC_MONTH(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.14
|
| function DateGetDate() {
|
| - var t = DATE_DAY(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_DAY(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_DAY(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.15
|
| function DateGetUTCDate() {
|
| - var t = DATE_VALUE(this);
|
| - return NAN_OR_DATE_FROM_TIME(t);
|
| + CHECK_DATE(this);
|
| + return UTC_DAY(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.16
|
| function DateGetDay() {
|
| - var t = DATE_WEEKDAY(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_WEEKDAY(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_WEEKDAY(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.17
|
| function DateGetUTCDay() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return WeekDay(t);
|
| + CHECK_DATE(this);
|
| + return UTC_WEEKDAY(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.18
|
| function DateGetHours() {
|
| - var t = DATE_HOUR(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_HOUR(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_HOUR(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.19
|
| function DateGetUTCHours() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return HOUR_FROM_TIME(t);
|
| + CHECK_DATE(this);
|
| + return UTC_HOUR(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.20
|
| function DateGetMinutes() {
|
| - var t = DATE_MIN(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_MIN(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_MIN(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.21
|
| function DateGetUTCMinutes() {
|
| - var t = DATE_VALUE(this);
|
| - return NAN_OR_MIN_FROM_TIME(t);
|
| + CHECK_DATE(this);
|
| + return UTC_MIN(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.22
|
| function DateGetSeconds() {
|
| - var t = DATE_SEC(this);
|
| - if (!IS_UNDEFINED(t) && DATE_LOCAL(this) === local_time_offset) return t;
|
| - ResetDate(this, DATE_VALUE_UNCHECKED(this));
|
| - return DATE_SEC(this);
|
| + CHECK_DATE(this);
|
| + return LOCAL_SEC(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.23
|
| function DateGetUTCSeconds() {
|
| - var t = DATE_VALUE(this);
|
| - return NAN_OR_SEC_FROM_TIME(t);
|
| + CHECK_DATE(this);
|
| + return UTC_SEC(this)
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.24
|
| function DateGetMilliseconds() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return MS_FROM_TIME(LocalTimeNoCheck(t));
|
| + CHECK_DATE(this);
|
| + return LOCAL_MS(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.25
|
| function DateGetUTCMilliseconds() {
|
| - var t = DATE_VALUE(this);
|
| - return NAN_OR_MS_FROM_TIME(t);
|
| + CHECK_DATE(this);
|
| + return UTC_MS(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.26
|
| function DateGetTimezoneOffset() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return t;
|
| - return (t - LocalTimeNoCheck(t)) / msPerMinute;
|
| + CHECK_DATE(this);
|
| + return TIMEZONE_OFFSET(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.27
|
| function DateSetTime(ms) {
|
| - if (!IS_DATE(this)) ThrowDateTypeError();
|
| - return ResetDate(this, TimeClip(ToNumber(ms)));
|
| + CHECK_DATE(this);
|
| + SET_UTC_DATE_VALUE(this, ToNumber(ms));
|
| + return UTC_DATE_VALUE(this);
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.28
|
| function DateSetMilliseconds(ms) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| ms = ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t),
|
| - MIN_FROM_TIME(t),
|
| - SEC_FROM_TIME(t),
|
| - ms);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(DAY(t), time))));
|
| + var time = MakeTime(LOCAL_HOUR(this), LOCAL_MIN(this), LOCAL_SEC(this), ms);
|
| + SET_LOCAL_DATE_VALUE(this, MakeDate(LOCAL_DAYS(this), time));
|
| + return this;
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.29
|
| function DateSetUTCMilliseconds(ms) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| ms = ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t),
|
| - MIN_FROM_TIME(t),
|
| - SEC_FROM_TIME(t),
|
| + var time = MakeTime(UTC_HOUR(this),
|
| + UTC_MIN(this),
|
| + UTC_SEC(this),
|
| ms);
|
| - return ResetDate(this, TimeClip(MakeDate(DAY(t), time)));
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(UTC_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.30
|
| function DateSetSeconds(sec, ms) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| sec = ToNumber(sec);
|
| - ms = %_ArgumentsLength() < 2 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t), MIN_FROM_TIME(t), sec, ms);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(DAY(t), time))));
|
| + ms = %_ArgumentsLength() < 2 ? LOCAL_MS(this) : ToNumber(ms);
|
| + var time = MakeTime(LOCAL_HOUR(this), LOCAL_MIN(this), sec, ms);
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(LOCAL_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.31
|
| function DateSetUTCSeconds(sec, ms) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| sec = ToNumber(sec);
|
| - ms = %_ArgumentsLength() < 2 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t), MIN_FROM_TIME(t), sec, ms);
|
| - return ResetDate(this, TimeClip(MakeDate(DAY(t), time)));
|
| + ms = %_ArgumentsLength() < 2 ? UTC_MS(this) : ToNumber(ms);
|
| + var time = MakeTime(UTC_HOUR(this), UTC_MIN(this), sec, ms);
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(UTC_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.33
|
| function DateSetMinutes(min, sec, ms) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| min = ToNumber(min);
|
| var argc = %_ArgumentsLength();
|
| - sec = argc < 2 ? NAN_OR_SEC_FROM_TIME(t) : ToNumber(sec);
|
| - ms = argc < 3 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t), min, sec, ms);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(DAY(t), time))));
|
| + sec = argc < 2 ? LOCAL_SEC(this) : ToNumber(sec);
|
| + ms = argc < 3 ? LOCAL_MS(this) : ToNumber(ms);
|
| + var time = MakeTime(LOCAL_HOUR(this), min, sec, ms);
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(LOCAL_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.34
|
| function DateSetUTCMinutes(min, sec, ms) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| min = ToNumber(min);
|
| var argc = %_ArgumentsLength();
|
| - sec = argc < 2 ? NAN_OR_SEC_FROM_TIME(t) : ToNumber(sec);
|
| - ms = argc < 3 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| - var time = MakeTime(HOUR_FROM_TIME(t), min, sec, ms);
|
| - return ResetDate(this, TimeClip(MakeDate(DAY(t), time)));
|
| + sec = argc < 2 ? UTC_SEC(this) : ToNumber(sec);
|
| + ms = argc < 3 ? UTC_MS(this) : ToNumber(ms);
|
| + var time = MakeTime(UTC_HOUR(this), min, sec, ms);
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(UTC_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.35
|
| function DateSetHours(hour, min, sec, ms) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| hour = ToNumber(hour);
|
| var argc = %_ArgumentsLength();
|
| - min = argc < 2 ? NAN_OR_MIN_FROM_TIME(t) : ToNumber(min);
|
| - sec = argc < 3 ? NAN_OR_SEC_FROM_TIME(t) : ToNumber(sec);
|
| - ms = argc < 4 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| + min = argc < 2 ? LOCAL_MIN(this) : ToNumber(min);
|
| + sec = argc < 3 ? LOCAL_SEC(this) : ToNumber(sec);
|
| + ms = argc < 4 ? LOCAL_MS(this) : ToNumber(ms);
|
| var time = MakeTime(hour, min, sec, ms);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(DAY(t), time))));
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(LOCAL_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.34
|
| function DateSetUTCHours(hour, min, sec, ms) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| hour = ToNumber(hour);
|
| var argc = %_ArgumentsLength();
|
| - min = argc < 2 ? NAN_OR_MIN_FROM_TIME(t) : ToNumber(min);
|
| - sec = argc < 3 ? NAN_OR_SEC_FROM_TIME(t) : ToNumber(sec);
|
| - ms = argc < 4 ? NAN_OR_MS_FROM_TIME(t) : ToNumber(ms);
|
| + min = argc < 2 ? UTC_MIN(this) : ToNumber(min);
|
| + sec = argc < 3 ? UTC_SEC(this) : ToNumber(sec);
|
| + ms = argc < 4 ? UTC_MS(this) : ToNumber(ms);
|
| var time = MakeTime(hour, min, sec, ms);
|
| - return ResetDate(this, TimeClip(MakeDate(DAY(t), time)));
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(UTC_DAYS(this), time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.36
|
| function DateSetDate(date) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| date = ToNumber(date);
|
| - var day = MakeDay(YearFromTime(t), MonthFromTime(t), date);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(day, TimeWithinDay(t)))));
|
| + var day = MakeDay(LOCAL_YEAR(this), LOCAL_MONTH(this), date);
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(day, LOCAL_TIME_IN_DAY(this)));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.37
|
| function DateSetUTCDate(date) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| date = ToNumber(date);
|
| - var day = MakeDay(YearFromTime(t), MonthFromTime(t), date);
|
| - return ResetDate(this, TimeClip(MakeDate(day, TimeWithinDay(t))));
|
| + var day = MakeDay(UTC_YEAR(this), UTC_MONTH(this), date);
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(day, UTC_TIME_IN_DAY(this)));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.38
|
| function DateSetMonth(month, date) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| month = ToNumber(month);
|
| - date = %_ArgumentsLength() < 2 ? NAN_OR_DATE_FROM_TIME(t) : ToNumber(date);
|
| - var day = MakeDay(YearFromTime(t), month, date);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(day, TimeWithinDay(t)))));
|
| + date = %_ArgumentsLength() < 2 ? LOCAL_DAY(this) : ToNumber(date);
|
| + var day = MakeDay(LOCAL_YEAR(this), month, date);
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(day, LOCAL_TIME_IN_DAY(this)));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.39
|
| function DateSetUTCMonth(month, date) {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| month = ToNumber(month);
|
| - date = %_ArgumentsLength() < 2 ? NAN_OR_DATE_FROM_TIME(t) : ToNumber(date);
|
| - var day = MakeDay(YearFromTime(t), month, date);
|
| - return ResetDate(this, TimeClip(MakeDate(day, TimeWithinDay(t))));
|
| + date = %_ArgumentsLength() < 2 ? UTC_DAY(this) : ToNumber(date);
|
| + var day = MakeDay(UTC_YEAR(this), month, date);
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(day, UTC_TIME_IN_DAY(this)));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.40
|
| function DateSetFullYear(year, month, date) {
|
| - var t = DATE_VALUE(this);
|
| - t = NUMBER_IS_NAN(t) ? 0 : LocalTimeNoCheck(t);
|
| + CHECK_DATE(this);
|
| + var t = LOCAL_DATE_VALUE(this);
|
| year = ToNumber(year);
|
| var argc = %_ArgumentsLength();
|
| - month = argc < 2 ? MonthFromTime(t) : ToNumber(month);
|
| - date = argc < 3 ? DateFromTime(t) : ToNumber(date);
|
| + var time ;
|
| + if (NUMBER_IS_NAN(t)) {
|
| + month = argc < 2 ? 0 : ToNumber(month);
|
| + date = argc < 3 ? 1 : ToNumber(date);
|
| + time = 0;
|
| + } else {
|
| + month = argc < 2 ? LOCAL_MONTH(this) : ToNumber(month);
|
| + date = argc < 3 ? LOCAL_DAY(this) : ToNumber(date);
|
| + time = LOCAL_TIME_IN_DAY(this);
|
| + }
|
| var day = MakeDay(year, month, date);
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(day, TimeWithinDay(t)))));
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(day, time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.41
|
| function DateSetUTCFullYear(year, month, date) {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) t = 0;
|
| - var argc = %_ArgumentsLength();
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| year = ToNumber(year);
|
| - month = argc < 2 ? MonthFromTime(t) : ToNumber(month);
|
| - date = argc < 3 ? DateFromTime(t) : ToNumber(date);
|
| + var argc = %_ArgumentsLength();
|
| + var time ;
|
| + if (NUMBER_IS_NAN(t)) {
|
| + month = argc < 2 ? 0 : ToNumber(month);
|
| + date = argc < 3 ? 1 : ToNumber(date);
|
| + time = 0;
|
| + } else {
|
| + month = argc < 2 ? UTC_MONTH(this) : ToNumber(month);
|
| + date = argc < 3 ? UTC_DAY(this) : ToNumber(date);
|
| + time = UTC_TIME_IN_DAY(this);
|
| + }
|
| var day = MakeDay(year, month, date);
|
| - return ResetDate(this, TimeClip(MakeDate(day, TimeWithinDay(t))));
|
| + return SET_UTC_DATE_VALUE(this, MakeDate(day, time));
|
| }
|
|
|
|
|
| // ECMA 262 - 15.9.5.42
|
| function DateToUTCString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) return kInvalidDate;
|
| // Return UTC string of the form: Sat, 31 Jan 1970 23:00:00 GMT
|
| - return WeekDays[WeekDay(t)] + ', '
|
| - + TwoDigitString(DateFromTime(t)) + ' '
|
| - + Months[MonthFromTime(t)] + ' '
|
| - + YearFromTime(t) + ' '
|
| - + TimeString(t) + ' GMT';
|
| + return WeekDays[UTC_WEEKDAY(this)] + ', '
|
| + + TwoDigitString(UTC_DAY(this)) + ' '
|
| + + Months[UTC_MONTH(this)] + ' '
|
| + + UTC_YEAR(this) + ' '
|
| + + TimeStringUTC(this) + ' GMT';
|
| }
|
|
|
|
|
| // ECMA 262 - B.2.4
|
| function DateGetYear() {
|
| - var t = DATE_VALUE(this);
|
| - if (NUMBER_IS_NAN(t)) return $NaN;
|
| - return YearFromTime(LocalTimeNoCheck(t)) - 1900;
|
| + CHECK_DATE(this);
|
| + return LOCAL_YEAR(this) - 1900;
|
| }
|
|
|
|
|
| // ECMA 262 - B.2.5
|
| function DateSetYear(year) {
|
| - var t = LocalTime(DATE_VALUE(this));
|
| - if (NUMBER_IS_NAN(t)) t = 0;
|
| + CHECK_DATE(this);
|
| year = ToNumber(year);
|
| - if (NUMBER_IS_NAN(year)) return ResetDate(this, $NaN);
|
| + if (NUMBER_IS_NAN(year)) return SET_UTC_DATE_VALUE(this, $NaN);
|
| year = (0 <= TO_INTEGER(year) && TO_INTEGER(year) <= 99)
|
| ? 1900 + TO_INTEGER(year) : year;
|
| - var day = MakeDay(year, MonthFromTime(t), DateFromTime(t));
|
| - return ResetDate(this, TimeClip(UTC(MakeDate(day, TimeWithinDay(t)))));
|
| + var t = LOCAL_DATE_VALUE(this);
|
| + var month, date, time;
|
| + if (NUMBER_IS_NAN(t)) {
|
| + month = 0;
|
| + date = 1;
|
| + time = 0;
|
| + } else {
|
| + month = LOCAL_MONTH(this);
|
| + date = LOCAL_DAY(this);
|
| + time = LOCAL_TIME_IN_DAY(this);
|
| + }
|
| + var day = MakeDay(year, month, date);
|
| + return SET_LOCAL_DATE_VALUE(this, MakeDate(day, time));
|
| }
|
|
|
|
|
| @@ -1025,7 +758,8 @@ function PadInt(n, digits) {
|
|
|
| // ECMA 262 - 15.9.5.43
|
| function DateToISOString() {
|
| - var t = DATE_VALUE(this);
|
| + CHECK_DATE(this);
|
| + var t = UTC_DATE_VALUE(this);
|
| if (NUMBER_IS_NAN(t)) throw MakeRangeError("invalid_time_value", []);
|
| var year = this.getUTCFullYear();
|
| var year_string;
|
| @@ -1060,34 +794,13 @@ function DateToJSON(key) {
|
|
|
|
|
| function ResetDateCache() {
|
| -
|
| - // Reset the local_time_offset:
|
| - local_time_offset = %DateLocalTimeOffset();
|
| -
|
| - // Reset the DST offset cache:
|
| - var cache = DST_offset_cache;
|
| - cache.offset = 0;
|
| - cache.start = 0;
|
| - cache.end = -1;
|
| - cache.increment = 0;
|
| - cache.initial_increment = 19 * msPerDay;
|
| -
|
| // Reset the timezone cache:
|
| timezone_cache_time = $NaN;
|
| timezone_cache_timezone = undefined;
|
|
|
| - // Reset the ltcache:
|
| - ltcache.key = null;
|
| - ltcache.val = null;
|
| -
|
| - // Reset the ymd_from_time_cache:
|
| - ymd_from_time_cache = [$NaN, $NaN, $NaN];
|
| - ymd_from_time_cached_time = $NaN;
|
| -
|
| // Reset the date cache:
|
| cache = Date_cache;
|
| cache.time = $NaN;
|
| - cache.year = $NaN;
|
| cache.string = null;
|
| }
|
|
|
|
|