DateConversion::formatDateTime incorrectly formats negative years
[WebKit-https.git] / Source / JavaScriptCore / runtime / DateConversion.cpp
1 /*
2  * Copyright (C) 2012 Patrick Gansterer <paroga@paroga.com>
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
15  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
16  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
17  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
18  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
19  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
20  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
22  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23  */
24
25 #include "config.h"
26 #include "DateConversion.h"
27
28 #include <wtf/Assertions.h>
29 #include <wtf/DateMath.h>
30 #include <wtf/text/StringBuilder.h>
31 #include <wtf/text/WTFString.h>
32
33 #if OS(WINDOWS)
34 #include <windows.h>
35 #endif
36
37 namespace JSC {
38
39 using namespace WTF;
40
41 template<int width>
42 static inline void appendNumber(StringBuilder& builder, int value)
43 {
44     if (value < 0) {
45         builder.append('-');
46         value = -value;
47     }
48     String valueString = String::number(value);
49     int fillingZerosCount = width - valueString.length();
50     for (int i = 0; i < fillingZerosCount; ++i)
51         builder.append('0');
52     builder.append(valueString);
53 }
54
55 template<>
56 void appendNumber<2>(StringBuilder& builder, int value)
57 {
58     ASSERT(0 <= value && value <= 99);
59     builder.append(static_cast<char>('0' + value / 10));
60     builder.append(static_cast<char>('0' + value % 10));
61 }
62
63 String formatDateTime(const GregorianDateTime& t, DateTimeFormat format, bool asUTCVariant)
64 {
65     bool appendDate = format & DateTimeFormatDate;
66     bool appendTime = format & DateTimeFormatTime;
67
68     StringBuilder builder;
69
70     if (appendDate) {
71         builder.append(weekdayName[(t.weekDay() + 6) % 7]);
72
73         if (asUTCVariant) {
74             builder.appendLiteral(", ");
75             appendNumber<2>(builder, t.monthDay());
76             builder.append(' ');
77             builder.append(monthName[t.month()]);
78         } else {
79             builder.append(' ');
80             builder.append(monthName[t.month()]);
81             builder.append(' ');
82             appendNumber<2>(builder, t.monthDay());
83         }
84         builder.append(' ');
85         appendNumber<4>(builder, t.year());
86     }
87
88     if (appendDate && appendTime)
89         builder.append(' ');
90
91     if (appendTime) {
92         appendNumber<2>(builder, t.hour());
93         builder.append(':');
94         appendNumber<2>(builder, t.minute());
95         builder.append(':');
96         appendNumber<2>(builder, t.second());
97         builder.appendLiteral(" GMT");
98
99         if (!asUTCVariant) {
100             int offset = abs(t.utcOffset()) / 60;
101             builder.append(t.utcOffset() < 0 ? '-' : '+');
102             appendNumber<2>(builder, offset / 60);
103             appendNumber<2>(builder, offset % 60);
104
105 #if OS(WINDOWS)
106             TIME_ZONE_INFORMATION timeZoneInformation;
107             GetTimeZoneInformation(&timeZoneInformation);
108             const WCHAR* winTimeZoneName = t.isDST() ? timeZoneInformation.DaylightName : timeZoneInformation.StandardName;
109             String timeZoneName(winTimeZoneName);
110 #else
111             struct tm gtm = t;
112             char timeZoneName[70];
113             strftime(timeZoneName, sizeof(timeZoneName), "%Z", &gtm);
114 #endif
115             if (timeZoneName[0]) {
116                 builder.appendLiteral(" (");
117                 builder.append(timeZoneName);
118                 builder.append(')');
119             }
120         }
121     }
122
123     return builder.toString().impl();
124 }
125
126 } // namespace JSC