| Index: src/date.h | 
| diff --git a/src/date.h b/src/date.h | 
| index 2e5ce39a042c791de83a4446c960225a4784cf61..813d3126ede972f628e325336e94e9e5661f132c 100644 | 
| --- a/src/date.h | 
| +++ b/src/date.h | 
| @@ -104,21 +104,51 @@ class DateCache { | 
|  | 
| // ECMA 262 - 15.9.1.9 | 
| // LocalTime(t) = t + LocalTZA + DaylightSavingTA(t) | 
| -  // ECMA 262 assumes that DaylightSavingTA is computed using UTC time, | 
| -  // but we fetch DST from OS using local time, therefore we need: | 
| -  // LocalTime(t) = t + LocalTZA + DaylightSavingTA(t + LocalTZA). | 
| int64_t ToLocal(int64_t time_ms) { | 
| -    time_ms += LocalOffsetInMs(); | 
| -    return time_ms + DaylightSavingsOffsetInMs(time_ms); | 
| +    return time_ms + LocalOffsetInMs() + DaylightSavingsOffsetInMs(time_ms); | 
| } | 
|  | 
| // ECMA 262 - 15.9.1.9 | 
| // UTC(t) = t - LocalTZA - DaylightSavingTA(t - LocalTZA) | 
| -  // ECMA 262 assumes that DaylightSavingTA is computed using UTC time, | 
| -  // but we fetch DST from OS using local time, therefore we need: | 
| -  // UTC(t) = t - LocalTZA - DaylightSavingTA(t). | 
| int64_t ToUTC(int64_t time_ms) { | 
| -    return time_ms - LocalOffsetInMs() - DaylightSavingsOffsetInMs(time_ms); | 
| +    // We need to compute UTC time that corresponds to the given local time. | 
| +    // Literally following spec here leads to incorrect time computation at | 
| +    // the points were we transition to and from DST. | 
| +    // | 
| +    // The following shows that using DST for (t - LocalTZA - hour) produces | 
| +    // correct conversion. | 
| +    // | 
| +    // Consider transition to DST at local time L1. | 
| +    // Let L0 = L1 - hour, L2 = L1 + hour, | 
| +    //     U1 = UTC time that corresponds to L1, | 
| +    //     U0 = U1 - hour. | 
| +    // Transitioning to DST moves local clock one hour forward L1 => L2, so | 
| +    // U0 = UTC time that corresponds to L0 = L0 - LocalTZA, | 
| +    // U1 = UTC time that corresponds to L1 = L1 - LocalTZA, | 
| +    // U1 = UTC time that corresponds to L2 = L2 - LocalTZA - hour. | 
| +    // Note that DST(U0 - hour) = 0, DST(U0) = 0, DST(U1) = 1. | 
| +    // U0 = L0 - LocalTZA - DST(L0 - LocalTZA - hour), | 
| +    // U1 = L1 - LocalTZA - DST(L1 - LocalTZA - hour), | 
| +    // U1 = L2 - LocalTZA - DST(L2 - LocalTZA - hour). | 
| +    // | 
| +    // Consider transition from DST at local time L1. | 
| +    // Let L0 = L1 - hour, | 
| +    //     U1 = UTC time that corresponds to L1, | 
| +    //     U0 = U1 - hour, U2 = U1 + hour. | 
| +    // Transitioning from DST moves local clock one hour back L1 => L0, so | 
| +    // U0 = UTC time that corresponds to L0 (before transition) | 
| +    //    = L0 - LocalTZA - hour. | 
| +    // U1 = UTC time that corresponds to L0 (after transition) | 
| +    //    = L0 - LocalTZA = L1 - LocalTZA - hour | 
| +    // U2 = UTC time that corresponds to L1 = L1 - LocalTZA. | 
| +    // Note that DST(U0) = 1, DST(U1) = 0, DST(U2) = 0. | 
| +    // U0 = L0 - LocalTZA - DST(L0 - LocalTZA - hour) = L0 - LocalTZA - DST(U0). | 
| +    // U2 = L1 - LocalTZA - DST(L1 - LocalTZA - hour) = L1 - LocalTZA - DST(U1). | 
| +    // It is impossible to get U1 from local time. | 
| + | 
| +    const int kMsPerHour = 3600 * 1000; | 
| +    time_ms -= LocalOffsetInMs(); | 
| +    return time_ms - DaylightSavingsOffsetInMs(time_ms - kMsPerHour); | 
| } | 
|  | 
|  | 
|  |