[svn] Synched strptime with glibc 2.1.3.

This commit is contained in:
abbotti 2002-05-09 03:14:15 -07:00
parent fd42ae1311
commit 55f59d0022
2 changed files with 294 additions and 76 deletions

View File

@ -1,3 +1,13 @@
2002-05-09 Ian Abbott <abbotti@mev.co.uk>
* cmpt.c (strptime_internal): Synched with glibc-2.1.3.
(get_number): Ditto.
(get_alt_number): Ditto.
(__isleap): New function-like macro used by strptime.
(day_of_the_week): New function used by strptime.
(day_of_the_year): Ditto.
(__mon_yday): Now shared by mktime and strptime implementations.
2002-05-08 Hrvoje Niksic <hniksic@arsdigita.com> 2002-05-08 Hrvoje Niksic <hniksic@arsdigita.com>
* cookies.c (check_domain_match): Use match_tail in case * cookies.c (check_domain_match): Use match_tail in case

View File

@ -279,13 +279,12 @@ strpbrk (const char *s, const char *accept)
#endif #endif
/* How many days come before each month (0-12). */ /* How many days come before each month (0-12). */
const unsigned short int __mon_yday[2][13] = /* __mon_yday[][] is common to mktime and strptime implementations.
{ --abbotti */
/* Normal years. */ const unsigned short int __mon_yday[2][13];
{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 }, #ifndef NEED_MON_YDAY
/* Leap years. */ # define NEED_MON_YDAY
{ 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 } #endif
};
static time_t ydhms_tm_diff __P ((int, int, int, int, int, const struct tm *)); static time_t ydhms_tm_diff __P ((int, int, int, int, int, const struct tm *));
time_t __mktime_internal __P ((struct tm *, time_t __mktime_internal __P ((struct tm *,
@ -499,11 +498,11 @@ weak_alias (mktime, timelocal)
#ifndef HAVE_STRPTIME #ifndef HAVE_STRPTIME
/* From GNU libc 2.0.6. */ /* From GNU libc 2.1.3. */
/* Ulrich, thanks for helping me out with this! --hniksic */ /* Ulrich, thanks for helping me out with this! --hniksic */
/* strptime - Convert a string representation of time to a time value. /* strptime - Convert a string representation of time to a time value.
Copyright (C) 1996, 1997 Free Software Foundation, Inc. Copyright (C) 1996, 1997, 1998, 1999, 2000 Free Software Foundation, Inc.
This file is part of the GNU C Library. This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996. */ Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996. */
@ -516,12 +515,12 @@ weak_alias (mktime, timelocal)
# define __P(args) PARAMS (args) # define __P(args) PARAMS (args)
#endif /* not __P */ #endif /* not __P */
#if ! HAVE_LOCALTIME_R && ! defined (localtime_r) #if ! HAVE_LOCALTIME_R && ! defined localtime_r
#ifdef _LIBC # ifdef _LIBC
#define localtime_r __localtime_r # define localtime_r __localtime_r
#else # else
/* Approximate localtime_r as best we can in its absence. */ /* Approximate localtime_r as best we can in its absence. */
#define localtime_r my_localtime_r # define localtime_r my_localtime_r
static struct tm *localtime_r __P ((const time_t *, struct tm *)); static struct tm *localtime_r __P ((const time_t *, struct tm *));
static struct tm * static struct tm *
localtime_r (t, tp) localtime_r (t, tp)
@ -534,7 +533,7 @@ localtime_r (t, tp)
*tp = *l; *tp = *l;
return tp; return tp;
} }
#endif /* ! _LIBC */ # endif /* ! _LIBC */
#endif /* ! HAVE_LOCALTIME_R && ! defined (localtime_r) */ #endif /* ! HAVE_LOCALTIME_R && ! defined (localtime_r) */
@ -552,50 +551,118 @@ localtime_r (t, tp)
#endif #endif
/* We intentionally do not use isdigit() for testing because this will /* We intentionally do not use isdigit() for testing because this will
lead to problems with the wide character version. */ lead to problems with the wide character version. */
#define get_number(from, to) \ #define get_number(from, to, n) \
do { \ do { \
int __n = n; \
val = 0; \ val = 0; \
while (*rp == ' ') \
++rp; \
if (*rp < '0' || *rp > '9') \ if (*rp < '0' || *rp > '9') \
return NULL; \ return NULL; \
do { \ do { \
val *= 10; \ val *= 10; \
val += *rp++ - '0'; \ val += *rp++ - '0'; \
} while (val * 10 <= to && *rp >= '0' && *rp <= '9'); \ } while (--__n > 0 && val * 10 <= to && *rp >= '0' && *rp <= '9'); \
if (val < from || val > to) \ if (val < from || val > to) \
return NULL; \ return NULL; \
} while (0) } while (0)
#ifdef _NL_CURRENT #ifdef _NL_CURRENT
# define get_alt_number(from, to) \ /* Added check for __GNUC__ extensions here for Wget. --abbotti */
do { \ # if defined __GNUC__ && __GNUC__ >= 2
# define get_alt_number(from, to, n) \
({ \
__label__ do_normal; \
if (*decided != raw) \ if (*decided != raw) \
{ \ { \
const char *alts = _NL_CURRENT (LC_TIME, ALT_DIGITS); \ const char *alts = _NL_CURRENT (LC_TIME, ALT_DIGITS); \
int __n = n; \
int any = 0; \
while (*rp == ' ') \
++rp; \
val = 0; \ val = 0; \
while (*alts != '\0') \ do { \
{ \ val *= 10; \
size_t len = strlen (alts); \ while (*alts != '\0') \
if (strncasecmp (alts, rp, len) == 0) \ { \
size_t len = strlen (alts); \
if (strncasecmp (alts, rp, len) == 0) \
break; \
alts += len + 1; \
++val; \
} \
if (*alts == '\0') \
{ \
if (*decided == not && ! any) \
goto do_normal; \
/* If we haven't read anything it's an error. */ \
if (! any) \
return NULL; \
/* Correct the premature multiplication. */ \
val /= 10; \
break; \ break; \
alts = strchr (alts, '\0') + 1; \ } \
++val; \ else \
} \
if (*alts == '\0') \
{ \
if (*decided == loc && val != 0) \
return NULL; \
} \
else \
{ \
*decided = loc; \ *decided = loc; \
break; \ } while (--__n > 0 && val * 10 <= to); \
} \ if (val < from || val > to) \
return NULL; \
} \
else \
{ \
do_normal: \
get_number (from, to, n); \
} \
0; \
})
# else
# define get_alt_number(from, to, n) \
do {
if (*decided != raw) \
{ \
const char *alts = _NL_CURRENT (LC_TIME, ALT_DIGITS); \
int __n = n; \
int any = 0; \
while (*rp == ' ') \
++rp; \
val = 0; \
do { \
val *= 10; \
while (*alts != '\0') \
{ \
size_t len = strlen (alts); \
if (strncasecmp (alts, rp, len) == 0) \
break; \
alts += len + 1; \
++val; \
} \
if (*alts == '\0') \
{ \
if (*decided == not && ! any) \
goto do_normal; \
/* If we haven't read anything it's an error. */ \
if (! any) \
return NULL; \
/* Correct the premature multiplication. */ \
val /= 10; \
break; \
} \
else \
*decided = loc; \
} while (--__n > 0 && val * 10 <= to); \
if (val < from || val > to) \
return NULL; \
} \
else \
{ \
do_normal: \
get_number (from, to, n); \
} \ } \
get_number (from, to); \
} while (0) } while (0)
# endif /* defined __GNUC__ && __GNUC__ >= 2 */
#else #else
# define get_alt_number(from, to) \ # define get_alt_number(from, to, n) \
/* We don't have the alternate representation. */ \ /* We don't have the alternate representation. */ \
get_number(from, to) get_number(from, to, n)
#endif #endif
#define recursive(new_fmt) \ #define recursive(new_fmt) \
(*(new_fmt) != '\0' \ (*(new_fmt) != '\0' \
@ -605,6 +672,7 @@ localtime_r (t, tp)
#ifdef _LIBC #ifdef _LIBC
/* This is defined in locale/C-time.c in the GNU libc. */ /* This is defined in locale/C-time.c in the GNU libc. */
extern const struct locale_data _nl_C_LC_TIME; extern const struct locale_data _nl_C_LC_TIME;
extern const unsigned short int __mon_yday[2][13];
# define weekday_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (DAY_1)].string) # define weekday_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (DAY_1)].string)
# define ab_weekday_name \ # define ab_weekday_name \
@ -612,12 +680,14 @@ extern const struct locale_data _nl_C_LC_TIME;
# define month_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (MON_1)].string) # define month_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (MON_1)].string)
# define ab_month_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (ABMON_1)].string) # define ab_month_name (&_nl_C_LC_TIME.values[_NL_ITEM_INDEX (ABMON_1)].string)
# define HERE_D_T_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (D_T_FMT)].string) # define HERE_D_T_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (D_T_FMT)].string)
# define HERE_D_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (D_T_FMT)].string) # define HERE_D_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (D_FMT)].string)
# define HERE_AM_STR (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (AM_STR)].string) # define HERE_AM_STR (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (AM_STR)].string)
# define HERE_PM_STR (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (PM_STR)].string) # define HERE_PM_STR (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (PM_STR)].string)
# define HERE_T_FMT_AMPM \ # define HERE_T_FMT_AMPM \
(_nl_C_LC_TIME.values[_NL_ITEM_INDEX (T_FMT_AMPM)].string) (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (T_FMT_AMPM)].string)
# define HERE_T_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (T_FMT)].string) # define HERE_T_FMT (_nl_C_LC_TIME.values[_NL_ITEM_INDEX (T_FMT)].string)
# define strncasecmp(s1, s2, n) __strncasecmp (s1, s2, n)
#else #else
static char const weekday_name[][10] = static char const weekday_name[][10] =
{ {
@ -644,31 +714,82 @@ static char const ab_month_name[][4] =
# define HERE_PM_STR "PM" # define HERE_PM_STR "PM"
# define HERE_T_FMT_AMPM "%I:%M:%S %p" # define HERE_T_FMT_AMPM "%I:%M:%S %p"
# define HERE_T_FMT "%H:%M:%S" # define HERE_T_FMT "%H:%M:%S"
/* __mon_yday[][] is common to mktime and strptime implementations.
--abbotti */
const unsigned short int __mon_yday[2][13];
# ifndef NEED_MON_YDAY
# define NEED_MON_YDAY
# endif
#endif #endif
/* Status of lookup: do we use the locale data or the raw data? */ /* Status of lookup: do we use the locale data or the raw data? */
enum locale_status { not, loc, raw }; enum locale_status { not, loc, raw };
#ifndef __isleap
/* Nonzero if YEAR is a leap year (every 4 years,
except every 100th isn't, and every 400th is). */
# define __isleap(year) \
((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0))
#endif
/* Compute the day of the week. */
static void
day_of_the_week (struct tm *tm)
{
/* We know that January 1st 1970 was a Thursday (= 4). Compute the
the difference between this data in the one on TM and so determine
the weekday. */
int corr_year = 1900 + tm->tm_year - (tm->tm_mon < 2);
int wday = (-473
+ (365 * (tm->tm_year - 70))
+ (corr_year / 4)
- ((corr_year / 4) / 25) + ((corr_year / 4) % 25 < 0)
+ (((corr_year / 4) / 25) / 4)
+ __mon_yday[0][tm->tm_mon]
+ tm->tm_mday - 1);
tm->tm_wday = ((wday % 7) + 7) % 7;
}
/* Compute the day of the year. */
static void
day_of_the_year (struct tm *tm)
{
tm->tm_yday = (__mon_yday[__isleap (1900 + tm->tm_year)][tm->tm_mon]
+ (tm->tm_mday - 1));
}
static char * static char *
#ifdef _LIBC
internal_function
#endif
strptime_internal __P ((const char *buf, const char *format, struct tm *tm, strptime_internal __P ((const char *buf, const char *format, struct tm *tm,
enum locale_status *decided)); enum locale_status *decided));
static char * static char *
strptime_internal (buf, format, tm, decided) #ifdef _LIBC
const char *buf; internal_function
const char *format; #endif
strptime_internal (rp, fmt, tm, decided)
const char *rp;
const char *fmt;
struct tm *tm; struct tm *tm;
enum locale_status *decided; enum locale_status *decided;
{ {
const char *rp; const char *rp_backup;
const char *fmt;
int cnt; int cnt;
size_t val; size_t val;
int have_I, is_pm; int have_I, is_pm;
int century, want_century;
int have_wday, want_xday;
int have_yday;
int have_mon, have_mday;
rp = buf;
fmt = format;
have_I = is_pm = 0; have_I = is_pm = 0;
century = -1;
want_century = 0;
have_wday = want_xday = have_yday = have_mon = have_mday = 0;
while (*fmt != '\0') while (*fmt != '\0')
{ {
@ -695,6 +816,10 @@ strptime_internal (buf, format, tm, decided)
/* We need this for handling the `E' modifier. */ /* We need this for handling the `E' modifier. */
start_over: start_over:
#endif #endif
/* Make back up of current processing pointer. */
rp_backup = rp;
switch (*fmt++) switch (*fmt++)
{ {
case '%': case '%':
@ -739,6 +864,7 @@ strptime_internal (buf, format, tm, decided)
/* Does not match a weekday name. */ /* Does not match a weekday name. */
return NULL; return NULL;
tm->tm_wday = cnt; tm->tm_wday = cnt;
have_wday = 1;
break; break;
case 'b': case 'b':
case 'B': case 'B':
@ -778,6 +904,7 @@ strptime_internal (buf, format, tm, decided)
/* Does not match a month name. */ /* Does not match a month name. */
return NULL; return NULL;
tm->tm_mon = cnt; tm->tm_mon = cnt;
want_xday = 1;
break; break;
case 'c': case 'c':
/* Match locale's date and time format. */ /* Match locale's date and time format. */
@ -788,12 +915,15 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
if (*decided == not && if (*decided == not &&
strcmp (_NL_CURRENT (LC_TIME, D_T_FMT), HERE_D_T_FMT)) strcmp (_NL_CURRENT (LC_TIME, D_T_FMT), HERE_D_T_FMT))
*decided = loc; *decided = loc;
want_xday = 1;
break; break;
} }
*decided = raw; *decided = raw;
@ -801,17 +931,26 @@ strptime_internal (buf, format, tm, decided)
#endif #endif
if (!recursive (HERE_D_T_FMT)) if (!recursive (HERE_D_T_FMT))
return NULL; return NULL;
want_xday = 1;
break; break;
case 'C': case 'C':
/* Match century number. */ /* Match century number. */
get_number (0, 99); get_number (0, 99, 2);
/* We don't need the number. */ century = val;
want_xday = 1;
break; break;
case 'd': case 'd':
case 'e': case 'e':
/* Match day of month. */ /* Match day of month. */
get_number (1, 31); get_number (1, 31, 2);
tm->tm_mday = val; tm->tm_mday = val;
have_mday = 1;
want_xday = 1;
break;
case 'F':
if (!recursive ("%Y-%m-%d"))
return NULL;
want_xday = 1;
break; break;
case 'x': case 'x':
#ifdef _NL_CURRENT #ifdef _NL_CURRENT
@ -821,12 +960,15 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
if (decided == not if (*decided == not
&& strcmp (_NL_CURRENT (LC_TIME, D_FMT), HERE_D_FMT)) && strcmp (_NL_CURRENT (LC_TIME, D_FMT), HERE_D_FMT))
*decided = loc; *decided = loc;
want_xday = 1;
break; break;
} }
*decided = raw; *decided = raw;
@ -837,32 +979,37 @@ strptime_internal (buf, format, tm, decided)
/* Match standard day format. */ /* Match standard day format. */
if (!recursive (HERE_D_FMT)) if (!recursive (HERE_D_FMT))
return NULL; return NULL;
want_xday = 1;
break; break;
case 'k':
case 'H': case 'H':
/* Match hour in 24-hour clock. */ /* Match hour in 24-hour clock. */
get_number (0, 23); get_number (0, 23, 2);
tm->tm_hour = val; tm->tm_hour = val;
have_I = 0; have_I = 0;
break; break;
case 'I': case 'I':
/* Match hour in 12-hour clock. */ /* Match hour in 12-hour clock. */
get_number (1, 12); get_number (1, 12, 2);
tm->tm_hour = val % 12; tm->tm_hour = val % 12;
have_I = 1; have_I = 1;
break; break;
case 'j': case 'j':
/* Match day number of year. */ /* Match day number of year. */
get_number (1, 366); get_number (1, 366, 3);
tm->tm_yday = val - 1; tm->tm_yday = val - 1;
have_yday = 1;
break; break;
case 'm': case 'm':
/* Match number of month. */ /* Match number of month. */
get_number (1, 12); get_number (1, 12, 2);
tm->tm_mon = val - 1; tm->tm_mon = val - 1;
have_mon = 1;
want_xday = 1;
break; break;
case 'M': case 'M':
/* Match minute. */ /* Match minute. */
get_number (0, 59); get_number (0, 59, 2);
tm->tm_min = val; tm->tm_min = val;
break; break;
case 'n': case 'n':
@ -906,6 +1053,8 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
@ -931,7 +1080,7 @@ strptime_internal (buf, format, tm, decided)
the `get_number' macro. Instead read the number the `get_number' macro. Instead read the number
character for character and construct the result while character for character and construct the result while
doing this. */ doing this. */
time_t secs; time_t secs = 0;
if (*rp < '0' || *rp > '9') if (*rp < '0' || *rp > '9')
/* We need at least one digit. */ /* We need at least one digit. */
return NULL; return NULL;
@ -949,7 +1098,7 @@ strptime_internal (buf, format, tm, decided)
} }
break; break;
case 'S': case 'S':
get_number (0, 61); get_number (0, 61, 2);
tm->tm_sec = val; tm->tm_sec = val;
break; break;
case 'X': case 'X':
@ -960,6 +1109,8 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
@ -976,11 +1127,12 @@ strptime_internal (buf, format, tm, decided)
return NULL; return NULL;
break; break;
case 'u': case 'u':
get_number (1, 7); get_number (1, 7, 1);
tm->tm_wday = val % 7; tm->tm_wday = val % 7;
have_wday = 1;
break; break;
case 'g': case 'g':
get_number (0, 99); get_number (0, 99, 2);
/* XXX This cannot determine any field in TM. */ /* XXX This cannot determine any field in TM. */
break; break;
case 'G': case 'G':
@ -995,27 +1147,32 @@ strptime_internal (buf, format, tm, decided)
case 'U': case 'U':
case 'V': case 'V':
case 'W': case 'W':
get_number (0, 53); get_number (0, 53, 2);
/* XXX This cannot determine any field in TM without some /* XXX This cannot determine any field in TM without some
information. */ information. */
break; break;
case 'w': case 'w':
/* Match number of weekday. */ /* Match number of weekday. */
get_number (0, 6); get_number (0, 6, 1);
tm->tm_wday = val; tm->tm_wday = val;
have_wday = 1;
break; break;
case 'y': case 'y':
/* Match year within century. */ /* Match year within century. */
get_number (0, 99); get_number (0, 99, 2);
tm->tm_year = val >= 50 ? val : val + 100; /* The "Year 2000: The Millennium Rollover" paper suggests that
values in the range 69-99 refer to the twentieth century. */
tm->tm_year = val >= 69 ? val : val + 100;
/* Indicate that we want to use the century, if specified. */
want_century = 1;
want_xday = 1;
break; break;
case 'Y': case 'Y':
/* Match year including century number. */ /* Match year including century number. */
if (sizeof (time_t) > 4) get_number (0, 9999, 4);
get_number (0, 9999);
else
get_number (0, 2036);
tm->tm_year = val - 1900; tm->tm_year = val - 1900;
want_century = 0;
want_xday = 1;
break; break;
case 'Z': case 'Z':
/* XXX How to handle this? */ /* XXX How to handle this? */
@ -1037,11 +1194,14 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
if (strcmp (fmt, HERE_D_T_FMT)) if (strcmp (fmt, HERE_D_T_FMT))
*decided = loc; *decided = loc;
want_xday = 1;
break; break;
} }
*decided = raw; *decided = raw;
@ -1050,6 +1210,7 @@ strptime_internal (buf, format, tm, decided)
normal representation. */ normal representation. */
if (!recursive (HERE_D_T_FMT)) if (!recursive (HERE_D_T_FMT))
return NULL; return NULL;
want_xday = 1;
break; break;
case 'C': case 'C':
case 'y': case 'y':
@ -1071,6 +1232,8 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
@ -1095,6 +1258,8 @@ strptime_internal (buf, format, tm, decided)
{ {
if (*decided == loc) if (*decided == loc)
return NULL; return NULL;
else
rp = rp_backup;
} }
else else
{ {
@ -1127,53 +1292,60 @@ strptime_internal (buf, format, tm, decided)
case 'd': case 'd':
case 'e': case 'e':
/* Match day of month using alternate numeric symbols. */ /* Match day of month using alternate numeric symbols. */
get_alt_number (1, 31); get_alt_number (1, 31, 2);
tm->tm_mday = val; tm->tm_mday = val;
have_mday = 1;
want_xday = 1;
break; break;
case 'H': case 'H':
/* Match hour in 24-hour clock using alternate numeric /* Match hour in 24-hour clock using alternate numeric
symbols. */ symbols. */
get_alt_number (0, 23); get_alt_number (0, 23, 2);
tm->tm_hour = val; tm->tm_hour = val;
have_I = 0; have_I = 0;
break; break;
case 'I': case 'I':
/* Match hour in 12-hour clock using alternate numeric /* Match hour in 12-hour clock using alternate numeric
symbols. */ symbols. */
get_alt_number (1, 12); get_alt_number (1, 12, 2);
tm->tm_hour = val - 1; tm->tm_hour = val - 1;
have_I = 1; have_I = 1;
break; break;
case 'm': case 'm':
/* Match month using alternate numeric symbols. */ /* Match month using alternate numeric symbols. */
get_alt_number (1, 12); get_alt_number (1, 12, 2);
tm->tm_mon = val - 1; tm->tm_mon = val - 1;
have_mon = 1;
want_xday = 1;
break; break;
case 'M': case 'M':
/* Match minutes using alternate numeric symbols. */ /* Match minutes using alternate numeric symbols. */
get_alt_number (0, 59); get_alt_number (0, 59, 2);
tm->tm_min = val; tm->tm_min = val;
break; break;
case 'S': case 'S':
/* Match seconds using alternate numeric symbols. */ /* Match seconds using alternate numeric symbols. */
get_alt_number (0, 61); get_alt_number (0, 61, 2);
tm->tm_sec = val; tm->tm_sec = val;
break; break;
case 'U': case 'U':
case 'V': case 'V':
case 'W': case 'W':
get_alt_number (0, 53); get_alt_number (0, 53, 2);
/* XXX This cannot determine any field in TM without /* XXX This cannot determine any field in TM without
further information. */ further information. */
break; break;
case 'w': case 'w':
/* Match number of weekday using alternate numeric symbols. */ /* Match number of weekday using alternate numeric symbols. */
get_alt_number (0, 6); get_alt_number (0, 6, 1);
tm->tm_wday = val; tm->tm_wday = val;
have_wday = 1;
break; break;
case 'y': case 'y':
/* Match year within century using alternate numeric symbols. */ /* Match year within century using alternate numeric symbols. */
get_alt_number (0, 99); get_alt_number (0, 99, 2);
tm->tm_year = val >= 69 ? val : val + 100;
want_xday = 1;
break; break;
default: default:
return NULL; return NULL;
@ -1187,6 +1359,31 @@ strptime_internal (buf, format, tm, decided)
if (have_I && is_pm) if (have_I && is_pm)
tm->tm_hour += 12; tm->tm_hour += 12;
if (century != -1)
{
if (want_century)
tm->tm_year = tm->tm_year % 100 + (century - 19) * 100;
else
/* Only the century, but not the year. Strange, but so be it. */
tm->tm_year = (century - 19) * 100;
}
if (want_xday && !have_wday) {
if ( !(have_mon && have_mday) && have_yday) {
/* we don't have tm_mon and/or tm_mday, compute them */
int t_mon = 0;
while (__mon_yday[__isleap(1900 + tm->tm_year)][t_mon] <= tm->tm_yday)
t_mon++;
if (!have_mon)
tm->tm_mon = t_mon - 1;
if (!have_mday)
tm->tm_mday = tm->tm_yday - __mon_yday[__isleap(1900 + tm->tm_year)][t_mon - 1] + 1;
}
day_of_the_week (tm);
}
if (want_xday && !have_yday)
day_of_the_year (tm);
return (char *) rp; return (char *) rp;
} }
@ -1207,6 +1404,17 @@ strptime (buf, format, tm)
} }
#endif /* not HAVE_STRPTIME */ #endif /* not HAVE_STRPTIME */
#ifdef NEED_MON_YDAY
/* __mon_yday[][] is common to mktime and strptime implementations.
--abbotti */
const unsigned short int __mon_yday[2][13] =
{
/* Normal years. */
{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 },
/* Leap years. */
{ 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 }
};
#endif
#ifndef HAVE_USLEEP #ifndef HAVE_USLEEP
#ifndef WINDOWS #ifndef WINDOWS