Update contrib.
2 * © Portions copyright (c) 2006-2007 Nokia Corporation. All rights reserved.
5 * Silicon Graphics Computer Systems, Inc.
10 * This material is provided "as is", with absolutely no warranty expressed
11 * or implied. Any use is at your own risk.
13 * Permission to use or copy this software for any purpose is hereby granted
14 * without fee, provided the above notices are retained on all copies.
15 * Permission to modify the code and to distribute modified code is granted,
16 * provided the above notices are retained, and a notice that the code was
17 * modified is included with the above copyright notice.
22 #include "stlport_prefix.h"
23 #include <stl/_time_facets.h>
24 #include <stl/_istream.h>
27 #if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
28 #include "libstdcppwsd.h"
34 __write_integer(char* buf, ios_base::fmtflags flags, long x);*/
36 // The function copy_cstring is used to initialize a string
37 // with a C-style string. Used to initialize the month and weekday
38 // tables in time_get and time_put. Called only by _Init_timeinfo
39 // so its name does not require leading underscores.
41 static inline void copy_cstring(const char * s, string& v) {
42 copy(s, s + strlen(s), back_insert_iterator<string >(v));
45 // default "C" values for month and day names
47 const char default_dayname[][14] = {
48 "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat",
49 "Sunday", "Monday", "Tuesday", "Wednesday", "Thursday",
50 "Friday", "Saturday"};
52 const char default_monthname[][24] = {
53 "Jan", "Feb", "Mar", "Apr", "May", "Jun",
54 "Jul", "Aug", "Sep", "Oct", "Nov", "Dec",
55 "January", "February", "March", "April", "May", "June",
56 "July", "August", "September", "October", "November", "December"};
58 // _Init_time_info: initialize table with
59 // "C" values (note these are not defined in the C standard, so this
60 // is somewhat arbitrary).
62 _STLP_EXP_DECLSPEC void _STLP_CALL _Init_timeinfo(_Time_Info& table) {
64 for (i = 0; i < 14; ++i)
65 copy_cstring(default_dayname[i], table._M_dayname[i]);
66 for (i = 0; i < 24; ++i)
67 copy_cstring(default_monthname[i], table._M_monthname[i]);
68 copy_cstring("AM", table._M_am_pm[0]);
69 copy_cstring("PM", table._M_am_pm[1]);
70 copy_cstring("%H:%M:%S", table._M_time_format);
72 copy_cstring("%m/%d/%Y", table._M_date_format); //linux follows the same default. Y = year in 4 digit
74 copy_cstring("%m/%d/%y", table._M_date_format);
76 copy_cstring("%a %b %e %H:%M:%S %Y", table._M_date_time_format);
79 _STLP_EXP_DECLSPEC void _STLP_CALL _Init_timeinfo(_Time_Info& table, _Locale_time * time) {
81 for (i = 0; i < 7; ++i)
82 copy_cstring(_Locale_abbrev_dayofweek(time, i),
84 for (i = 0; i < 7; ++i)
85 copy_cstring(_Locale_full_dayofweek(time, i),
86 table._M_dayname[i+7]);
87 for (i = 0; i < 12; ++i)
88 copy_cstring(_Locale_abbrev_monthname(time, i),
89 table._M_monthname[i]);
90 for (i = 0; i < 12; ++i)
91 copy_cstring(_Locale_full_monthname(time, i),
92 table._M_monthname[i+12]);
93 copy_cstring(_Locale_am_str(time),
95 copy_cstring(_Locale_pm_str(time),
97 copy_cstring(_Locale_t_fmt(time), table._M_time_format);
98 copy_cstring(_Locale_d_fmt(time), table._M_date_format);
99 copy_cstring(_Locale_d_t_fmt(time), table._M_date_time_format);
100 copy_cstring(_Locale_long_d_fmt(time), table._M_long_date_format);
101 copy_cstring(_Locale_long_d_t_fmt(time), table._M_long_date_time_format);
104 inline char* __subformat(string format, char*& buf,
105 const _Time_Info& table, const tm* t) {
106 const char * cp = format.data();
107 const char * cp_end = cp + format.size();
108 while (cp != cp_end) {
115 buf = __write_formatted_time(buf, *cp++, mod, table, t);
122 /* The number of days from the first day of the first ISO week of this
123 year to the year day YDAY with week day WDAY. ISO weeks start on
124 Monday; the first ISO week has the year's first Thursday. YDAY may
125 be as small as YDAY_MINIMUM. */
126 #define __ISO_WEEK_START_WDAY 1 /* Monday */
127 #define __ISO_WEEK1_WDAY 4 /* Thursday */
128 #define __YDAY_MINIMUM (-366)
129 #define __TM_YEAR_BASE 1900
130 #if defined(__GNUC__)
132 __iso_week_days (int yday, int wday)
134 /* Add enough to the first operand of % to make it nonnegative. */
135 int big_enough_multiple_of_7 = (-__YDAY_MINIMUM / 7 + 2) * 7;
137 - (yday - wday + __ISO_WEEK1_WDAY + big_enough_multiple_of_7) % 7
138 + __ISO_WEEK1_WDAY - __ISO_WEEK_START_WDAY);
142 #define __is_leap(year) \
143 ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
145 #define __hour12(hour) \
146 (((hour) % 12 == 0) ? (12) : (hour) % 12)
148 _STLP_EXP_DECLSPEC char * __write_formatted_time(char* buf, char format, char modifier,
149 const _Time_Info& table, const tm* t) {
152 return copy(table._M_dayname[t->tm_wday].begin(),
153 table._M_dayname[t->tm_wday].end(),
157 return copy(table._M_dayname[t->tm_wday+7].begin(),
158 table._M_dayname[t->tm_wday+7].end(),
162 return copy(table._M_monthname[t->tm_mon].begin(),
163 table._M_monthname[t->tm_mon].end(),
167 return copy(table._M_monthname[t->tm_mon+12].begin(),
168 table._M_monthname[t->tm_mon+12].end(),
172 const char * cp = (modifier != '#') ? table._M_date_time_format.data():\
173 table._M_long_date_time_format.data();
174 const char* cp_end = (modifier != '#') ? cp + table._M_date_time_format.size():\
175 cp + table._M_long_date_time_format.size();
177 while (cp != cp_end) {
179 ++cp; if(*cp == '#') mod = *cp++; else mod = 0;
180 buf = __write_formatted_time(buf, *cp++, mod, table, t);
189 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)t->tm_mday);
190 return ((long)t->tm_mday < 10L && modifier == '#')?buf+1:buf + 2;
193 sprintf(buf, "%2ld", (long)t->tm_mday);
197 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)t->tm_hour);
198 return ((long)t->tm_hour < 10L && modifier == '#')?buf+1:buf + 2;
201 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)__hour12(t->tm_hour));
202 return ((long)__hour12(t->tm_hour) < 10L && modifier == '#')?buf+1:buf + 2;
205 return __write_integer(buf, 0, (long)((long)t->tm_yday + 1));
208 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)t->tm_mon + 1);
209 return ((long)(t->tm_mon + 1) < 10L && modifier == '#')?buf+1:buf + 2;
212 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)t->tm_min);
213 return ((long)t->tm_min < 10L && modifier == '#')?buf+1:buf + 2;
216 return copy(table._M_am_pm[t->tm_hour/12].begin(),
217 table._M_am_pm[t->tm_hour/12].end(),
220 case 'S': // pad with zeros
221 sprintf(buf, (modifier != '#')?"%.2ld":"%ld", (long)t->tm_sec);
222 return ((long)t->tm_sec < 10L && modifier == '#')?buf+1:buf + 2;
225 return __write_integer(buf, 0,
226 long((t->tm_yday - t->tm_wday + 7) / 7));
230 return __write_integer(buf, 0, (long)t->tm_wday);
234 return __write_integer(buf, 0,
235 (long)(t->tm_wday == 0 ?
236 (t->tm_yday + 1) / 7 :
237 (t->tm_yday + 8 - t->tm_wday) / 7));
240 const char * cp = (modifier != '#') ? table._M_date_format.data():\
241 table._M_long_date_format.data();
242 const char* cp_end = (modifier != '#') ? cp + table._M_date_format.size():\
243 cp + table._M_long_date_format.size();
245 while (cp != cp_end) {
247 ++cp; if(*cp == '#') mod = *cp++; else mod = 0;
248 buf = __write_formatted_time(buf, *cp++, mod, table, t);
257 const char * cp = table._M_time_format.data();
258 const char* cp_end = cp + table._M_time_format.size();
260 while (cp != cp_end) {
262 ++cp; if(*cp == '#') mod = *cp++; else mod = 0;
263 buf = __write_formatted_time(buf, *cp++, mod, table, t);
271 return __write_integer(buf, 0, (long)((long)(t->tm_year + 1900) % 100));
274 return __write_integer(buf, 0, (long)((long)t->tm_year + 1900));
282 // fbp : at least on SUN
283 # if defined ( _STLP_UNIX ) && ! defined (__linux__)
287 /*********************************************
288 * JGS, handle various extensions *
289 *********************************************/
291 case 'h': /* POSIX.2 extension */
292 // same as 'b', abbrev month name
293 return copy(table._M_monthname[t->tm_mon].begin(),
294 table._M_monthname[t->tm_mon].end(),
297 case 'C': /* POSIX.2 extension */
298 // same as 'd', the day
299 sprintf(buf, "%2ld", (long)t->tm_mday);
302 case 'D': /* POSIX.2 extension */
304 return __subformat(table._M_date_format, buf, table, t);
306 case 'k': /* GNU extension */
307 sprintf(buf, "%2ld", (long)t->tm_hour);
310 case 'l': /* GNU extension */
311 sprintf(buf, "%2ld", (long)t->tm_hour % 12);
314 case 'n': /* POSIX.2 extension */
318 case 'R': /* GNU extension */
319 return __subformat("%H:%M", buf, table, t);
321 case 'r': /* POSIX.2 extension */
322 return __subformat("%I:%M:%S %p", buf, table, t);
324 case 'T': /* POSIX.2 extension. */
325 return __subformat("%H:%M:%S", buf, table, t);
327 case 't': /* POSIX.2 extension. */
331 case 'u': /* POSIX.2 extension. */
332 return __write_integer(buf, 0, long((t->tm_wday - 1 + 7)) % 7 + 1);
336 __t = mktime ((tm*)t);
337 return __write_integer(buf, 0, (long)__t );
339 case 'g': /* GNU extension */
341 int year = t->tm_year + __TM_YEAR_BASE;
342 int days = __iso_week_days (t->tm_yday, t->tm_wday);
344 /* This ISO week belongs to the previous year. */
346 days = __iso_week_days (t->tm_yday + (365 + __is_leap (year)),
349 int d = __iso_week_days (t->tm_yday - (365 + __is_leap (year)),
352 /* This ISO week belongs to the next year. */
359 return __write_integer(buf, 0, (long)(year % 100 + 100) % 100);
361 return __write_integer(buf, 0, (long)year);
363 return __write_integer(buf, 0, (long)days / 7 + 1);
367 # if defined ( _STLP_USE_GLIBC ) && ! defined (__CYGWIN__)
368 case 'z': /* GNU extension. */
373 #if defined(__USE_BSD) || defined(__BEOS__)
376 diff = t->__tm_gmtoff;
385 sprintf(buf, "%.4d", (diff / 60) * 100 + diff % 60);
388 # endif /* __GLIBC__ */
389 #endif /* __GNUC__ */
398 _STLP_EXP_DECLSPEC time_base::dateorder _STLP_CALL
399 __get_date_order(_Locale_time* time)
401 const char * fmt = _Locale_d_fmt(time);
402 char first, second, third;
404 while (*fmt != 0 && *fmt != '%') ++fmt;
406 return time_base::no_order;
408 while (*fmt != 0 && *fmt != '%') ++fmt;
410 return time_base::no_order;
412 while (*fmt != 0 && *fmt != '%') ++fmt;
414 return time_base::no_order;
419 return (second == 'm' && third == 'y') ? time_base::dmy
420 : time_base::no_order;
422 return (second == 'd' && third == 'y') ? time_base::mdy
423 : time_base::no_order;
427 return third == 'm' ? time_base::ydm : time_base::no_order;
429 return third == 'd' ? time_base::ymd : time_base::no_order;
431 return time_base::no_order;
434 return time_base::no_order;
438 #if !defined(_STLP_NO_FORCE_INSTANTIATE)
439 template class time_get<char, istreambuf_iterator<char, char_traits<char> > >;
440 // template class time_get<char, const char*>;
441 template class time_put<char, ostreambuf_iterator<char, char_traits<char> > >;
442 // template class time_put<char, char*>;
444 #ifndef _STLP_NO_WCHAR_T
445 template class time_get<wchar_t, istreambuf_iterator<wchar_t, char_traits<wchar_t> > >;
446 // template class time_get<wchar_t, const wchar_t*>;
447 template class time_put<wchar_t, ostreambuf_iterator<wchar_t, char_traits<wchar_t> > >;
448 // template class time_put<wchar_t, wchar_t*>;
449 #endif /* INSTANTIATE_WIDE_STREAMS */