1 /* Copyright (C) 2002-2022 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 <https://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 HAVE_STRFTIME 0
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
109 # define ABALTMON_1 _NL_ABALTMON_1
111 # if !defined STDC_HEADERS && !defined HAVE_MEMCPY
112 # define MEMCPY(d, s, n) bcopy ((s), (d), (n))
114 # define MEMCPY(d, s, n) memcpy ((d), (s), (n))
116 # define STRLEN(s) strlen (s)
119 # define MEMPCPY(d, s, n) __mempcpy (d, s, n)
121 # ifndef HAVE_MEMPCPY
122 # define MEMPCPY(d, s, n) ((void *) ((char *) memcpy (d, s, n) + (n)))
139 #define TYPE_SIGNED(t) ((t) -1 < 0)
141 /* Bound on length of the string representing an integer value of type t.
142 Subtract one for the sign bit if t is signed;
143 302 / 1000 is log10 (2) rounded up;
144 add one for integer division truncation;
145 add one more for a minus sign if t is signed. */
146 #define INT_STRLEN_BOUND(t) \
147 ((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 + 1 + TYPE_SIGNED (t))
149 #define TM_YEAR_BASE 1900
152 /* Nonzero if YEAR is a leap year (every 4 years,
153 except every 100th isn't, and every 400th is). */
154 # define __isleap(year) \
155 ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
160 # define tzname __tzname
161 # define tzset __tzset
165 /* Portable standalone applications should supply a "time_r.h" that
166 declares a POSIX-compliant localtime_r, for the benefit of older
167 implementations that lack localtime_r or have a nonstandard one.
168 Similarly for gmtime_r. See the gnulib time_r module for one way
169 to implement this. */
172 # undef __localtime_r
173 # define __gmtime_r gmtime_r
174 # define __localtime_r localtime_r
178 #if !defined memset && !defined HAVE_MEMSET && !defined _LIBC
179 /* Some systems lack the `memset' function and we don't want to
180 introduce additional dependencies. */
181 /* The SGI compiler reportedly barfs on the trailing null
182 if we use a string constant as the initializer. 28 June 1997, rms. */
183 static const CHAR_T spaces
[16] = /* " " */
185 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),
186 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' ')
188 static const CHAR_T zeroes
[16] = /* "0000000000000000" */
190 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),
191 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0')
194 # define memset_space(P, Len) \
200 int _this = _len > 16 ? 16 : _len; \
201 (P) = MEMPCPY ((P), spaces, _this * sizeof (CHAR_T)); \
207 # define memset_zero(P, Len) \
213 int _this = _len > 16 ? 16 : _len; \
214 (P) = MEMPCPY ((P), zeroes, _this * sizeof (CHAR_T)); \
221 # define memset_space(P, Len) (wmemset ((P), L' ', (Len)), (P) += (Len))
222 # define memset_zero(P, Len) (wmemset ((P), L'0', (Len)), (P) += (Len))
224 # define memset_space(P, Len) (memset ((P), ' ', (Len)), (P) += (Len))
225 # define memset_zero(P, Len) (memset ((P), '0', (Len)), (P) += (Len))
233 int _delta = width - _n; \
234 int _incr = _n + (_delta > 0 ? _delta : 0); \
235 if ((size_t) _incr >= maxsize - i) \
241 if (pad == L_('0')) \
242 memset_zero (p, _delta); \
244 memset_space (p, _delta); \
255 memcpy_lowcase (p, (s), _n LOCALE_ARG); \
256 else if (to_uppcase) \
257 memcpy_uppcase (p, (s), _n LOCALE_ARG); \
259 MEMCPY ((PTR) p, (const PTR) (s), _n))
262 # ifndef USE_IN_EXTENDED_LOCALE_MODEL
263 # undef __mbsrtowcs_l
264 # define __mbsrtowcs_l(d, s, l, st, loc) __mbsrtowcs (d, s, l, st)
266 # define widen(os, ws, l) \
269 const char *__s = os; \
270 memset (&__st, '\0', sizeof (__st)); \
271 l = __mbsrtowcs_l (NULL, &__s, 0, &__st, loc); \
272 ws = alloca ((l + 1) * sizeof (wchar_t)); \
273 (void) __mbsrtowcs_l (ws, &__s, l, &__st, loc); \
278 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
279 /* We use this code also for the extended locale handling where the
280 function gets as an additional argument the locale which has to be
281 used. To access the values we have to redefine the _NL_CURRENT
283 # define strftime __strftime_l
284 # define wcsftime __wcsftime_l
286 # define _NL_CURRENT(category, item) \
287 (current->values[_NL_ITEM_INDEX (item)].string)
288 # define LOCALE_PARAM , locale_t loc
289 # define LOCALE_ARG , loc
290 # define HELPER_LOCALE_ARG , current
292 # define LOCALE_PARAM
295 # define HELPER_LOCALE_ARG , _NL_CURRENT_DATA (LC_TIME)
297 # define HELPER_LOCALE_ARG
302 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
303 # define TOUPPER(Ch, L) __towupper_l (Ch, L)
304 # define TOLOWER(Ch, L) __towlower_l (Ch, L)
306 # define TOUPPER(Ch, L) towupper (Ch)
307 # define TOLOWER(Ch, L) towlower (Ch)
311 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
312 # define TOUPPER(Ch, L) __toupper_l (Ch, L)
313 # define TOLOWER(Ch, L) __tolower_l (Ch, L)
315 # define TOUPPER(Ch, L) toupper (Ch)
316 # define TOLOWER(Ch, L) tolower (Ch)
319 # define TOUPPER(Ch, L) (islower (Ch) ? toupper (Ch) : (Ch))
320 # define TOLOWER(Ch, L) (isupper (Ch) ? tolower (Ch) : (Ch))
323 /* We don't use `isdigit' here since the locale dependent
324 interpretation is not what we want here. We only need to accept
325 the arabic digits in the ASCII range. One day there is perhaps a
326 more reliable way to accept other sets of digits. */
327 #define ISDIGIT(Ch) ((unsigned int) (Ch) - L_('0') <= 9)
329 static CHAR_T
*memcpy_lowcase (CHAR_T
*dest
, const CHAR_T
*src
,
330 size_t len LOCALE_PARAM
) __THROW
;
333 memcpy_lowcase (CHAR_T
*dest
, const CHAR_T
*src
, size_t len LOCALE_PARAM
)
336 dest
[len
] = TOLOWER ((UCHAR_T
) src
[len
], loc
);
340 static CHAR_T
*memcpy_uppcase (CHAR_T
*dest
, const CHAR_T
*src
,
341 size_t len LOCALE_PARAM
) __THROW
;
344 memcpy_uppcase (CHAR_T
*dest
, const CHAR_T
*src
, size_t len LOCALE_PARAM
)
347 dest
[len
] = TOUPPER ((UCHAR_T
) src
[len
], loc
);
353 /* Yield the difference between *A and *B,
354 measured in seconds, ignoring leap seconds. */
355 # define tm_diff ftime_tm_diff
356 static int tm_diff (const struct tm
*, const struct tm
*) __THROW
;
358 tm_diff (const struct tm
*a
, const struct tm
*b
)
360 /* Compute intervening leap days correctly even if year is negative.
361 Take care to avoid int overflow in leap day calculations,
362 but it's OK to assume that A and B are close to each other. */
363 int a4
= (a
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (a
->tm_year
& 3);
364 int b4
= (b
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (b
->tm_year
& 3);
365 int a100
= a4
/ 25 - (a4
% 25 < 0);
366 int b100
= b4
/ 25 - (b4
% 25 < 0);
367 int a400
= a100
>> 2;
368 int b400
= b100
>> 2;
369 int intervening_leap_days
= (a4
- b4
) - (a100
- b100
) + (a400
- b400
);
370 int years
= a
->tm_year
- b
->tm_year
;
371 int days
= (365 * years
+ intervening_leap_days
372 + (a
->tm_yday
- b
->tm_yday
));
373 return (60 * (60 * (24 * days
+ (a
->tm_hour
- b
->tm_hour
))
374 + (a
->tm_min
- b
->tm_min
))
375 + (a
->tm_sec
- b
->tm_sec
));
377 #endif /* ! HAVE_TM_GMTOFF */
381 /* The number of days from the first day of the first ISO week of this
382 year to the year day YDAY with week day WDAY. ISO weeks start on
383 Monday; the first ISO week has the year's first Thursday. YDAY may
384 be as small as YDAY_MINIMUM. */
385 #define ISO_WEEK_START_WDAY 1 /* Monday */
386 #define ISO_WEEK1_WDAY 4 /* Thursday */
387 #define YDAY_MINIMUM (-366)
388 static int iso_week_days (int, int) __THROW
;
393 iso_week_days (int yday
, int wday
)
395 /* Add enough to the first operand of % to make it nonnegative. */
396 int big_enough_multiple_of_7
= (-YDAY_MINIMUM
/ 7 + 2) * 7;
398 - (yday
- wday
+ ISO_WEEK1_WDAY
+ big_enough_multiple_of_7
) % 7
399 + ISO_WEEK1_WDAY
- ISO_WEEK_START_WDAY
);
403 #if !(defined _NL_CURRENT || HAVE_STRFTIME)
404 static CHAR_T
const weekday_name
[][10] =
406 L_("Sunday"), L_("Monday"), L_("Tuesday"), L_("Wednesday"),
407 L_("Thursday"), L_("Friday"), L_("Saturday")
409 static CHAR_T
const month_name
[][10] =
411 L_("January"), L_("February"), L_("March"), L_("April"), L_("May"),
412 L_("June"), L_("July"), L_("August"), L_("September"), L_("October"),
413 L_("November"), L_("December")
419 # define my_strftime emacs_strftimeu
420 # define ut_argument , ut
421 # define ut_argument_spec , int ut
424 # define my_strftime wcsftime
425 # define nl_get_alt_digit _nl_get_walt_digit
427 # define my_strftime strftime
428 # define nl_get_alt_digit _nl_get_alt_digit
431 # define ut_argument_spec
432 /* We don't have this information in general. */
436 static size_t __strftime_internal (CHAR_T
*, size_t, const CHAR_T
*,
437 const struct tm
*, int, bool *
439 LOCALE_PARAM
) __THROW
;
441 /* Write information from TP into S according to the format
442 string FORMAT, writing no more that MAXSIZE characters
443 (including the terminating '\0') and returning number of
444 characters written. If S is NULL, nothing will be written
445 anywhere, so to determine how many characters would be
446 written, use NULL for S and (size_t) UINT_MAX for MAXSIZE. */
449 my_strftime (CHAR_T
*s
, size_t maxsize
, const CHAR_T
*format
,
450 const struct tm
*tp ut_argument_spec LOCALE_PARAM
)
452 #if !defined _LIBC && HAVE_TZNAME && HAVE_TZSET
453 /* Solaris 2.5 tzset sometimes modifies the storage returned by localtime.
454 Work around this bug by copying *tp before it might be munged. */
459 bool tzset_called
= false;
460 return __strftime_internal (s
, maxsize
, format
, tp
, 0, &tzset_called
461 ut_argument LOCALE_ARG
);
464 libc_hidden_def (my_strftime
)
468 __strftime_internal (CHAR_T
*s
, size_t maxsize
, const CHAR_T
*format
,
469 const struct tm
*tp
, int yr_spec
, bool *tzset_called
470 ut_argument_spec LOCALE_PARAM
)
472 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
473 struct __locale_data
*const current
= loc
->__locales
[LC_TIME
];
476 int hour12
= tp
->tm_hour
;
478 /* We cannot make the following values variables since we must delay
479 the evaluation of these values until really needed since some
480 expressions might not be valid in every situation. The `struct tm'
481 might be generated by a strptime() call that initialized
482 only a few elements. Dereference the pointers only if the format
483 requires this. Then it is ok to fail if the pointers are invalid. */
485 ((const CHAR_T *) (tp->tm_wday < 0 || tp->tm_wday > 6 \
486 ? "?" : _NL_CURRENT (LC_TIME, NLW(ABDAY_1) + tp->tm_wday)))
488 ((const CHAR_T *) (tp->tm_wday < 0 || tp->tm_wday > 6 \
489 ? "?" : _NL_CURRENT (LC_TIME, NLW(DAY_1) + tp->tm_wday)))
491 ((const CHAR_T *) (tp->tm_mon < 0 || tp->tm_mon > 11 \
492 ? "?" : _NL_CURRENT (LC_TIME, NLW(ABMON_1) + tp->tm_mon)))
494 ((const CHAR_T *) (tp->tm_mon < 0 || tp->tm_mon > 11 \
495 ? "?" : _NL_CURRENT (LC_TIME, NLW(MON_1) + tp->tm_mon)))
496 # define a_altmonth \
497 ((const CHAR_T *) (tp->tm_mon < 0 || tp->tm_mon > 11 \
498 ? "?" : _NL_CURRENT (LC_TIME, NLW(ABALTMON_1) + tp->tm_mon)))
499 # define f_altmonth \
500 ((const CHAR_T *) (tp->tm_mon < 0 || tp->tm_mon > 11 \
501 ? "?" : _NL_CURRENT (LC_TIME, NLW(ALTMON_1) + tp->tm_mon)))
503 ((const CHAR_T *) _NL_CURRENT (LC_TIME, tp->tm_hour > 11 \
504 ? NLW(PM_STR) : NLW(AM_STR)))
506 # define aw_len STRLEN (a_wkday)
507 # define am_len STRLEN (a_month)
508 # define aam_len STRLEN (a_altmonth)
509 # define ap_len STRLEN (ampm)
512 # define f_wkday (tp->tm_wday < 0 || tp->tm_wday > 6 \
513 ? "?" : weekday_name[tp->tm_wday])
514 # define f_month (tp->tm_mon < 0 || tp->tm_mon > 11 \
515 ? "?" : month_name[tp->tm_mon])
516 # define a_wkday f_wkday
517 # define a_month f_month
518 # define a_altmonth a_month
519 # define f_altmonth f_month
520 # define ampm (L_("AMPM") + 2 * (tp->tm_hour > 11))
532 #if DO_MULTIBYTE && !defined COMPILE_WIDE
533 const char *format_end
= NULL
;
538 /* The POSIX test suite assumes that setting
539 the environment variable TZ to a new value before calling strftime()
540 will influence the result (the %Z format) even if the information in
541 TP is computed with a totally different time zone.
542 This is bogus: though POSIX allows bad behavior like this,
543 POSIX does not require it. Do the right thing instead. */
544 zone
= (const char *) tp
->tm_zone
;
549 if (! (zone
&& *zone
))
560 for (f
= format
; *f
!= '\0'; ++f
)
562 int pad
= 0; /* Padding for number ('-', '_', or 0). */
563 int modifier
; /* Field modifier ('E', 'O', or 0). */
564 int digits
; /* Max digits for numeric format. */
565 int number_value
; /* Numeric value to be printed. */
566 int negative_number
; /* 1 if the number is negative. */
567 const CHAR_T
*subfmt
;
569 CHAR_T buf
[1 + (sizeof (int) < sizeof (time_t)
570 ? INT_STRLEN_BOUND (time_t)
571 : INT_STRLEN_BOUND (int))];
578 #if DO_MULTIBYTE && !defined COMPILE_WIDE
584 case L_('\b'): case L_('\t'): case L_('\n'):
585 case L_('\v'): case L_('\f'): case L_('\r'):
586 case L_(' '): case L_('!'): case L_('"'): case L_('#'): case L_('&'):
587 case L_('\''): case L_('('): case L_(')'): case L_('*'): case L_('+'):
588 case L_(','): case L_('-'): case L_('.'): case L_('/'): case L_('0'):
589 case L_('1'): case L_('2'): case L_('3'): case L_('4'): case L_('5'):
590 case L_('6'): case L_('7'): case L_('8'): case L_('9'): case L_(':'):
591 case L_(';'): case L_('<'): case L_('='): case L_('>'): case L_('?'):
592 case L_('A'): case L_('B'): case L_('C'): case L_('D'): case L_('E'):
593 case L_('F'): case L_('G'): case L_('H'): case L_('I'): case L_('J'):
594 case L_('K'): case L_('L'): case L_('M'): case L_('N'): case L_('O'):
595 case L_('P'): case L_('Q'): case L_('R'): case L_('S'): case L_('T'):
596 case L_('U'): case L_('V'): case L_('W'): case L_('X'): case L_('Y'):
597 case L_('Z'): case L_('['): case L_('\\'): case L_(']'): case L_('^'):
598 case L_('_'): case L_('a'): case L_('b'): case L_('c'): case L_('d'):
599 case L_('e'): case L_('f'): case L_('g'): case L_('h'): case L_('i'):
600 case L_('j'): case L_('k'): case L_('l'): case L_('m'): case L_('n'):
601 case L_('o'): case L_('p'): case L_('q'): case L_('r'): case L_('s'):
602 case L_('t'): case L_('u'): case L_('v'): case L_('w'): case L_('x'):
603 case L_('y'): case L_('z'): case L_('{'): case L_('|'): case L_('}'):
605 /* The C Standard requires these 98 characters (plus '%') to
606 be in the basic execution character set. None of these
607 characters can start a multibyte sequence, so they need
608 not be analyzed further. */
613 /* Copy this multibyte sequence until we reach its end, find
614 an error, or come back to the initial shift state. */
616 mbstate_t mbstate
= mbstate_zero
;
621 format_end
= f
+ strlen (f
) + 1;
622 fsize
= format_end
- f
;
626 size_t bytes
= mbrlen (f
+ len
, fsize
- len
, &mbstate
);
631 if (bytes
== (size_t) -2)
633 len
+= strlen (f
+ len
);
637 if (bytes
== (size_t) -1)
645 while (! mbsinit (&mbstate
));
653 #else /* ! DO_MULTIBYTE */
655 /* Either multibyte encodings are not supported, they are
656 safe for formats, so any non-'%' byte can be copied through,
657 or this is the wide character version. */
664 #endif /* ! DO_MULTIBYTE */
666 /* Check for flags that can modify a format. */
671 /* This influences the number formats. */
678 /* This changes textual output. */
692 /* As a GNU extension we allow to specify the field width. */
698 if (width
> INT_MAX
/ 10
699 || (width
== INT_MAX
/ 10 && *f
- L_('0') > INT_MAX
% 10))
700 /* Avoid overflow. */
705 width
+= *f
- L_('0');
709 while (ISDIGIT (*f
));
712 /* Check for modifiers. */
725 /* Now do the specified format. */
729 #define DO_NUMBER(d, v) \
732 digits = d > width ? d : width; \
737 #define DO_NUMBER_SPACEPAD(d, v) \
740 digits = d > width ? d : width; \
742 goto do_number_spacepad; \
760 #if defined _NL_CURRENT || !HAVE_STRFTIME
761 cpy (aw_len
, a_wkday
);
764 goto underlying_strftime
;
775 #if defined _NL_CURRENT || !HAVE_STRFTIME
776 cpy (STRLEN (f_wkday
), f_wkday
);
779 goto underlying_strftime
;
789 if (modifier
== L_('E'))
791 #if defined _NL_CURRENT || !HAVE_STRFTIME
792 if (modifier
== L_('O'))
793 cpy (aam_len
, a_altmonth
);
795 cpy (am_len
, a_month
);
798 goto underlying_strftime
;
802 if (modifier
== L_('E'))
809 #if defined _NL_CURRENT || !HAVE_STRFTIME
810 if (modifier
== L_('O'))
811 cpy (STRLEN (f_altmonth
), f_altmonth
);
813 cpy (STRLEN (f_month
), f_month
);
816 goto underlying_strftime
;
820 if (modifier
== L_('O'))
823 if (! (modifier
== L_('E')
825 (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
828 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_T_FMT
));
831 goto underlying_strftime
;
833 subfmt
= L_("%a %b %e %H:%M:%S %Y");
839 CHAR_T
*old_start
= p
;
840 size_t len
= __strftime_internal (NULL
, (size_t) -1, subfmt
,
841 tp
, yr_spec
, tzset_called
842 ut_argument LOCALE_ARG
);
843 add (len
, __strftime_internal (p
, maxsize
- i
, subfmt
,
844 tp
, yr_spec
, tzset_called
845 ut_argument LOCALE_ARG
));
848 while (old_start
< p
)
850 *old_start
= TOUPPER ((UCHAR_T
) *old_start
, loc
);
856 #if HAVE_STRFTIME && ! (defined _NL_CURRENT && HAVE_STRUCT_ERA_ENTRY)
859 /* The relevant information is available only via the
860 underlying strftime implementation, so use that. */
863 char ubuf
[1024]; /* enough for any single format in practice */
865 /* Make sure we're calling the actual underlying strftime.
866 In some cases, config.h contains something like
867 "#define strftime rpl_strftime". */
878 len
= strftime (ubuf
, sizeof ubuf
, ufmt
, tp
);
879 if (len
== 0 && ubuf
[0] != '\0')
887 if (modifier
== L_('E'))
889 #if HAVE_STRUCT_ERA_ENTRY
890 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
894 size_t len
= __wcslen (era
->era_wname
);
895 cpy (len
, era
->era_wname
);
897 size_t len
= strlen (era
->era_name
);
898 cpy (len
, era
->era_name
);
904 goto underlying_strftime
;
910 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
911 DO_NUMBER (1, year
/ 100 - (year
% 100 < 0));
915 if (modifier
== L_('O'))
918 if (! (modifier
== L_('E')
920 (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(ERA_D_FMT
)))
922 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_FMT
));
926 goto underlying_strftime
;
934 subfmt
= L_("%m/%d/%y");
938 if (modifier
== L_('E'))
941 DO_NUMBER (2, tp
->tm_mday
);
944 if (modifier
== L_('E'))
947 DO_NUMBER_SPACEPAD (2, tp
->tm_mday
);
949 /* All numeric formats set DIGITS and NUMBER_VALUE and then
950 jump to one of these two labels. */
953 /* Force `_' flag unless overwritten by `0' or '-' flag. */
954 if (pad
!= L_('0') && pad
!= L_('-'))
958 /* Format the number according to the MODIFIER flag. */
960 if (modifier
== L_('O') && 0 <= number_value
)
963 /* Get the locale specific alternate representation of
964 the number NUMBER_VALUE. If none exist NULL is returned. */
965 const CHAR_T
*cp
= nl_get_alt_digit (number_value
970 size_t digitlen
= STRLEN (cp
);
979 goto underlying_strftime
;
984 unsigned int u
= number_value
;
986 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
987 negative_number
= number_value
< 0;
993 *--bufp
= u
% 10 + L_('0');
994 while ((u
/= 10) != 0);
997 do_number_sign_and_padding
:
1003 int padding
= digits
- (buf
+ (sizeof (buf
) / sizeof (buf
[0]))
1010 if ((size_t) padding
>= maxsize
- i
)
1014 memset_space (p
, padding
);
1016 width
= width
> padding
? width
- padding
: 0;
1020 if ((size_t) digits
>= maxsize
- i
)
1023 if (negative_number
)
1033 memset_zero (p
, padding
);
1040 cpy (buf
+ sizeof (buf
) / sizeof (buf
[0]) - bufp
, bufp
);
1046 subfmt
= L_("%Y-%m-%d");
1050 if (modifier
== L_('E'))
1053 DO_NUMBER (2, tp
->tm_hour
);
1056 if (modifier
== L_('E'))
1059 DO_NUMBER (2, hour12
);
1061 case L_('k'): /* GNU extension. */
1062 if (modifier
== L_('E'))
1065 DO_NUMBER_SPACEPAD (2, tp
->tm_hour
);
1067 case L_('l'): /* GNU extension. */
1068 if (modifier
== L_('E'))
1071 DO_NUMBER_SPACEPAD (2, hour12
);
1074 if (modifier
== L_('E'))
1077 DO_NUMBER (3, 1 + tp
->tm_yday
);
1080 if (modifier
== L_('E'))
1083 DO_NUMBER (2, tp
->tm_min
);
1086 if (modifier
== L_('E'))
1089 DO_NUMBER (2, tp
->tm_mon
+ 1);
1092 add (1, *p
= L_('\n'));
1097 #if !defined _NL_CURRENT && HAVE_STRFTIME
1098 format_char
= L_('p');
1108 #if defined _NL_CURRENT || !HAVE_STRFTIME
1112 goto underlying_strftime
;
1116 subfmt
= L_("%H:%M");
1120 #if !defined _NL_CURRENT && HAVE_STRFTIME
1121 goto underlying_strftime
;
1124 if (*(subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
1128 subfmt
= L_("%I:%M:%S %p");
1133 if (modifier
== L_('E'))
1136 DO_NUMBER (2, tp
->tm_sec
);
1138 case L_('s'): /* GNU extension. */
1146 /* Generate string value for T using time_t arithmetic;
1147 this works even if sizeof (long) < sizeof (time_t). */
1149 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
1150 negative_number
= t
< 0;
1157 if (negative_number
)
1161 /* Adjust if division truncates to minus infinity. */
1162 if (0 < -1 % 10 && d
< 0)
1169 *--bufp
= d
+ L_('0');
1174 goto do_number_sign_and_padding
;
1178 if (modifier
== L_('O'))
1181 if (! (modifier
== L_('E')
1183 (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(ERA_T_FMT
)))
1185 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(T_FMT
));
1189 goto underlying_strftime
;
1195 subfmt
= L_("%H:%M:%S");
1199 add (1, *p
= L_('\t'));
1203 DO_NUMBER (1, (tp
->tm_wday
- 1 + 7) % 7 + 1);
1206 if (modifier
== L_('E'))
1209 DO_NUMBER (2, (tp
->tm_yday
- tp
->tm_wday
+ 7) / 7);
1214 if (modifier
== L_('E'))
1217 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
1218 int days
= iso_week_days (tp
->tm_yday
, tp
->tm_wday
);
1222 /* This ISO week belongs to the previous year. */
1224 days
= iso_week_days (tp
->tm_yday
+ (365 + __isleap (year
)),
1229 int d
= iso_week_days (tp
->tm_yday
- (365 + __isleap (year
)),
1233 /* This ISO week belongs to the next year. */
1242 DO_NUMBER (2, (year
% 100 + 100) % 100);
1245 DO_NUMBER (1, year
);
1248 DO_NUMBER (2, days
/ 7 + 1);
1253 if (modifier
== L_('E'))
1256 DO_NUMBER (2, (tp
->tm_yday
- (tp
->tm_wday
- 1 + 7) % 7 + 7) / 7);
1259 if (modifier
== L_('E'))
1262 DO_NUMBER (1, tp
->tm_wday
);
1265 if (modifier
== L_('E'))
1267 #if HAVE_STRUCT_ERA_ENTRY
1268 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1271 # ifdef COMPILE_WIDE
1272 subfmt
= era
->era_wformat
;
1274 subfmt
= era
->era_format
;
1282 goto underlying_strftime
;
1286 if (modifier
== L_('O'))
1289 DO_NUMBER (1, tp
->tm_year
+ TM_YEAR_BASE
);
1292 if (modifier
== L_('E'))
1294 #if HAVE_STRUCT_ERA_ENTRY
1295 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1298 int delta
= tp
->tm_year
- era
->start_date
[0];
1301 DO_NUMBER (2, (era
->offset
1302 + delta
* era
->absolute_direction
));
1306 goto underlying_strftime
;
1310 DO_NUMBER (2, (tp
->tm_year
% 100 + 100) % 100);
1320 /* The tzset() call might have changed the value. */
1321 if (!(zone
&& *zone
) && tp
->tm_isdst
>= 0)
1323 /* POSIX.1 requires that local time zone information is used as
1324 though strftime called tzset. */
1329 *tzset_called
= true;
1332 zone
= tp
->tm_isdst
<= 1 ? tzname
[tp
->tm_isdst
] : "?";
1340 /* The zone string is always given in multibyte form. We have
1341 to transform it first. */
1344 widen (zone
, wczone
, len
);
1348 cpy (strlen (zone
), zone
);
1353 if (tp
->tm_isdst
< 0)
1359 diff
= tp
->tm_gmtoff
;
1369 /* POSIX.1 requires that local time zone information is used as
1370 though strftime called tzset. */
1375 *tzset_called
= true;
1382 if (lt
== (time_t) -1)
1384 /* mktime returns -1 for errors, but -1 is also a
1385 valid time_t value. Check whether an error really
1389 if (! __localtime_r (<
, &tm
)
1390 || ((ltm
.tm_sec
^ tm
.tm_sec
)
1391 | (ltm
.tm_min
^ tm
.tm_min
)
1392 | (ltm
.tm_hour
^ tm
.tm_hour
)
1393 | (ltm
.tm_mday
^ tm
.tm_mday
)
1394 | (ltm
.tm_mon
^ tm
.tm_mon
)
1395 | (ltm
.tm_year
^ tm
.tm_year
)))
1399 if (! __gmtime_r (<
, >m
))
1402 diff
= tm_diff (<m
, >m
);
1408 add (1, *p
= L_('-'));
1412 add (1, *p
= L_('+'));
1415 DO_NUMBER (4, (diff
/ 60) * 100 + diff
% 60);
1418 case L_('\0'): /* GNU extension: % at end of format. */
1422 /* Unknown format; output the format, including the '%',
1423 since this is most likely the right thing to do if a
1424 multibyte string has been misparsed. */
1428 for (flen
= 1; f
[1 - flen
] != L_('%'); flen
++)
1430 cpy (flen
, &f
[1 - flen
]);
1436 if (p
&& maxsize
!= 0)
1443 /* For Emacs we have a separate interface which corresponds to the normal
1444 strftime function and does not have the extra information whether the
1445 TP arguments comes from a `gmtime' call or not. */
1447 emacs_strftime (char *s
, size_t maxsize
, const char *format
,
1448 const struct tm
*tp
)
1450 return my_strftime (s
, maxsize
, format
, tp
, 0);
1454 #if defined _LIBC && !defined COMPILE_WIDE
1455 weak_alias (__strftime_l
, strftime_l
)