1 /* Copyright (C) 2002-2013 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <http://www.gnu.org/licenses/>. */
23 # define USE_IN_EXTENDED_LOCALE_MODEL 1
24 # define HAVE_LIMITS_H 1
26 # define HAVE_MBRLEN 1
27 # define HAVE_STRUCT_ERA_ENTRY 1
28 # define HAVE_TM_GMTOFF 1
29 # define HAVE_TM_ZONE 1
30 # define HAVE_TZNAME 1
32 # define MULTIBYTE_IS_FORMAT_SAFE 1
33 # define STDC_HEADERS 1
34 # include "../locale/localeinfo.h"
37 #if defined emacs && !defined HAVE_BCOPY
38 # define HAVE_MEMCPY 1
42 #include <sys/types.h> /* Some systems define `time_t' here. */
44 #ifdef TIME_WITH_SYS_TIME
45 # include <sys/time.h>
48 # ifdef HAVE_SYS_TIME_H
49 # include <sys/time.h>
55 extern char *tzname
[];
58 /* Do multibyte processing if multibytes are supported, unless
59 multibyte sequences are safe in formats. Multibyte sequences are
60 safe if they cannot contain byte sequences that look like format
61 conversion specifications. The GNU C Library uses UTF8 multibyte
62 encoding, which is safe for formats, but strftime.c can be used
63 with other C libraries that use unsafe encodings. */
64 #define DO_MULTIBYTE (HAVE_MBLEN && ! MULTIBYTE_IS_FORMAT_SAFE)
70 /* Simulate mbrlen with mblen as best we can. */
71 # define mbstate_t int
72 # define mbrlen(s, n, ps) mblen (s, n)
73 # define mbsinit(ps) (*(ps) == 0)
75 static const mbstate_t mbstate_zero
;
89 # define memcpy(d, s, n) bcopy ((s), (d), (n))
95 # define CHAR_T wchar_t
96 # define UCHAR_T unsigned int
97 # define L_(Str) L##Str
98 # define NLW(Sym) _NL_W##Sym
100 # define MEMCPY(d, s, n) __wmemcpy (d, s, n)
101 # define STRLEN(s) __wcslen (s)
105 # define UCHAR_T unsigned char
107 # define NLW(Sym) Sym
109 # if !defined STDC_HEADERS && !defined HAVE_MEMCPY
110 # define MEMCPY(d, s, n) bcopy ((s), (d), (n))
112 # define MEMCPY(d, s, n) memcpy ((d), (s), (n))
114 # define STRLEN(s) strlen (s)
117 # define MEMPCPY(d, s, n) __mempcpy (d, s, n)
119 # ifndef HAVE_MEMPCPY
120 # define MEMPCPY(d, s, n) ((void *) ((char *) memcpy (d, s, n) + (n)))
137 #define TYPE_SIGNED(t) ((t) -1 < 0)
139 /* Bound on length of the string representing an integer value of type t.
140 Subtract one for the sign bit if t is signed;
141 302 / 1000 is log10 (2) rounded up;
142 add one for integer division truncation;
143 add one more for a minus sign if t is signed. */
144 #define INT_STRLEN_BOUND(t) \
145 ((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 + 1 + TYPE_SIGNED (t))
147 #define TM_YEAR_BASE 1900
150 /* Nonzero if YEAR is a leap year (every 4 years,
151 except every 100th isn't, and every 400th is). */
152 # define __isleap(year) \
153 ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
158 # define tzname __tzname
159 # define tzset __tzset
163 /* Portable standalone applications should supply a "time_r.h" that
164 declares a POSIX-compliant localtime_r, for the benefit of older
165 implementations that lack localtime_r or have a nonstandard one.
166 Similarly for gmtime_r. See the gnulib time_r module for one way
167 to implement this. */
170 # undef __localtime_r
171 # define __gmtime_r gmtime_r
172 # define __localtime_r localtime_r
176 #if !defined memset && !defined HAVE_MEMSET && !defined _LIBC
177 /* Some systems lack the `memset' function and we don't want to
178 introduce additional dependencies. */
179 /* The SGI compiler reportedly barfs on the trailing null
180 if we use a string constant as the initializer. 28 June 1997, rms. */
181 static const CHAR_T spaces
[16] = /* " " */
183 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),
184 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' ')
186 static const CHAR_T zeroes
[16] = /* "0000000000000000" */
188 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),
189 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0')
192 # define memset_space(P, Len) \
198 int _this = _len > 16 ? 16 : _len; \
199 (P) = MEMPCPY ((P), spaces, _this * sizeof (CHAR_T)); \
205 # define memset_zero(P, Len) \
211 int _this = _len > 16 ? 16 : _len; \
212 (P) = MEMPCPY ((P), zeroes, _this * sizeof (CHAR_T)); \
219 # define memset_space(P, Len) (wmemset ((P), L' ', (Len)), (P) += (Len))
220 # define memset_zero(P, Len) (wmemset ((P), L'0', (Len)), (P) += (Len))
222 # define memset_space(P, Len) (memset ((P), ' ', (Len)), (P) += (Len))
223 # define memset_zero(P, Len) (memset ((P), '0', (Len)), (P) += (Len))
231 int _delta = width - _n; \
232 int _incr = _n + (_delta > 0 ? _delta : 0); \
233 if ((size_t) _incr >= maxsize - i) \
239 if (pad == L_('0')) \
240 memset_zero (p, _delta); \
242 memset_space (p, _delta); \
253 memcpy_lowcase (p, (s), _n LOCALE_ARG); \
254 else if (to_uppcase) \
255 memcpy_uppcase (p, (s), _n LOCALE_ARG); \
257 MEMCPY ((PTR) p, (const PTR) (s), _n))
260 # ifndef USE_IN_EXTENDED_LOCALE_MODEL
261 # undef __mbsrtowcs_l
262 # define __mbsrtowcs_l(d, s, l, st, loc) __mbsrtowcs (d, s, l, st)
264 # define widen(os, ws, l) \
267 const char *__s = os; \
268 memset (&__st, '\0', sizeof (__st)); \
269 l = __mbsrtowcs_l (NULL, &__s, 0, &__st, loc); \
270 ws = alloca ((l + 1) * sizeof (wchar_t)); \
271 (void) __mbsrtowcs_l (ws, &__s, l, &__st, loc); \
276 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
277 /* We use this code also for the extended locale handling where the
278 function gets as an additional argument the locale which has to be
279 used. To access the values we have to redefine the _NL_CURRENT
281 # define strftime __strftime_l
282 # define wcsftime __wcsftime_l
284 # define _NL_CURRENT(category, item) \
285 (current->values[_NL_ITEM_INDEX (item)].string)
286 # define LOCALE_PARAM , loc
287 # define LOCALE_ARG , loc
288 # define LOCALE_PARAM_DECL __locale_t loc;
289 # define LOCALE_PARAM_PROTO , __locale_t loc
290 # define HELPER_LOCALE_ARG , current
292 # define LOCALE_PARAM
293 # define LOCALE_PARAM_PROTO
295 # define LOCALE_PARAM_DECL
297 # define HELPER_LOCALE_ARG , _NL_CURRENT_DATA (LC_TIME)
299 # define HELPER_LOCALE_ARG
304 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
305 # define TOUPPER(Ch, L) __towupper_l (Ch, L)
306 # define TOLOWER(Ch, L) __towlower_l (Ch, L)
308 # define TOUPPER(Ch, L) towupper (Ch)
309 # define TOLOWER(Ch, L) towlower (Ch)
313 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
314 # define TOUPPER(Ch, L) __toupper_l (Ch, L)
315 # define TOLOWER(Ch, L) __tolower_l (Ch, L)
317 # define TOUPPER(Ch, L) toupper (Ch)
318 # define TOLOWER(Ch, L) tolower (Ch)
321 # define TOUPPER(Ch, L) (islower (Ch) ? toupper (Ch) : (Ch))
322 # define TOLOWER(Ch, L) (isupper (Ch) ? tolower (Ch) : (Ch))
325 /* We don't use `isdigit' here since the locale dependent
326 interpretation is not what we want here. We only need to accept
327 the arabic digits in the ASCII range. One day there is perhaps a
328 more reliable way to accept other sets of digits. */
329 #define ISDIGIT(Ch) ((unsigned int) (Ch) - L_('0') <= 9)
331 static CHAR_T
*memcpy_lowcase (CHAR_T
*dest
, const CHAR_T
*src
,
332 size_t len LOCALE_PARAM_PROTO
) __THROW
;
335 memcpy_lowcase (dest
, src
, len LOCALE_PARAM
)
342 dest
[len
] = TOLOWER ((UCHAR_T
) src
[len
], loc
);
346 static CHAR_T
*memcpy_uppcase (CHAR_T
*dest
, const CHAR_T
*src
,
347 size_t len LOCALE_PARAM_PROTO
) __THROW
;
350 memcpy_uppcase (dest
, src
, len LOCALE_PARAM
)
357 dest
[len
] = TOUPPER ((UCHAR_T
) src
[len
], loc
);
363 /* Yield the difference between *A and *B,
364 measured in seconds, ignoring leap seconds. */
365 # define tm_diff ftime_tm_diff
366 static int tm_diff (const struct tm
*, const struct tm
*) __THROW
;
372 /* Compute intervening leap days correctly even if year is negative.
373 Take care to avoid int overflow in leap day calculations,
374 but it's OK to assume that A and B are close to each other. */
375 int a4
= (a
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (a
->tm_year
& 3);
376 int b4
= (b
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (b
->tm_year
& 3);
377 int a100
= a4
/ 25 - (a4
% 25 < 0);
378 int b100
= b4
/ 25 - (b4
% 25 < 0);
379 int a400
= a100
>> 2;
380 int b400
= b100
>> 2;
381 int intervening_leap_days
= (a4
- b4
) - (a100
- b100
) + (a400
- b400
);
382 int years
= a
->tm_year
- b
->tm_year
;
383 int days
= (365 * years
+ intervening_leap_days
384 + (a
->tm_yday
- b
->tm_yday
));
385 return (60 * (60 * (24 * days
+ (a
->tm_hour
- b
->tm_hour
))
386 + (a
->tm_min
- b
->tm_min
))
387 + (a
->tm_sec
- b
->tm_sec
));
389 #endif /* ! HAVE_TM_GMTOFF */
393 /* The number of days from the first day of the first ISO week of this
394 year to the year day YDAY with week day WDAY. ISO weeks start on
395 Monday; the first ISO week has the year's first Thursday. YDAY may
396 be as small as YDAY_MINIMUM. */
397 #define ISO_WEEK_START_WDAY 1 /* Monday */
398 #define ISO_WEEK1_WDAY 4 /* Thursday */
399 #define YDAY_MINIMUM (-366)
400 static int iso_week_days (int, int) __THROW
;
405 iso_week_days (yday
, wday
)
409 /* Add enough to the first operand of % to make it nonnegative. */
410 int big_enough_multiple_of_7
= (-YDAY_MINIMUM
/ 7 + 2) * 7;
412 - (yday
- wday
+ ISO_WEEK1_WDAY
+ big_enough_multiple_of_7
) % 7
413 + ISO_WEEK1_WDAY
- ISO_WEEK_START_WDAY
);
417 #if !(defined _NL_CURRENT || HAVE_STRFTIME)
418 static CHAR_T
const weekday_name
[][10] =
420 L_("Sunday"), L_("Monday"), L_("Tuesday"), L_("Wednesday"),
421 L_("Thursday"), L_("Friday"), L_("Saturday")
423 static CHAR_T
const month_name
[][10] =
425 L_("January"), L_("February"), L_("March"), L_("April"), L_("May"),
426 L_("June"), L_("July"), L_("August"), L_("September"), L_("October"),
427 L_("November"), L_("December")
433 # define my_strftime emacs_strftimeu
434 # define ut_argument , ut
435 # define ut_argument_spec int ut;
436 # define ut_argument_spec_iso , int ut
439 # define my_strftime wcsftime
440 # define nl_get_alt_digit _nl_get_walt_digit
442 # define my_strftime strftime
443 # define nl_get_alt_digit _nl_get_alt_digit
446 # define ut_argument_spec
447 # define ut_argument_spec_iso
448 /* We don't have this information in general. */
452 static size_t __strftime_internal (CHAR_T
*, size_t, const CHAR_T
*,
453 const struct tm
*, bool *
455 LOCALE_PARAM_PROTO
) __THROW
;
457 /* Write information from TP into S according to the format
458 string FORMAT, writing no more that MAXSIZE characters
459 (including the terminating '\0') and returning number of
460 characters written. If S is NULL, nothing will be written
461 anywhere, so to determine how many characters would be
462 written, use NULL for S and (size_t) UINT_MAX for MAXSIZE. */
465 my_strftime (s
, maxsize
, format
, tp ut_argument LOCALE_PARAM
)
468 const CHAR_T
*format
;
473 #if !defined _LIBC && HAVE_TZNAME && HAVE_TZSET
474 /* Solaris 2.5 tzset sometimes modifies the storage returned by localtime.
475 Work around this bug by copying *tp before it might be munged. */
480 bool tzset_called
= false;
481 return __strftime_internal (s
, maxsize
, format
, tp
, &tzset_called
482 ut_argument LOCALE_ARG
);
485 libc_hidden_def (my_strftime
)
489 __strftime_internal (s
, maxsize
, format
, tp
, tzset_called ut_argument
493 const CHAR_T
*format
;
499 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
500 struct __locale_data
*const current
= loc
->__locales
[LC_TIME
];
503 int hour12
= tp
->tm_hour
;
505 /* We cannot make the following values variables since we must delay
506 the evaluation of these values until really needed since some
507 expressions might not be valid in every situation. The `struct tm'
508 might be generated by a strptime() call that initialized
509 only a few elements. Dereference the pointers only if the format
510 requires this. Then it is ok to fail if the pointers are invalid. */
512 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABDAY_1) + tp->tm_wday))
514 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(DAY_1) + tp->tm_wday))
516 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABMON_1) + tp->tm_mon))
518 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(MON_1) + tp->tm_mon))
520 ((const CHAR_T *) _NL_CURRENT (LC_TIME, tp->tm_hour > 11 \
521 ? NLW(PM_STR) : NLW(AM_STR)))
523 # define aw_len STRLEN (a_wkday)
524 # define am_len STRLEN (a_month)
525 # define ap_len STRLEN (ampm)
528 # define f_wkday (weekday_name[tp->tm_wday])
529 # define f_month (month_name[tp->tm_mon])
530 # define a_wkday f_wkday
531 # define a_month f_month
532 # define ampm (L_("AMPM") + 2 * (tp->tm_hour > 11))
543 #if DO_MULTIBYTE && !defined COMPILE_WIDE
544 const char *format_end
= NULL
;
549 /* The POSIX test suite assumes that setting
550 the environment variable TZ to a new value before calling strftime()
551 will influence the result (the %Z format) even if the information in
552 TP is computed with a totally different time zone.
553 This is bogus: though POSIX allows bad behavior like this,
554 POSIX does not require it. Do the right thing instead. */
555 zone
= (const char *) tp
->tm_zone
;
560 if (! (zone
&& *zone
))
571 for (f
= format
; *f
!= '\0'; ++f
)
573 int pad
= 0; /* Padding for number ('-', '_', or 0). */
574 int modifier
; /* Field modifier ('E', 'O', or 0). */
575 int digits
; /* Max digits for numeric format. */
576 int number_value
; /* Numeric value to be printed. */
577 int negative_number
; /* 1 if the number is negative. */
578 const CHAR_T
*subfmt
;
580 CHAR_T buf
[1 + (sizeof (int) < sizeof (time_t)
581 ? INT_STRLEN_BOUND (time_t)
582 : INT_STRLEN_BOUND (int))];
589 #if DO_MULTIBYTE && !defined COMPILE_WIDE
595 case L_('\b'): case L_('\t'): case L_('\n'):
596 case L_('\v'): case L_('\f'): case L_('\r'):
597 case L_(' '): case L_('!'): case L_('"'): case L_('#'): case L_('&'):
598 case L_('\''): case L_('('): case L_(')'): case L_('*'): case L_('+'):
599 case L_(','): case L_('-'): case L_('.'): case L_('/'): case L_('0'):
600 case L_('1'): case L_('2'): case L_('3'): case L_('4'): case L_('5'):
601 case L_('6'): case L_('7'): case L_('8'): case L_('9'): case L_(':'):
602 case L_(';'): case L_('<'): case L_('='): case L_('>'): case L_('?'):
603 case L_('A'): case L_('B'): case L_('C'): case L_('D'): case L_('E'):
604 case L_('F'): case L_('G'): case L_('H'): case L_('I'): case L_('J'):
605 case L_('K'): case L_('L'): case L_('M'): case L_('N'): case L_('O'):
606 case L_('P'): case L_('Q'): case L_('R'): case L_('S'): case L_('T'):
607 case L_('U'): case L_('V'): case L_('W'): case L_('X'): case L_('Y'):
608 case L_('Z'): case L_('['): case L_('\\'): case L_(']'): case L_('^'):
609 case L_('_'): case L_('a'): case L_('b'): case L_('c'): case L_('d'):
610 case L_('e'): case L_('f'): case L_('g'): case L_('h'): case L_('i'):
611 case L_('j'): case L_('k'): case L_('l'): case L_('m'): case L_('n'):
612 case L_('o'): case L_('p'): case L_('q'): case L_('r'): case L_('s'):
613 case L_('t'): case L_('u'): case L_('v'): case L_('w'): case L_('x'):
614 case L_('y'): case L_('z'): case L_('{'): case L_('|'): case L_('}'):
616 /* The C Standard requires these 98 characters (plus '%') to
617 be in the basic execution character set. None of these
618 characters can start a multibyte sequence, so they need
619 not be analyzed further. */
624 /* Copy this multibyte sequence until we reach its end, find
625 an error, or come back to the initial shift state. */
627 mbstate_t mbstate
= mbstate_zero
;
632 format_end
= f
+ strlen (f
) + 1;
633 fsize
= format_end
- f
;
637 size_t bytes
= mbrlen (f
+ len
, fsize
- len
, &mbstate
);
642 if (bytes
== (size_t) -2)
644 len
+= strlen (f
+ len
);
648 if (bytes
== (size_t) -1)
656 while (! mbsinit (&mbstate
));
664 #else /* ! DO_MULTIBYTE */
666 /* Either multibyte encodings are not supported, they are
667 safe for formats, so any non-'%' byte can be copied through,
668 or this is the wide character version. */
675 #endif /* ! DO_MULTIBYTE */
677 /* Check for flags that can modify a format. */
682 /* This influences the number formats. */
689 /* This changes textual output. */
703 /* As a GNU extension we allow to specify the field width. */
709 if (width
> INT_MAX
/ 10
710 || (width
== INT_MAX
/ 10 && *f
- L_('0') > INT_MAX
% 10))
711 /* Avoid overflow. */
716 width
+= *f
- L_('0');
720 while (ISDIGIT (*f
));
723 /* Check for modifiers. */
736 /* Now do the specified format. */
740 #define DO_NUMBER(d, v) \
741 digits = d > width ? d : width; \
742 number_value = v; goto do_number
743 #define DO_NUMBER_SPACEPAD(d, v) \
744 digits = d > width ? d : width; \
745 number_value = v; goto do_number_spacepad
761 #if defined _NL_CURRENT || !HAVE_STRFTIME
762 cpy (aw_len
, a_wkday
);
765 goto underlying_strftime
;
776 #if defined _NL_CURRENT || !HAVE_STRFTIME
777 cpy (STRLEN (f_wkday
), f_wkday
);
780 goto underlying_strftime
;
792 #if defined _NL_CURRENT || !HAVE_STRFTIME
793 cpy (am_len
, a_month
);
796 goto underlying_strftime
;
807 #if defined _NL_CURRENT || !HAVE_STRFTIME
808 cpy (STRLEN (f_month
), f_month
);
811 goto underlying_strftime
;
815 if (modifier
== L_('O'))
818 if (! (modifier
== 'E'
820 (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
823 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_T_FMT
));
826 goto underlying_strftime
;
828 subfmt
= L_("%a %b %e %H:%M:%S %Y");
834 CHAR_T
*old_start
= p
;
835 size_t len
= __strftime_internal (NULL
, (size_t) -1, subfmt
,
836 tp
, tzset_called ut_argument
838 add (len
, __strftime_internal (p
, maxsize
- i
, subfmt
,
839 tp
, tzset_called ut_argument
843 while (old_start
< p
)
845 *old_start
= TOUPPER ((UCHAR_T
) *old_start
, loc
);
851 #if HAVE_STRFTIME && ! (defined _NL_CURRENT && HAVE_STRUCT_ERA_ENTRY)
854 /* The relevant information is available only via the
855 underlying strftime implementation, so use that. */
858 char ubuf
[1024]; /* enough for any single format in practice */
860 /* Make sure we're calling the actual underlying strftime.
861 In some cases, config.h contains something like
862 "#define strftime rpl_strftime". */
873 len
= strftime (ubuf
, sizeof ubuf
, ufmt
, tp
);
874 if (len
== 0 && ubuf
[0] != '\0')
882 if (modifier
== L_('E'))
884 #if HAVE_STRUCT_ERA_ENTRY
885 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
889 size_t len
= __wcslen (era
->era_wname
);
890 cpy (len
, era
->era_wname
);
892 size_t len
= strlen (era
->era_name
);
893 cpy (len
, era
->era_name
);
899 goto underlying_strftime
;
905 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
906 DO_NUMBER (1, year
/ 100 - (year
% 100 < 0));
910 if (modifier
== L_('O'))
913 if (! (modifier
== L_('E')
915 (const CHAR_T
*)_NL_CURRENT (LC_TIME
, NLW(ERA_D_FMT
)))
917 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_FMT
));
921 goto underlying_strftime
;
929 subfmt
= L_("%m/%d/%y");
933 if (modifier
== L_('E'))
936 DO_NUMBER (2, tp
->tm_mday
);
939 if (modifier
== L_('E'))
942 DO_NUMBER_SPACEPAD (2, tp
->tm_mday
);
944 /* All numeric formats set DIGITS and NUMBER_VALUE and then
945 jump to one of these two labels. */
948 /* Force `_' flag unless overwritten by `0' or '-' flag. */
949 if (pad
!= L_('0') && pad
!= L_('-'))
953 /* Format the number according to the MODIFIER flag. */
955 if (modifier
== L_('O') && 0 <= number_value
)
958 /* Get the locale specific alternate representation of
959 the number NUMBER_VALUE. If none exist NULL is returned. */
960 const CHAR_T
*cp
= nl_get_alt_digit (number_value
965 size_t digitlen
= STRLEN (cp
);
974 goto underlying_strftime
;
979 unsigned int u
= number_value
;
981 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
982 negative_number
= number_value
< 0;
988 *--bufp
= u
% 10 + L_('0');
989 while ((u
/= 10) != 0);
992 do_number_sign_and_padding
:
998 int padding
= digits
- (buf
+ (sizeof (buf
) / sizeof (buf
[0]))
1005 if ((size_t) padding
>= maxsize
- i
)
1009 memset_space (p
, padding
);
1011 width
= width
> padding
? width
- padding
: 0;
1015 if ((size_t) digits
>= maxsize
- i
)
1018 if (negative_number
)
1028 memset_zero (p
, padding
);
1035 cpy (buf
+ sizeof (buf
) / sizeof (buf
[0]) - bufp
, bufp
);
1041 subfmt
= L_("%Y-%m-%d");
1045 if (modifier
== L_('E'))
1048 DO_NUMBER (2, tp
->tm_hour
);
1051 if (modifier
== L_('E'))
1054 DO_NUMBER (2, hour12
);
1056 case L_('k'): /* GNU extension. */
1057 if (modifier
== L_('E'))
1060 DO_NUMBER_SPACEPAD (2, tp
->tm_hour
);
1062 case L_('l'): /* GNU extension. */
1063 if (modifier
== L_('E'))
1066 DO_NUMBER_SPACEPAD (2, hour12
);
1069 if (modifier
== L_('E'))
1072 DO_NUMBER (3, 1 + tp
->tm_yday
);
1075 if (modifier
== L_('E'))
1078 DO_NUMBER (2, tp
->tm_min
);
1081 if (modifier
== L_('E'))
1084 DO_NUMBER (2, tp
->tm_mon
+ 1);
1087 add (1, *p
= L_('\n'));
1092 #if !defined _NL_CURRENT && HAVE_STRFTIME
1093 format_char
= L_('p');
1103 #if defined _NL_CURRENT || !HAVE_STRFTIME
1107 goto underlying_strftime
;
1111 subfmt
= L_("%H:%M");
1115 #if !defined _NL_CURRENT && HAVE_STRFTIME
1116 goto underlying_strftime
;
1119 if (*(subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
1123 subfmt
= L_("%I:%M:%S %p");
1128 if (modifier
== L_('E'))
1131 DO_NUMBER (2, tp
->tm_sec
);
1133 case L_('s'): /* GNU extension. */
1141 /* Generate string value for T using time_t arithmetic;
1142 this works even if sizeof (long) < sizeof (time_t). */
1144 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
1145 negative_number
= t
< 0;
1152 if (negative_number
)
1156 /* Adjust if division truncates to minus infinity. */
1157 if (0 < -1 % 10 && d
< 0)
1164 *--bufp
= d
+ L_('0');
1169 goto do_number_sign_and_padding
;
1173 if (modifier
== L_('O'))
1176 if (! (modifier
== L_('E')
1178 (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(ERA_T_FMT
)))
1180 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(T_FMT
));
1184 goto underlying_strftime
;
1190 subfmt
= L_("%H:%M:%S");
1194 add (1, *p
= L_('\t'));
1198 DO_NUMBER (1, (tp
->tm_wday
- 1 + 7) % 7 + 1);
1201 if (modifier
== L_('E'))
1204 DO_NUMBER (2, (tp
->tm_yday
- tp
->tm_wday
+ 7) / 7);
1209 if (modifier
== L_('E'))
1212 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
1213 int days
= iso_week_days (tp
->tm_yday
, tp
->tm_wday
);
1217 /* This ISO week belongs to the previous year. */
1219 days
= iso_week_days (tp
->tm_yday
+ (365 + __isleap (year
)),
1224 int d
= iso_week_days (tp
->tm_yday
- (365 + __isleap (year
)),
1228 /* This ISO week belongs to the next year. */
1237 DO_NUMBER (2, (year
% 100 + 100) % 100);
1240 DO_NUMBER (1, year
);
1243 DO_NUMBER (2, days
/ 7 + 1);
1248 if (modifier
== L_('E'))
1251 DO_NUMBER (2, (tp
->tm_yday
- (tp
->tm_wday
- 1 + 7) % 7 + 7) / 7);
1254 if (modifier
== L_('E'))
1257 DO_NUMBER (1, tp
->tm_wday
);
1260 if (modifier
== 'E')
1262 #if HAVE_STRUCT_ERA_ENTRY
1263 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1266 # ifdef COMPILE_WIDE
1267 subfmt
= era
->era_wformat
;
1269 subfmt
= era
->era_format
;
1275 goto underlying_strftime
;
1279 if (modifier
== L_('O'))
1282 DO_NUMBER (1, tp
->tm_year
+ TM_YEAR_BASE
);
1285 if (modifier
== L_('E'))
1287 #if HAVE_STRUCT_ERA_ENTRY
1288 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1291 int delta
= tp
->tm_year
- era
->start_date
[0];
1292 DO_NUMBER (1, (era
->offset
1293 + delta
* era
->absolute_direction
));
1297 goto underlying_strftime
;
1301 DO_NUMBER (2, (tp
->tm_year
% 100 + 100) % 100);
1311 /* The tzset() call might have changed the value. */
1312 if (!(zone
&& *zone
) && tp
->tm_isdst
>= 0)
1314 /* POSIX.1 requires that local time zone information is used as
1315 though strftime called tzset. */
1320 *tzset_called
= true;
1323 zone
= tzname
[tp
->tm_isdst
];
1331 /* The zone string is always given in multibyte form. We have
1332 to transform it first. */
1335 widen (zone
, wczone
, len
);
1339 cpy (strlen (zone
), zone
);
1344 if (tp
->tm_isdst
< 0)
1350 diff
= tp
->tm_gmtoff
;
1360 /* POSIX.1 requires that local time zone information is used as
1361 though strftime called tzset. */
1366 *tzset_called
= true;
1373 if (lt
== (time_t) -1)
1375 /* mktime returns -1 for errors, but -1 is also a
1376 valid time_t value. Check whether an error really
1380 if (! __localtime_r (<
, &tm
)
1381 || ((ltm
.tm_sec
^ tm
.tm_sec
)
1382 | (ltm
.tm_min
^ tm
.tm_min
)
1383 | (ltm
.tm_hour
^ tm
.tm_hour
)
1384 | (ltm
.tm_mday
^ tm
.tm_mday
)
1385 | (ltm
.tm_mon
^ tm
.tm_mon
)
1386 | (ltm
.tm_year
^ tm
.tm_year
)))
1390 if (! __gmtime_r (<
, >m
))
1393 diff
= tm_diff (<m
, >m
);
1399 add (1, *p
= L_('-'));
1403 add (1, *p
= L_('+'));
1406 DO_NUMBER (4, (diff
/ 60) * 100 + diff
% 60);
1409 case L_('\0'): /* GNU extension: % at end of format. */
1413 /* Unknown format; output the format, including the '%',
1414 since this is most likely the right thing to do if a
1415 multibyte string has been misparsed. */
1419 for (flen
= 1; f
[1 - flen
] != L_('%'); flen
++)
1421 cpy (flen
, &f
[1 - flen
]);
1427 if (p
&& maxsize
!= 0)
1434 /* For Emacs we have a separate interface which corresponds to the normal
1435 strftime function and does not have the extra information whether the
1436 TP arguments comes from a `gmtime' call or not. */
1438 emacs_strftime (s
, maxsize
, format
, tp
)
1442 const struct tm
*tp
;
1444 return my_strftime (s
, maxsize
, format
, tp
, 0);
1448 #if defined _LIBC && !defined COMPILE_WIDE
1449 weak_alias (__strftime_l
, strftime_l
)