| Index: src/date.js | 
| diff --git a/src/date.js b/src/date.js | 
| index 8c51a931e4bfda2140fb281d5cb15937adbf6482..75edf6d32d17c364023c39cb319c937c1421286d 100644 | 
| --- a/src/date.js | 
| +++ b/src/date.js | 
| @@ -44,173 +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; | 
| -  // Guard against returning -0. | 
| -  if (mod == 0) return 0; | 
| -  return mod >= 0 ? mod : mod + remainder; | 
| -} | 
| - | 
| - | 
| -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; | 
| @@ -220,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); | 
| } | 
|  | 
|  | 
| @@ -293,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: | 
| @@ -385,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 | 
| }; | 
| @@ -404,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. | 
| @@ -419,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; | 
| } | 
| } | 
| @@ -433,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); | 
| @@ -449,9 +196,9 @@ 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); | 
| } | 
| -  %_SetValueOf(this, value); | 
| }); | 
|  | 
|  | 
| @@ -468,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); | 
| } | 
|  | 
|  | 
| @@ -482,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)); | 
| +} | 
| + | 
| + | 
| +function LocalTimezoneString(date) { | 
| +  var timezone = LocalTimezone(UTC_DATE_VALUE(date)); | 
|  | 
| -  var timezoneOffset = | 
| -      (DaylightSavingsOffset(time) + local_time_offset) / msPerMinute; | 
| +  var timezoneOffset = -TIMEZONE_OFFSET(date); | 
| var sign = (timezoneOffset >= 0) ? 1 : -1; | 
| var hours = FLOOR((sign * timezoneOffset)/60); | 
| var min   = FLOOR((sign * timezoneOffset)%60); | 
| @@ -523,8 +264,8 @@ function LocalTimezoneString(time) { | 
| } | 
|  | 
|  | 
| -function DatePrintString(time) { | 
| -  return DateString(time) + ' ' + TimeString(time); | 
| +function DatePrintString(date) { | 
| +  return DateString(date) + ' ' + TimeString(date); | 
| } | 
|  | 
| // ------------------------------------------------------------------- | 
| @@ -564,7 +305,7 @@ function DateUTC(year, month, date, hours, minutes, seconds, ms) { | 
| TO_INTEGER(year) <= 99) ? 1900 + TO_INTEGER(year) : year; | 
| var day = MakeDay(year, month, date); | 
| var time = MakeTime(hours, minutes, seconds, ms); | 
| -  return %_SetValueOf(this, TimeClip(MakeDate(day, time))); | 
| +  return TimeClip(MakeDate(day, time)); | 
| } | 
|  | 
|  | 
| @@ -577,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; | 
| } | 
|  | 
|  | 
| @@ -609,363 +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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  var cache = Date_cache; | 
| -  if (cache.time === t) return cache.year; | 
| -  return YearFromTime(LocalTimeNoCheck(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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return MonthFromTime(LocalTimeNoCheck(t)); | 
| +  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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return DateFromTime(LocalTimeNoCheck(t)); | 
| +  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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return WeekDay(LocalTimeNoCheck(t)); | 
| +  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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return HOUR_FROM_TIME(LocalTimeNoCheck(t)); | 
| +  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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return MIN_FROM_TIME(LocalTimeNoCheck(t)); | 
| +  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_VALUE(this); | 
| -  if (NUMBER_IS_NAN(t)) return t; | 
| -  return SEC_FROM_TIME(LocalTimeNoCheck(t)); | 
| +  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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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 %_SetValueOf(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)); | 
| } | 
|  | 
|  | 
| @@ -989,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; | 
| @@ -1024,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; | 
| } | 
|  | 
|  |