1 /* Copyright (C) 2002, 2004, 2007 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, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
24 # define USE_IN_EXTENDED_LOCALE_MODEL 1
25 # define HAVE_LIMITS_H 1
27 # define HAVE_MBRLEN 1
28 # define HAVE_STRUCT_ERA_ENTRY 1
29 # define HAVE_TM_GMTOFF 1
30 # define HAVE_TM_ZONE 1
31 # define HAVE_TZNAME 1
33 # define MULTIBYTE_IS_FORMAT_SAFE 1
34 # define STDC_HEADERS 1
35 # include "../locale/localeinfo.h"
38 #if defined emacs && !defined HAVE_BCOPY
39 # define HAVE_MEMCPY 1
43 #include <sys/types.h> /* Some systems define `time_t' here. */
45 #ifdef TIME_WITH_SYS_TIME
46 # include <sys/time.h>
49 # ifdef HAVE_SYS_TIME_H
50 # include <sys/time.h>
56 extern char *tzname
[];
59 /* Do multibyte processing if multibytes are supported, unless
60 multibyte sequences are safe in formats. Multibyte sequences are
61 safe if they cannot contain byte sequences that look like format
62 conversion specifications. The GNU C Library uses UTF8 multibyte
63 encoding, which is safe for formats, but strftime.c can be used
64 with other C libraries that use unsafe encodings. */
65 #define DO_MULTIBYTE (HAVE_MBLEN && ! MULTIBYTE_IS_FORMAT_SAFE)
71 /* Simulate mbrlen with mblen as best we can. */
72 # define mbstate_t int
73 # define mbrlen(s, n, ps) mblen (s, n)
74 # define mbsinit(ps) (*(ps) == 0)
76 static const mbstate_t mbstate_zero
;
90 # define memcpy(d, s, n) bcopy ((s), (d), (n))
96 # define CHAR_T wchar_t
97 # define UCHAR_T unsigned int
98 # define L_(Str) L##Str
99 # define NLW(Sym) _NL_W##Sym
101 # define MEMCPY(d, s, n) __wmemcpy (d, s, n)
102 # define STRLEN(s) __wcslen (s)
106 # define UCHAR_T unsigned char
108 # define NLW(Sym) Sym
110 # if !defined STDC_HEADERS && !defined HAVE_MEMCPY
111 # define MEMCPY(d, s, n) bcopy ((s), (d), (n))
113 # define MEMCPY(d, s, n) memcpy ((d), (s), (n))
115 # define STRLEN(s) strlen (s)
118 # define MEMPCPY(d, s, n) __mempcpy (d, s, n)
120 # ifndef HAVE_MEMPCPY
121 # define MEMPCPY(d, s, n) ((void *) ((char *) memcpy (d, s, n) + (n)))
142 #define TYPE_SIGNED(t) ((t) -1 < 0)
144 /* Bound on length of the string representing an integer value of type t.
145 Subtract one for the sign bit if t is signed;
146 302 / 1000 is log10 (2) rounded up;
147 add one for integer division truncation;
148 add one more for a minus sign if t is signed. */
149 #define INT_STRLEN_BOUND(t) \
150 ((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 + 1 + TYPE_SIGNED (t))
152 #define TM_YEAR_BASE 1900
155 /* Nonzero if YEAR is a leap year (every 4 years,
156 except every 100th isn't, and every 400th is). */
157 # define __isleap(year) \
158 ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
163 # define tzname __tzname
164 # define tzset __tzset
168 /* Portable standalone applications should supply a "time_r.h" that
169 declares a POSIX-compliant localtime_r, for the benefit of older
170 implementations that lack localtime_r or have a nonstandard one.
171 Similarly for gmtime_r. See the gnulib time_r module for one way
172 to implement this. */
175 # undef __localtime_r
176 # define __gmtime_r gmtime_r
177 # define __localtime_r localtime_r
181 #if !defined memset && !defined HAVE_MEMSET && !defined _LIBC
182 /* Some systems lack the `memset' function and we don't want to
183 introduce additional dependencies. */
184 /* The SGI compiler reportedly barfs on the trailing null
185 if we use a string constant as the initializer. 28 June 1997, rms. */
186 static const CHAR_T spaces
[16] = /* " " */
188 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),
189 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' ')
191 static const CHAR_T zeroes
[16] = /* "0000000000000000" */
193 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),
194 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0')
197 # define memset_space(P, Len) \
203 int _this = _len > 16 ? 16 : _len; \
204 (P) = MEMPCPY ((P), spaces, _this * sizeof (CHAR_T)); \
210 # define memset_zero(P, Len) \
216 int _this = _len > 16 ? 16 : _len; \
217 (P) = MEMPCPY ((P), zeroes, _this * sizeof (CHAR_T)); \
224 # define memset_space(P, Len) (wmemset ((P), L' ', (Len)), (P) += (Len))
225 # define memset_zero(P, Len) (wmemset ((P), L'0', (Len)), (P) += (Len))
227 # define memset_space(P, Len) (memset ((P), ' ', (Len)), (P) += (Len))
228 # define memset_zero(P, Len) (memset ((P), '0', (Len)), (P) += (Len))
236 int _delta = width - _n; \
237 int _incr = _n + (_delta > 0 ? _delta : 0); \
238 if ((size_t) _incr >= maxsize - i) \
244 if (pad == L_('0')) \
245 memset_zero (p, _delta); \
247 memset_space (p, _delta); \
258 memcpy_lowcase (p, (s), _n LOCALE_ARG); \
259 else if (to_uppcase) \
260 memcpy_uppcase (p, (s), _n LOCALE_ARG); \
262 MEMCPY ((PTR) p, (const PTR) (s), _n))
265 # ifndef USE_IN_EXTENDED_LOCALE_MODEL
266 # undef __mbsrtowcs_l
267 # define __mbsrtowcs_l(d, s, l, st, loc) __mbsrtowcs (d, s, l, st)
269 # define widen(os, ws, l) \
272 const char *__s = os; \
273 memset (&__st, '\0', sizeof (__st)); \
274 l = __mbsrtowcs_l (NULL, &__s, 0, &__st, loc); \
275 ws = alloca ((l + 1) * sizeof (wchar_t)); \
276 (void) __mbsrtowcs_l (ws, &__s, l, &__st, loc); \
281 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
282 /* We use this code also for the extended locale handling where the
283 function gets as an additional argument the locale which has to be
284 used. To access the values we have to redefine the _NL_CURRENT
286 # define strftime __strftime_l
287 # define wcsftime __wcsftime_l
289 # define _NL_CURRENT(category, item) \
290 (current->values[_NL_ITEM_INDEX (item)].string)
291 # define LOCALE_PARAM , loc
292 # define LOCALE_ARG , loc
293 # define LOCALE_PARAM_DECL __locale_t loc;
294 # define LOCALE_PARAM_PROTO , __locale_t loc
295 # define HELPER_LOCALE_ARG , current
297 # define LOCALE_PARAM
298 # define LOCALE_PARAM_PROTO
300 # define LOCALE_PARAM_DECL
302 # define HELPER_LOCALE_ARG , _NL_CURRENT_DATA (LC_TIME)
304 # define HELPER_LOCALE_ARG
309 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
310 # define TOUPPER(Ch, L) __towupper_l (Ch, L)
311 # define TOLOWER(Ch, L) __towlower_l (Ch, L)
313 # define TOUPPER(Ch, L) towupper (Ch)
314 # define TOLOWER(Ch, L) towlower (Ch)
318 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
319 # define TOUPPER(Ch, L) __toupper_l (Ch, L)
320 # define TOLOWER(Ch, L) __tolower_l (Ch, L)
322 # define TOUPPER(Ch, L) toupper (Ch)
323 # define TOLOWER(Ch, L) tolower (Ch)
326 # define TOUPPER(Ch, L) (islower (Ch) ? toupper (Ch) : (Ch))
327 # define TOLOWER(Ch, L) (isupper (Ch) ? tolower (Ch) : (Ch))
330 /* We don't use `isdigit' here since the locale dependent
331 interpretation is not what we want here. We only need to accept
332 the arabic digits in the ASCII range. One day there is perhaps a
333 more reliable way to accept other sets of digits. */
334 #define ISDIGIT(Ch) ((unsigned int) (Ch) - L_('0') <= 9)
336 static CHAR_T
*memcpy_lowcase (CHAR_T
*dest
, const CHAR_T
*src
,
337 size_t len LOCALE_PARAM_PROTO
) __THROW
;
340 memcpy_lowcase (dest
, src
, len LOCALE_PARAM
)
347 dest
[len
] = TOLOWER ((UCHAR_T
) src
[len
], loc
);
351 static CHAR_T
*memcpy_uppcase (CHAR_T
*dest
, const CHAR_T
*src
,
352 size_t len LOCALE_PARAM_PROTO
) __THROW
;
355 memcpy_uppcase (dest
, src
, len LOCALE_PARAM
)
362 dest
[len
] = TOUPPER ((UCHAR_T
) src
[len
], loc
);
368 /* Yield the difference between *A and *B,
369 measured in seconds, ignoring leap seconds. */
370 # define tm_diff ftime_tm_diff
371 static int tm_diff (const struct tm
*, const struct tm
*) __THROW
;
377 /* Compute intervening leap days correctly even if year is negative.
378 Take care to avoid int overflow in leap day calculations,
379 but it's OK to assume that A and B are close to each other. */
380 int a4
= (a
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (a
->tm_year
& 3);
381 int b4
= (b
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (b
->tm_year
& 3);
382 int a100
= a4
/ 25 - (a4
% 25 < 0);
383 int b100
= b4
/ 25 - (b4
% 25 < 0);
384 int a400
= a100
>> 2;
385 int b400
= b100
>> 2;
386 int intervening_leap_days
= (a4
- b4
) - (a100
- b100
) + (a400
- b400
);
387 int years
= a
->tm_year
- b
->tm_year
;
388 int days
= (365 * years
+ intervening_leap_days
389 + (a
->tm_yday
- b
->tm_yday
));
390 return (60 * (60 * (24 * days
+ (a
->tm_hour
- b
->tm_hour
))
391 + (a
->tm_min
- b
->tm_min
))
392 + (a
->tm_sec
- b
->tm_sec
));
394 #endif /* ! HAVE_TM_GMTOFF */
398 /* The number of days from the first day of the first ISO week of this
399 year to the year day YDAY with week day WDAY. ISO weeks start on
400 Monday; the first ISO week has the year's first Thursday. YDAY may
401 be as small as YDAY_MINIMUM. */
402 #define ISO_WEEK_START_WDAY 1 /* Monday */
403 #define ISO_WEEK1_WDAY 4 /* Thursday */
404 #define YDAY_MINIMUM (-366)
405 static int iso_week_days (int, int) __THROW
;
410 iso_week_days (yday
, wday
)
414 /* Add enough to the first operand of % to make it nonnegative. */
415 int big_enough_multiple_of_7
= (-YDAY_MINIMUM
/ 7 + 2) * 7;
417 - (yday
- wday
+ ISO_WEEK1_WDAY
+ big_enough_multiple_of_7
) % 7
418 + ISO_WEEK1_WDAY
- ISO_WEEK_START_WDAY
);
422 #if !(defined _NL_CURRENT || HAVE_STRFTIME)
423 static CHAR_T
const weekday_name
[][10] =
425 L_("Sunday"), L_("Monday"), L_("Tuesday"), L_("Wednesday"),
426 L_("Thursday"), L_("Friday"), L_("Saturday")
428 static CHAR_T
const month_name
[][10] =
430 L_("January"), L_("February"), L_("March"), L_("April"), L_("May"),
431 L_("June"), L_("July"), L_("August"), L_("September"), L_("October"),
432 L_("November"), L_("December")
438 # define my_strftime emacs_strftimeu
439 # define ut_argument , ut
440 # define ut_argument_spec int ut;
441 # define ut_argument_spec_iso , int ut
444 # define my_strftime wcsftime
445 # define nl_get_alt_digit _nl_get_walt_digit
447 # define my_strftime strftime
448 # define nl_get_alt_digit _nl_get_alt_digit
451 # define ut_argument_spec
452 # define ut_argument_spec_iso
453 /* We don't have this information in general. */
457 static size_t __strftime_internal (CHAR_T
*, size_t, const CHAR_T
*,
458 const struct tm
*, bool ut_argument_spec_iso
459 LOCALE_PARAM_PROTO
) __THROW
;
461 /* Write information from TP into S according to the format
462 string FORMAT, writing no more that MAXSIZE characters
463 (including the terminating '\0') and returning number of
464 characters written. If S is NULL, nothing will be written
465 anywhere, so to determine how many characters would be
466 written, use NULL for S and (size_t) UINT_MAX for MAXSIZE. */
469 my_strftime (s
, maxsize
, format
, tp ut_argument LOCALE_PARAM
)
472 const CHAR_T
*format
;
477 #if !defined _LIBC && HAVE_TZNAME && HAVE_TZSET
478 /* Solaris 2.5 tzset sometimes modifies the storage returned by localtime.
479 Work around this bug by copying *tp before it might be munged. */
484 return __strftime_internal (s
, maxsize
, format
, tp
, false
485 ut_argument LOCALE_ARG
);
488 libc_hidden_def (my_strftime
)
492 __strftime_internal (s
, maxsize
, format
, tp
, tzset_called ut_argument
496 const CHAR_T
*format
;
502 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
503 struct locale_data
*const current
= loc
->__locales
[LC_TIME
];
506 int hour12
= tp
->tm_hour
;
508 /* We cannot make the following values variables since we must delay
509 the evaluation of these values until really needed since some
510 expressions might not be valid in every situation. The `struct tm'
511 might be generated by a strptime() call that initialized
512 only a few elements. Dereference the pointers only if the format
513 requires this. Then it is ok to fail if the pointers are invalid. */
515 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABDAY_1) + tp->tm_wday))
517 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(DAY_1) + tp->tm_wday))
519 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABMON_1) + tp->tm_mon))
521 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(MON_1) + tp->tm_mon))
523 ((const CHAR_T *) _NL_CURRENT (LC_TIME, tp->tm_hour > 11 \
524 ? NLW(PM_STR) : NLW(AM_STR)))
526 # define aw_len STRLEN (a_wkday)
527 # define am_len STRLEN (a_month)
528 # define ap_len STRLEN (ampm)
531 # define f_wkday (weekday_name[tp->tm_wday])
532 # define f_month (month_name[tp->tm_mon])
533 # define a_wkday f_wkday
534 # define a_month f_month
535 # define ampm (L_("AMPM") + 2 * (tp->tm_hour > 11))
546 #if DO_MULTIBYTE && !defined COMPILE_WIDE
547 const char *format_end
= NULL
;
552 /* The POSIX test suite assumes that setting
553 the environment variable TZ to a new value before calling strftime()
554 will influence the result (the %Z format) even if the information in
555 TP is computed with a totally different time zone.
556 This is bogus: though POSIX allows bad behavior like this,
557 POSIX does not require it. Do the right thing instead. */
558 zone
= (const char *) tp
->tm_zone
;
563 if (! (zone
&& *zone
))
568 /* POSIX.1 requires that local time zone information is used as
569 though strftime called tzset. */
584 for (f
= format
; *f
!= '\0'; ++f
)
586 int pad
= 0; /* Padding for number ('-', '_', or 0). */
587 int modifier
; /* Field modifier ('E', 'O', or 0). */
588 int digits
; /* Max digits for numeric format. */
589 int number_value
; /* Numeric value to be printed. */
590 int negative_number
; /* 1 if the number is negative. */
591 const CHAR_T
*subfmt
;
593 CHAR_T buf
[1 + (sizeof (int) < sizeof (time_t)
594 ? INT_STRLEN_BOUND (time_t)
595 : INT_STRLEN_BOUND (int))];
602 #if DO_MULTIBYTE && !defined COMPILE_WIDE
608 case L_('\b'): case L_('\t'): case L_('\n'):
609 case L_('\v'): case L_('\f'): case L_('\r'):
610 case L_(' '): case L_('!'): case L_('"'): case L_('#'): case L_('&'):
611 case L_('\''): case L_('('): case L_(')'): case L_('*'): case L_('+'):
612 case L_(','): case L_('-'): case L_('.'): case L_('/'): case L_('0'):
613 case L_('1'): case L_('2'): case L_('3'): case L_('4'): case L_('5'):
614 case L_('6'): case L_('7'): case L_('8'): case L_('9'): case L_(':'):
615 case L_(';'): case L_('<'): case L_('='): case L_('>'): case L_('?'):
616 case L_('A'): case L_('B'): case L_('C'): case L_('D'): case L_('E'):
617 case L_('F'): case L_('G'): case L_('H'): case L_('I'): case L_('J'):
618 case L_('K'): case L_('L'): case L_('M'): case L_('N'): case L_('O'):
619 case L_('P'): case L_('Q'): case L_('R'): case L_('S'): case L_('T'):
620 case L_('U'): case L_('V'): case L_('W'): case L_('X'): case L_('Y'):
621 case L_('Z'): case L_('['): case L_('\\'): case L_(']'): case L_('^'):
622 case L_('_'): case L_('a'): case L_('b'): case L_('c'): case L_('d'):
623 case L_('e'): case L_('f'): case L_('g'): case L_('h'): case L_('i'):
624 case L_('j'): case L_('k'): case L_('l'): case L_('m'): case L_('n'):
625 case L_('o'): case L_('p'): case L_('q'): case L_('r'): case L_('s'):
626 case L_('t'): case L_('u'): case L_('v'): case L_('w'): case L_('x'):
627 case L_('y'): case L_('z'): case L_('{'): case L_('|'): case L_('}'):
629 /* The C Standard requires these 98 characters (plus '%') to
630 be in the basic execution character set. None of these
631 characters can start a multibyte sequence, so they need
632 not be analyzed further. */
637 /* Copy this multibyte sequence until we reach its end, find
638 an error, or come back to the initial shift state. */
640 mbstate_t mbstate
= mbstate_zero
;
645 format_end
= f
+ strlen (f
) + 1;
646 fsize
= format_end
- f
;
650 size_t bytes
= mbrlen (f
+ len
, fsize
- len
, &mbstate
);
655 if (bytes
== (size_t) -2)
657 len
+= strlen (f
+ len
);
661 if (bytes
== (size_t) -1)
669 while (! mbsinit (&mbstate
));
677 #else /* ! DO_MULTIBYTE */
679 /* Either multibyte encodings are not supported, they are
680 safe for formats, so any non-'%' byte can be copied through,
681 or this is the wide character version. */
688 #endif /* ! DO_MULTIBYTE */
690 /* Check for flags that can modify a format. */
695 /* This influences the number formats. */
702 /* This changes textual output. */
716 /* As a GNU extension we allow to specify the field width. */
722 if (width
> INT_MAX
/ 10
723 || (width
== INT_MAX
/ 10 && *f
- L_('0') > INT_MAX
% 10))
724 /* Avoid overflow. */
729 width
+= *f
- L_('0');
733 while (ISDIGIT (*f
));
736 /* Check for modifiers. */
749 /* Now do the specified format. */
753 #define DO_NUMBER(d, v) \
754 digits = d > width ? d : width; \
755 number_value = v; goto do_number
756 #define DO_NUMBER_SPACEPAD(d, v) \
757 digits = d > width ? d : width; \
758 number_value = v; goto do_number_spacepad
774 #if defined _NL_CURRENT || !HAVE_STRFTIME
775 cpy (aw_len
, a_wkday
);
778 goto underlying_strftime
;
789 #if defined _NL_CURRENT || !HAVE_STRFTIME
790 cpy (STRLEN (f_wkday
), f_wkday
);
793 goto underlying_strftime
;
805 #if defined _NL_CURRENT || !HAVE_STRFTIME
806 cpy (am_len
, a_month
);
809 goto underlying_strftime
;
820 #if defined _NL_CURRENT || !HAVE_STRFTIME
821 cpy (STRLEN (f_month
), f_month
);
824 goto underlying_strftime
;
828 if (modifier
== L_('O'))
831 if (! (modifier
== 'E'
833 (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
836 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_T_FMT
));
839 goto underlying_strftime
;
841 subfmt
= L_("%a %b %e %H:%M:%S %Y");
847 CHAR_T
*old_start
= p
;
848 size_t len
= __strftime_internal (NULL
, (size_t) -1, subfmt
,
849 tp
, tzset_called ut_argument
851 add (len
, __strftime_internal (p
, maxsize
- i
, subfmt
,
852 tp
, tzset_called ut_argument
856 while (old_start
< p
)
858 *old_start
= TOUPPER ((UCHAR_T
) *old_start
, loc
);
864 #if HAVE_STRFTIME && ! (defined _NL_CURRENT && HAVE_STRUCT_ERA_ENTRY)
867 /* The relevant information is available only via the
868 underlying strftime implementation, so use that. */
871 char ubuf
[1024]; /* enough for any single format in practice */
873 /* Make sure we're calling the actual underlying strftime.
874 In some cases, config.h contains something like
875 "#define strftime rpl_strftime". */
886 len
= strftime (ubuf
, sizeof ubuf
, ufmt
, tp
);
887 if (len
== 0 && ubuf
[0] != '\0')
895 if (modifier
== L_('O'))
897 if (modifier
== L_('E'))
899 #if HAVE_STRUCT_ERA_ENTRY
900 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
904 size_t len
= __wcslen (era
->era_wname
);
905 cpy (len
, era
->era_wname
);
907 size_t len
= strlen (era
->era_name
);
908 cpy (len
, era
->era_name
);
914 goto underlying_strftime
;
920 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
921 DO_NUMBER (1, year
/ 100 - (year
% 100 < 0));
925 if (modifier
== L_('O'))
928 if (! (modifier
== L_('E')
930 (const CHAR_T
*)_NL_CURRENT (LC_TIME
, NLW(ERA_D_FMT
)))
932 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_FMT
));
936 goto underlying_strftime
;
944 subfmt
= L_("%m/%d/%y");
948 if (modifier
== L_('E'))
951 DO_NUMBER (2, tp
->tm_mday
);
954 if (modifier
== L_('E'))
957 DO_NUMBER_SPACEPAD (2, tp
->tm_mday
);
959 /* All numeric formats set DIGITS and NUMBER_VALUE and then
960 jump to one of these two labels. */
963 /* Force `_' flag unless overwritten by `0' or '-' flag. */
964 if (pad
!= L_('0') && pad
!= L_('-'))
968 /* Format the number according to the MODIFIER flag. */
970 if (modifier
== L_('O') && 0 <= number_value
)
973 /* Get the locale specific alternate representation of
974 the number NUMBER_VALUE. If none exist NULL is returned. */
975 const CHAR_T
*cp
= nl_get_alt_digit (number_value
980 size_t digitlen
= STRLEN (cp
);
989 goto underlying_strftime
;
994 unsigned int u
= number_value
;
996 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
997 negative_number
= number_value
< 0;
1003 *--bufp
= u
% 10 + L_('0');
1004 while ((u
/= 10) != 0);
1007 do_number_sign_and_padding
:
1008 if (negative_number
)
1013 int padding
= digits
- (buf
+ (sizeof (buf
) / sizeof (buf
[0]))
1020 if ((size_t) padding
>= maxsize
- i
)
1024 memset_space (p
, padding
);
1026 width
= width
> padding
? width
- padding
: 0;
1030 if ((size_t) digits
>= maxsize
- i
)
1033 if (negative_number
)
1043 memset_zero (p
, padding
);
1050 cpy (buf
+ sizeof (buf
) / sizeof (buf
[0]) - bufp
, bufp
);
1056 subfmt
= L_("%Y-%m-%d");
1060 if (modifier
== L_('E'))
1063 DO_NUMBER (2, tp
->tm_hour
);
1066 if (modifier
== L_('E'))
1069 DO_NUMBER (2, hour12
);
1071 case L_('k'): /* GNU extension. */
1072 if (modifier
== L_('E'))
1075 DO_NUMBER_SPACEPAD (2, tp
->tm_hour
);
1077 case L_('l'): /* GNU extension. */
1078 if (modifier
== L_('E'))
1081 DO_NUMBER_SPACEPAD (2, hour12
);
1084 if (modifier
== L_('E'))
1087 DO_NUMBER (3, 1 + tp
->tm_yday
);
1090 if (modifier
== L_('E'))
1093 DO_NUMBER (2, tp
->tm_min
);
1096 if (modifier
== L_('E'))
1099 DO_NUMBER (2, tp
->tm_mon
+ 1);
1102 add (1, *p
= L_('\n'));
1107 #if !defined _NL_CURRENT && HAVE_STRFTIME
1108 format_char
= L_('p');
1118 #if defined _NL_CURRENT || !HAVE_STRFTIME
1122 goto underlying_strftime
;
1126 subfmt
= L_("%H:%M");
1130 #if !defined _NL_CURRENT && HAVE_STRFTIME
1131 goto underlying_strftime
;
1134 if (*(subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
1138 subfmt
= L_("%I:%M:%S %p");
1143 if (modifier
== L_('E'))
1146 DO_NUMBER (2, tp
->tm_sec
);
1148 case L_('s'): /* GNU extension. */
1156 /* Generate string value for T using time_t arithmetic;
1157 this works even if sizeof (long) < sizeof (time_t). */
1159 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
1160 negative_number
= t
< 0;
1167 if (negative_number
)
1171 /* Adjust if division truncates to minus infinity. */
1172 if (0 < -1 % 10 && d
< 0)
1179 *--bufp
= d
+ L_('0');
1184 goto do_number_sign_and_padding
;
1188 if (modifier
== L_('O'))
1191 if (! (modifier
== L_('E')
1193 (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(ERA_T_FMT
)))
1195 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(T_FMT
));
1199 goto underlying_strftime
;
1205 subfmt
= L_("%H:%M:%S");
1209 add (1, *p
= L_('\t'));
1213 DO_NUMBER (1, (tp
->tm_wday
- 1 + 7) % 7 + 1);
1216 if (modifier
== L_('E'))
1219 DO_NUMBER (2, (tp
->tm_yday
- tp
->tm_wday
+ 7) / 7);
1224 if (modifier
== L_('E'))
1227 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
1228 int days
= iso_week_days (tp
->tm_yday
, tp
->tm_wday
);
1232 /* This ISO week belongs to the previous year. */
1234 days
= iso_week_days (tp
->tm_yday
+ (365 + __isleap (year
)),
1239 int d
= iso_week_days (tp
->tm_yday
- (365 + __isleap (year
)),
1243 /* This ISO week belongs to the next year. */
1252 DO_NUMBER (2, (year
% 100 + 100) % 100);
1255 DO_NUMBER (1, year
);
1258 DO_NUMBER (2, days
/ 7 + 1);
1263 if (modifier
== L_('E'))
1266 DO_NUMBER (2, (tp
->tm_yday
- (tp
->tm_wday
- 1 + 7) % 7 + 7) / 7);
1269 if (modifier
== L_('E'))
1272 DO_NUMBER (1, tp
->tm_wday
);
1275 if (modifier
== 'E')
1277 #if HAVE_STRUCT_ERA_ENTRY
1278 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1281 # ifdef COMPILE_WIDE
1282 subfmt
= era
->era_wformat
;
1284 subfmt
= era
->era_format
;
1290 goto underlying_strftime
;
1294 if (modifier
== L_('O'))
1297 DO_NUMBER (1, tp
->tm_year
+ TM_YEAR_BASE
);
1300 if (modifier
== L_('E'))
1302 #if HAVE_STRUCT_ERA_ENTRY
1303 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1306 int delta
= tp
->tm_year
- era
->start_date
[0];
1307 DO_NUMBER (1, (era
->offset
1308 + delta
* era
->absolute_direction
));
1312 goto underlying_strftime
;
1316 DO_NUMBER (2, (tp
->tm_year
% 100 + 100) % 100);
1326 /* The tzset() call might have changed the value. */
1327 if (!(zone
&& *zone
) && tp
->tm_isdst
>= 0)
1328 zone
= tzname
[tp
->tm_isdst
];
1335 /* The zone string is always given in multibyte form. We have
1336 to transform it first. */
1339 widen (zone
, wczone
, len
);
1343 cpy (strlen (zone
), zone
);
1348 if (tp
->tm_isdst
< 0)
1354 diff
= tp
->tm_gmtoff
;
1367 if (lt
== (time_t) -1)
1369 /* mktime returns -1 for errors, but -1 is also a
1370 valid time_t value. Check whether an error really
1374 if (! __localtime_r (<
, &tm
)
1375 || ((ltm
.tm_sec
^ tm
.tm_sec
)
1376 | (ltm
.tm_min
^ tm
.tm_min
)
1377 | (ltm
.tm_hour
^ tm
.tm_hour
)
1378 | (ltm
.tm_mday
^ tm
.tm_mday
)
1379 | (ltm
.tm_mon
^ tm
.tm_mon
)
1380 | (ltm
.tm_year
^ tm
.tm_year
)))
1384 if (! __gmtime_r (<
, >m
))
1387 diff
= tm_diff (<m
, >m
);
1393 add (1, *p
= L_('-'));
1397 add (1, *p
= L_('+'));
1400 DO_NUMBER (4, (diff
/ 60) * 100 + diff
% 60);
1403 case L_('\0'): /* GNU extension: % at end of format. */
1407 /* Unknown format; output the format, including the '%',
1408 since this is most likely the right thing to do if a
1409 multibyte string has been misparsed. */
1413 for (flen
= 1; f
[1 - flen
] != L_('%'); flen
++)
1415 cpy (flen
, &f
[1 - flen
]);
1421 if (p
&& maxsize
!= 0)
1428 /* For Emacs we have a separate interface which corresponds to the normal
1429 strftime function and does not have the extra information whether the
1430 TP arguments comes from a `gmtime' call or not. */
1432 emacs_strftime (s
, maxsize
, format
, tp
)
1436 const struct tm
*tp
;
1438 return my_strftime (s
, maxsize
, format
, tp
, 0);
1442 #if defined _LIBC && !defined COMPILE_WIDE
1443 weak_alias (__strftime_l
, strftime_l
)