1 /* Copyright (C) 2002, 2004 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
;
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)))
126 # if defined __GNUC__ || (defined __STDC__ && __STDC__)
127 # define __P(args) args
129 # define __P(args) ()
131 #endif /* Not __P. */
149 #define TYPE_SIGNED(t) ((t) -1 < 0)
151 /* Bound on length of the string representing an integer value of type t.
152 Subtract one for the sign bit if t is signed;
153 302 / 1000 is log10 (2) rounded up;
154 add one for integer division truncation;
155 add one more for a minus sign if t is signed. */
156 #define INT_STRLEN_BOUND(t) \
157 ((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 + 1 + TYPE_SIGNED (t))
159 #define TM_YEAR_BASE 1900
162 /* Nonzero if YEAR is a leap year (every 4 years,
163 except every 100th isn't, and every 400th is). */
164 # define __isleap(year) \
165 ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
170 # define tzname __tzname
171 # define tzset __tzset
175 /* Portable standalone applications should supply a "time_r.h" that
176 declares a POSIX-compliant localtime_r, for the benefit of older
177 implementations that lack localtime_r or have a nonstandard one.
178 Similarly for gmtime_r. See the gnulib time_r module for one way
179 to implement this. */
182 # undef __localtime_r
183 # define __gmtime_r gmtime_r
184 # define __localtime_r localtime_r
188 #if !defined memset && !defined HAVE_MEMSET && !defined _LIBC
189 /* Some systems lack the `memset' function and we don't want to
190 introduce additional dependencies. */
191 /* The SGI compiler reportedly barfs on the trailing null
192 if we use a string constant as the initializer. 28 June 1997, rms. */
193 static const CHAR_T spaces
[16] = /* " " */
195 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),
196 L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' '),L_(' ')
198 static const CHAR_T zeroes
[16] = /* "0000000000000000" */
200 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),
201 L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0'),L_('0')
204 # define memset_space(P, Len) \
210 int _this = _len > 16 ? 16 : _len; \
211 (P) = MEMPCPY ((P), spaces, _this * sizeof (CHAR_T)); \
217 # define memset_zero(P, Len) \
223 int _this = _len > 16 ? 16 : _len; \
224 (P) = MEMPCPY ((P), zeroes, _this * sizeof (CHAR_T)); \
231 # define memset_space(P, Len) (wmemset ((P), L' ', (Len)), (P) += (Len))
232 # define memset_zero(P, Len) (wmemset ((P), L'0', (Len)), (P) += (Len))
234 # define memset_space(P, Len) (memset ((P), ' ', (Len)), (P) += (Len))
235 # define memset_zero(P, Len) (memset ((P), '0', (Len)), (P) += (Len))
243 int _delta = width - _n; \
244 int _incr = _n + (_delta > 0 ? _delta : 0); \
245 if ((size_t) _incr >= maxsize - i) \
251 if (pad == L_('0')) \
252 memset_zero (p, _delta); \
254 memset_space (p, _delta); \
265 memcpy_lowcase (p, (s), _n LOCALE_ARG); \
266 else if (to_uppcase) \
267 memcpy_uppcase (p, (s), _n LOCALE_ARG); \
269 MEMCPY ((PTR) p, (const PTR) (s), _n))
272 # ifndef USE_IN_EXTENDED_LOCALE_MODEL
273 # undef __mbsrtowcs_l
274 # define __mbsrtowcs_l(d, s, l, st, loc) __mbsrtowcs (d, s, l, st)
276 # define widen(os, ws, l) \
279 const char *__s = os; \
280 memset (&__st, '\0', sizeof (__st)); \
281 l = __mbsrtowcs_l (NULL, &__s, 0, &__st, loc); \
282 ws = alloca ((l + 1) * sizeof (wchar_t)); \
283 (void) __mbsrtowcs_l (ws, &__s, l, &__st, loc); \
288 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
289 /* We use this code also for the extended locale handling where the
290 function gets as an additional argument the locale which has to be
291 used. To access the values we have to redefine the _NL_CURRENT
293 # define strftime __strftime_l
294 # define wcsftime __wcsftime_l
296 # define _NL_CURRENT(category, item) \
297 (current->values[_NL_ITEM_INDEX (item)].string)
298 # define LOCALE_PARAM , loc
299 # define LOCALE_ARG , loc
300 # define LOCALE_PARAM_DECL __locale_t loc;
301 # define LOCALE_PARAM_PROTO , __locale_t loc
302 # define HELPER_LOCALE_ARG , current
304 # define LOCALE_PARAM
305 # define LOCALE_PARAM_PROTO
307 # define LOCALE_PARAM_DECL
309 # define HELPER_LOCALE_ARG , _NL_CURRENT_DATA (LC_TIME)
311 # define HELPER_LOCALE_ARG
316 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
317 # define TOUPPER(Ch, L) __towupper_l (Ch, L)
318 # define TOLOWER(Ch, L) __towlower_l (Ch, L)
320 # define TOUPPER(Ch, L) towupper (Ch)
321 # define TOLOWER(Ch, L) towlower (Ch)
325 # ifdef USE_IN_EXTENDED_LOCALE_MODEL
326 # define TOUPPER(Ch, L) __toupper_l (Ch, L)
327 # define TOLOWER(Ch, L) __tolower_l (Ch, L)
329 # define TOUPPER(Ch, L) toupper (Ch)
330 # define TOLOWER(Ch, L) tolower (Ch)
333 # define TOUPPER(Ch, L) (islower (Ch) ? toupper (Ch) : (Ch))
334 # define TOLOWER(Ch, L) (isupper (Ch) ? tolower (Ch) : (Ch))
337 /* We don't use `isdigit' here since the locale dependent
338 interpretation is not what we want here. We only need to accept
339 the arabic digits in the ASCII range. One day there is perhaps a
340 more reliable way to accept other sets of digits. */
341 #define ISDIGIT(Ch) ((unsigned int) (Ch) - L_('0') <= 9)
343 static CHAR_T
*memcpy_lowcase
__P ((CHAR_T
*dest
, const CHAR_T
*src
,
344 size_t len LOCALE_PARAM_PROTO
));
347 memcpy_lowcase (dest
, src
, len LOCALE_PARAM
)
354 dest
[len
] = TOLOWER ((UCHAR_T
) src
[len
], loc
);
358 static CHAR_T
*memcpy_uppcase
__P ((CHAR_T
*dest
, const CHAR_T
*src
,
359 size_t len LOCALE_PARAM_PROTO
));
362 memcpy_uppcase (dest
, src
, len LOCALE_PARAM
)
369 dest
[len
] = TOUPPER ((UCHAR_T
) src
[len
], loc
);
375 /* Yield the difference between *A and *B,
376 measured in seconds, ignoring leap seconds. */
377 # define tm_diff ftime_tm_diff
378 static int tm_diff
__P ((const struct tm
*, const struct tm
*));
384 /* Compute intervening leap days correctly even if year is negative.
385 Take care to avoid int overflow in leap day calculations,
386 but it's OK to assume that A and B are close to each other. */
387 int a4
= (a
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (a
->tm_year
& 3);
388 int b4
= (b
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (b
->tm_year
& 3);
389 int a100
= a4
/ 25 - (a4
% 25 < 0);
390 int b100
= b4
/ 25 - (b4
% 25 < 0);
391 int a400
= a100
>> 2;
392 int b400
= b100
>> 2;
393 int intervening_leap_days
= (a4
- b4
) - (a100
- b100
) + (a400
- b400
);
394 int years
= a
->tm_year
- b
->tm_year
;
395 int days
= (365 * years
+ intervening_leap_days
396 + (a
->tm_yday
- b
->tm_yday
));
397 return (60 * (60 * (24 * days
+ (a
->tm_hour
- b
->tm_hour
))
398 + (a
->tm_min
- b
->tm_min
))
399 + (a
->tm_sec
- b
->tm_sec
));
401 #endif /* ! HAVE_TM_GMTOFF */
405 /* The number of days from the first day of the first ISO week of this
406 year to the year day YDAY with week day WDAY. ISO weeks start on
407 Monday; the first ISO week has the year's first Thursday. YDAY may
408 be as small as YDAY_MINIMUM. */
409 #define ISO_WEEK_START_WDAY 1 /* Monday */
410 #define ISO_WEEK1_WDAY 4 /* Thursday */
411 #define YDAY_MINIMUM (-366)
412 static int iso_week_days
__P ((int, int));
417 iso_week_days (yday
, wday
)
421 /* Add enough to the first operand of % to make it nonnegative. */
422 int big_enough_multiple_of_7
= (-YDAY_MINIMUM
/ 7 + 2) * 7;
424 - (yday
- wday
+ ISO_WEEK1_WDAY
+ big_enough_multiple_of_7
) % 7
425 + ISO_WEEK1_WDAY
- ISO_WEEK_START_WDAY
);
429 #if !(defined _NL_CURRENT || HAVE_STRFTIME)
430 static CHAR_T
const weekday_name
[][10] =
432 L_("Sunday"), L_("Monday"), L_("Tuesday"), L_("Wednesday"),
433 L_("Thursday"), L_("Friday"), L_("Saturday")
435 static CHAR_T
const month_name
[][10] =
437 L_("January"), L_("February"), L_("March"), L_("April"), L_("May"),
438 L_("June"), L_("July"), L_("August"), L_("September"), L_("October"),
439 L_("November"), L_("December")
445 # define my_strftime emacs_strftimeu
446 # define ut_argument , ut
447 # define ut_argument_spec int ut;
448 # define ut_argument_spec_iso , int ut
451 # define my_strftime wcsftime
452 # define nl_get_alt_digit _nl_get_walt_digit
454 # define my_strftime strftime
455 # define nl_get_alt_digit _nl_get_alt_digit
458 # define ut_argument_spec
459 # define ut_argument_spec_iso
460 /* We don't have this information in general. */
464 #if !defined _LIBC && HAVE_TZNAME && HAVE_TZSET
465 /* Solaris 2.5 tzset sometimes modifies the storage returned by localtime.
466 Work around this bug by copying *tp before it might be munged. */
467 size_t _strftime_copytm
__P ((char *, size_t, const char *,
468 const struct tm
* ut_argument_spec_iso
));
470 my_strftime (s
, maxsize
, format
, tp ut_argument
)
473 const CHAR_T
*format
;
479 return _strftime_copytm (s
, maxsize
, format
, &tmcopy ut_argument
);
482 # define my_strftime _strftime_copytm
486 /* Write information from TP into S according to the format
487 string FORMAT, writing no more that MAXSIZE characters
488 (including the terminating '\0') and returning number of
489 characters written. If S is NULL, nothing will be written
490 anywhere, so to determine how many characters would be
491 written, use NULL for S and (size_t) UINT_MAX for MAXSIZE. */
493 my_strftime (s
, maxsize
, format
, tp ut_argument LOCALE_PARAM
)
496 const CHAR_T
*format
;
501 #if defined _LIBC && defined USE_IN_EXTENDED_LOCALE_MODEL
502 struct locale_data
*const current
= loc
->__locales
[LC_TIME
];
505 int hour12
= tp
->tm_hour
;
507 /* We cannot make the following values variables since we must delay
508 the evaluation of these values until really needed since some
509 expressions might not be valid in every situation. The `struct tm'
510 might be generated by a strptime() call that initialized
511 only a few elements. Dereference the pointers only if the format
512 requires this. Then it is ok to fail if the pointers are invalid. */
514 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABDAY_1) + tp->tm_wday))
516 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(DAY_1) + tp->tm_wday))
518 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(ABMON_1) + tp->tm_mon))
520 ((const CHAR_T *) _NL_CURRENT (LC_TIME, NLW(MON_1) + tp->tm_mon))
522 ((const CHAR_T *) _NL_CURRENT (LC_TIME, tp->tm_hour > 11 \
523 ? NLW(PM_STR) : NLW(AM_STR)))
525 # define aw_len STRLEN (a_wkday)
526 # define am_len STRLEN (a_month)
527 # define ap_len STRLEN (ampm)
530 # define f_wkday (weekday_name[tp->tm_wday])
531 # define f_month (month_name[tp->tm_mon])
532 # define a_wkday f_wkday
533 # define a_month f_month
534 # define ampm (L_("AMPM") + 2 * (tp->tm_hour > 11))
545 #if DO_MULTIBYTE && !defined COMPILE_WIDE
546 const char *format_end
= NULL
;
551 /* The POSIX test suite assumes that setting
552 the environment variable TZ to a new value before calling strftime()
553 will influence the result (the %Z format) even if the information in
554 TP is computed with a totally different time zone.
555 This is bogus: though POSIX allows bad behavior like this,
556 POSIX does not require it. Do the right thing instead. */
557 zone
= (const char *) tp
->tm_zone
;
562 if (! (zone
&& *zone
))
567 /* POSIX.1 requires that local time zone information is used as
568 though strftime called tzset. */
581 for (f
= format
; *f
!= '\0'; ++f
)
583 int pad
= 0; /* Padding for number ('-', '_', or 0). */
584 int modifier
; /* Field modifier ('E', 'O', or 0). */
585 int digits
; /* Max digits for numeric format. */
586 int number_value
; /* Numeric value to be printed. */
587 int negative_number
; /* 1 if the number is negative. */
588 const CHAR_T
*subfmt
;
590 CHAR_T buf
[1 + (sizeof (int) < sizeof (time_t)
591 ? INT_STRLEN_BOUND (time_t)
592 : INT_STRLEN_BOUND (int))];
599 #if DO_MULTIBYTE && !defined COMPILE_WIDE
605 case L_('\b'): case L_('\t'): case L_('\n'):
606 case L_('\v'): case L_('\f'): case L_('\r'):
607 case L_(' '): case L_('!'): case L_('"'): case L_('#'): case L_('&'):
608 case L_('\''): case L_('('): case L_(')'): case L_('*'): case L_('+'):
609 case L_(','): case L_('-'): case L_('.'): case L_('/'): case L_('0'):
610 case L_('1'): case L_('2'): case L_('3'): case L_('4'): case L_('5'):
611 case L_('6'): case L_('7'): case L_('8'): case L_('9'): case L_(':'):
612 case L_(';'): case L_('<'): case L_('='): case L_('>'): case L_('?'):
613 case L_('A'): case L_('B'): case L_('C'): case L_('D'): case L_('E'):
614 case L_('F'): case L_('G'): case L_('H'): case L_('I'): case L_('J'):
615 case L_('K'): case L_('L'): case L_('M'): case L_('N'): case L_('O'):
616 case L_('P'): case L_('Q'): case L_('R'): case L_('S'): case L_('T'):
617 case L_('U'): case L_('V'): case L_('W'): case L_('X'): case L_('Y'):
618 case L_('Z'): case L_('['): case L_('\\'): case L_(']'): case L_('^'):
619 case L_('_'): case L_('a'): case L_('b'): case L_('c'): case L_('d'):
620 case L_('e'): case L_('f'): case L_('g'): case L_('h'): case L_('i'):
621 case L_('j'): case L_('k'): case L_('l'): case L_('m'): case L_('n'):
622 case L_('o'): case L_('p'): case L_('q'): case L_('r'): case L_('s'):
623 case L_('t'): case L_('u'): case L_('v'): case L_('w'): case L_('x'):
624 case L_('y'): case L_('z'): case L_('{'): case L_('|'): case L_('}'):
626 /* The C Standard requires these 98 characters (plus '%') to
627 be in the basic execution character set. None of these
628 characters can start a multibyte sequence, so they need
629 not be analyzed further. */
634 /* Copy this multibyte sequence until we reach its end, find
635 an error, or come back to the initial shift state. */
637 mbstate_t mbstate
= mbstate_zero
;
642 format_end
= f
+ strlen (f
) + 1;
643 fsize
= format_end
- f
;
647 size_t bytes
= mbrlen (f
+ len
, fsize
- len
, &mbstate
);
652 if (bytes
== (size_t) -2)
654 len
+= strlen (f
+ len
);
658 if (bytes
== (size_t) -1)
666 while (! mbsinit (&mbstate
));
674 #else /* ! DO_MULTIBYTE */
676 /* Either multibyte encodings are not supported, they are
677 safe for formats, so any non-'%' byte can be copied through,
678 or this is the wide character version. */
685 #endif /* ! DO_MULTIBYTE */
687 /* Check for flags that can modify a format. */
692 /* This influences the number formats. */
699 /* This changes textual output. */
713 /* As a GNU extension we allow to specify the field width. */
719 if (width
> INT_MAX
/ 10
720 || (width
== INT_MAX
/ 10 && *f
- L_('0') > INT_MAX
% 10))
721 /* Avoid overflow. */
726 width
+= *f
- L_('0');
730 while (ISDIGIT (*f
));
733 /* Check for modifiers. */
746 /* Now do the specified format. */
750 #define DO_NUMBER(d, v) \
751 digits = d > width ? d : width; \
752 number_value = v; goto do_number
753 #define DO_NUMBER_SPACEPAD(d, v) \
754 digits = d > width ? d : width; \
755 number_value = v; goto do_number_spacepad
771 #if defined _NL_CURRENT || !HAVE_STRFTIME
772 cpy (aw_len
, a_wkday
);
775 goto underlying_strftime
;
786 #if defined _NL_CURRENT || !HAVE_STRFTIME
787 cpy (STRLEN (f_wkday
), f_wkday
);
790 goto underlying_strftime
;
802 #if defined _NL_CURRENT || !HAVE_STRFTIME
803 cpy (am_len
, a_month
);
806 goto underlying_strftime
;
817 #if defined _NL_CURRENT || !HAVE_STRFTIME
818 cpy (STRLEN (f_month
), f_month
);
821 goto underlying_strftime
;
825 if (modifier
== L_('O'))
828 if (! (modifier
== 'E'
830 (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
833 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_T_FMT
));
836 goto underlying_strftime
;
838 subfmt
= L_("%a %b %e %H:%M:%S %Y");
844 CHAR_T
*old_start
= p
;
845 size_t len
= my_strftime (NULL
, (size_t) -1, subfmt
,
846 tp ut_argument LOCALE_ARG
);
847 add (len
, my_strftime (p
, maxsize
- i
, subfmt
,
848 tp ut_argument LOCALE_ARG
));
851 while (old_start
< p
)
853 *old_start
= TOUPPER ((UCHAR_T
) *old_start
, loc
);
859 #if HAVE_STRFTIME && ! (defined _NL_CURRENT && HAVE_STRUCT_ERA_ENTRY)
862 /* The relevant information is available only via the
863 underlying strftime implementation, so use that. */
866 char ubuf
[1024]; /* enough for any single format in practice */
868 /* Make sure we're calling the actual underlying strftime.
869 In some cases, config.h contains something like
870 "#define strftime rpl_strftime". */
881 len
= strftime (ubuf
, sizeof ubuf
, ufmt
, tp
);
882 if (len
== 0 && ubuf
[0] != '\0')
890 if (modifier
== L_('O'))
892 if (modifier
== L_('E'))
894 #if HAVE_STRUCT_ERA_ENTRY
895 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
899 size_t len
= __wcslen (era
->era_wname
);
900 cpy (len
, era
->era_wname
);
902 size_t len
= strlen (era
->era_name
);
903 cpy (len
, era
->era_name
);
909 goto underlying_strftime
;
915 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
916 DO_NUMBER (1, year
/ 100 - (year
% 100 < 0));
920 if (modifier
== L_('O'))
923 if (! (modifier
== L_('E')
925 (const CHAR_T
*)_NL_CURRENT (LC_TIME
, NLW(ERA_D_FMT
)))
927 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(D_FMT
));
931 goto underlying_strftime
;
939 subfmt
= L_("%m/%d/%y");
943 if (modifier
== L_('E'))
946 DO_NUMBER (2, tp
->tm_mday
);
949 if (modifier
== L_('E'))
952 DO_NUMBER_SPACEPAD (2, tp
->tm_mday
);
954 /* All numeric formats set DIGITS and NUMBER_VALUE and then
955 jump to one of these two labels. */
958 /* Force `_' flag unless overwritten by `0' or '-' flag. */
959 if (pad
!= L_('0') && pad
!= L_('-'))
963 /* Format the number according to the MODIFIER flag. */
965 if (modifier
== L_('O') && 0 <= number_value
)
968 /* Get the locale specific alternate representation of
969 the number NUMBER_VALUE. If none exist NULL is returned. */
970 const CHAR_T
*cp
= nl_get_alt_digit (number_value
975 size_t digitlen
= STRLEN (cp
);
984 goto underlying_strftime
;
989 unsigned int u
= number_value
;
991 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
992 negative_number
= number_value
< 0;
998 *--bufp
= u
% 10 + L_('0');
999 while ((u
/= 10) != 0);
1002 do_number_sign_and_padding
:
1003 if (negative_number
)
1008 int padding
= digits
- (buf
+ (sizeof (buf
) / sizeof (buf
[0]))
1015 if ((size_t) padding
>= maxsize
- i
)
1019 memset_space (p
, padding
);
1021 width
= width
> padding
? width
- padding
: 0;
1025 if ((size_t) digits
>= maxsize
- i
)
1028 if (negative_number
)
1038 memset_zero (p
, padding
);
1045 cpy (buf
+ sizeof (buf
) / sizeof (buf
[0]) - bufp
, bufp
);
1051 subfmt
= L_("%Y-%m-%d");
1055 if (modifier
== L_('E'))
1058 DO_NUMBER (2, tp
->tm_hour
);
1061 if (modifier
== L_('E'))
1064 DO_NUMBER (2, hour12
);
1066 case L_('k'): /* GNU extension. */
1067 if (modifier
== L_('E'))
1070 DO_NUMBER_SPACEPAD (2, tp
->tm_hour
);
1072 case L_('l'): /* GNU extension. */
1073 if (modifier
== L_('E'))
1076 DO_NUMBER_SPACEPAD (2, hour12
);
1079 if (modifier
== L_('E'))
1082 DO_NUMBER (3, 1 + tp
->tm_yday
);
1085 if (modifier
== L_('E'))
1088 DO_NUMBER (2, tp
->tm_min
);
1091 if (modifier
== L_('E'))
1094 DO_NUMBER (2, tp
->tm_mon
+ 1);
1097 add (1, *p
= L_('\n'));
1102 #if !defined _NL_CURRENT && HAVE_STRFTIME
1103 format_char
= L_('p');
1113 #if defined _NL_CURRENT || !HAVE_STRFTIME
1117 goto underlying_strftime
;
1121 subfmt
= L_("%H:%M");
1125 #if !defined _NL_CURRENT && HAVE_STRFTIME
1126 goto underlying_strftime
;
1129 if (*(subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
,
1133 subfmt
= L_("%I:%M:%S %p");
1138 if (modifier
== L_('E'))
1141 DO_NUMBER (2, tp
->tm_sec
);
1143 case L_('s'): /* GNU extension. */
1151 /* Generate string value for T using time_t arithmetic;
1152 this works even if sizeof (long) < sizeof (time_t). */
1154 bufp
= buf
+ sizeof (buf
) / sizeof (buf
[0]);
1155 negative_number
= t
< 0;
1162 if (negative_number
)
1166 /* Adjust if division truncates to minus infinity. */
1167 if (0 < -1 % 10 && d
< 0)
1174 *--bufp
= d
+ L_('0');
1179 goto do_number_sign_and_padding
;
1183 if (modifier
== L_('O'))
1186 if (! (modifier
== L_('E')
1188 (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(ERA_T_FMT
)))
1190 subfmt
= (const CHAR_T
*) _NL_CURRENT (LC_TIME
, NLW(T_FMT
));
1194 goto underlying_strftime
;
1200 subfmt
= L_("%H:%M:%S");
1204 add (1, *p
= L_('\t'));
1208 DO_NUMBER (1, (tp
->tm_wday
- 1 + 7) % 7 + 1);
1211 if (modifier
== L_('E'))
1214 DO_NUMBER (2, (tp
->tm_yday
- tp
->tm_wday
+ 7) / 7);
1219 if (modifier
== L_('E'))
1222 int year
= tp
->tm_year
+ TM_YEAR_BASE
;
1223 int days
= iso_week_days (tp
->tm_yday
, tp
->tm_wday
);
1227 /* This ISO week belongs to the previous year. */
1229 days
= iso_week_days (tp
->tm_yday
+ (365 + __isleap (year
)),
1234 int d
= iso_week_days (tp
->tm_yday
- (365 + __isleap (year
)),
1238 /* This ISO week belongs to the next year. */
1247 DO_NUMBER (2, (year
% 100 + 100) % 100);
1250 DO_NUMBER (1, year
);
1253 DO_NUMBER (2, days
/ 7 + 1);
1258 if (modifier
== L_('E'))
1261 DO_NUMBER (2, (tp
->tm_yday
- (tp
->tm_wday
- 1 + 7) % 7 + 7) / 7);
1264 if (modifier
== L_('E'))
1267 DO_NUMBER (1, tp
->tm_wday
);
1270 if (modifier
== 'E')
1272 #if HAVE_STRUCT_ERA_ENTRY
1273 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1276 # ifdef COMPILE_WIDE
1277 subfmt
= era
->era_wformat
;
1279 subfmt
= era
->era_format
;
1285 goto underlying_strftime
;
1289 if (modifier
== L_('O'))
1292 DO_NUMBER (1, tp
->tm_year
+ TM_YEAR_BASE
);
1295 if (modifier
== L_('E'))
1297 #if HAVE_STRUCT_ERA_ENTRY
1298 struct era_entry
*era
= _nl_get_era_entry (tp HELPER_LOCALE_ARG
);
1301 int delta
= tp
->tm_year
- era
->start_date
[0];
1302 DO_NUMBER (1, (era
->offset
1303 + delta
* era
->absolute_direction
));
1307 goto underlying_strftime
;
1311 DO_NUMBER (2, (tp
->tm_year
% 100 + 100) % 100);
1321 /* The tzset() call might have changed the value. */
1322 if (!(zone
&& *zone
) && tp
->tm_isdst
>= 0)
1323 zone
= tzname
[tp
->tm_isdst
];
1330 /* The zone string is always given in multibyte form. We have
1331 to transform it first. */
1334 widen (zone
, wczone
, len
);
1338 cpy (strlen (zone
), zone
);
1343 if (tp
->tm_isdst
< 0)
1349 diff
= tp
->tm_gmtoff
;
1362 if (lt
== (time_t) -1)
1364 /* mktime returns -1 for errors, but -1 is also a
1365 valid time_t value. Check whether an error really
1369 if (! __localtime_r (<
, &tm
)
1370 || ((ltm
.tm_sec
^ tm
.tm_sec
)
1371 | (ltm
.tm_min
^ tm
.tm_min
)
1372 | (ltm
.tm_hour
^ tm
.tm_hour
)
1373 | (ltm
.tm_mday
^ tm
.tm_mday
)
1374 | (ltm
.tm_mon
^ tm
.tm_mon
)
1375 | (ltm
.tm_year
^ tm
.tm_year
)))
1379 if (! __gmtime_r (<
, >m
))
1382 diff
= tm_diff (<m
, >m
);
1388 add (1, *p
= L_('-'));
1392 add (1, *p
= L_('+'));
1395 DO_NUMBER (4, (diff
/ 60) * 100 + diff
% 60);
1398 case L_('\0'): /* GNU extension: % at end of format. */
1402 /* Unknown format; output the format, including the '%',
1403 since this is most likely the right thing to do if a
1404 multibyte string has been misparsed. */
1408 for (flen
= 1; f
[1 - flen
] != L_('%'); flen
++)
1410 cpy (flen
, &f
[1 - flen
]);
1416 if (p
&& maxsize
!= 0)
1421 libc_hidden_def (my_strftime
)
1426 /* For Emacs we have a separate interface which corresponds to the normal
1427 strftime function and does not have the extra information whether the
1428 TP arguments comes from a `gmtime' call or not. */
1430 emacs_strftime (s
, maxsize
, format
, tp
)
1434 const struct tm
*tp
;
1436 return my_strftime (s
, maxsize
, format
, tp
, 0);
1440 #if defined _LIBC && !defined COMPILE_WIDE
1441 weak_alias (__strftime_l
, strftime_l
)