FreeBSD Bugzilla – Attachment 147680 Details for
Bug 137307
[libc] [patch] Enhance strptime(3) to support %U and %W
Home
|
New
|
Browse
|
Search
|
[?]
|
Reports
|
Help
|
New Account
|
Log In
Remember
[x]
|
Forgot Password
Login:
[x]
[patch]
WIP patch
strptime-wip.diff (text/plain), 5.24 KB, created by
Pedro F. Giffuni
on 2014-09-25 23:16:29 UTC
(
hide
)
Description:
WIP patch
Filename:
MIME Type:
Creator:
Pedro F. Giffuni
Created:
2014-09-25 23:16:29 UTC
Size:
5.24 KB
patch
obsolete
>Index: lib/libc/stdtime/strptime.c >=================================================================== >--- lib/libc/stdtime/strptime.c (revision 272146) >+++ lib/libc/stdtime/strptime.c (working copy) >@@ -55,11 +55,33 @@ > #include "un-namespace.h" > #include "libc_private.h" > #include "timelocal.h" >+#include "tzfile.h" > > static char * _strptime(const char *, const char *, struct tm *, int *, locale_t); > >-#define asizeof(a) (sizeof (a) / sizeof ((a)[0])) >+#define asizeof(a) (sizeof(a) / sizeof((a)[0])) > >+#define FLAG_NONE (1 << 0) >+#define FLAG_YEAR (1 << 1) >+#define FLAG_MONTH (1 << 2) >+#define FLAG_YDAY (1 << 3) >+#define FLAG_MDAY (1 << 4) >+#define FLAG_WDAY (1 << 5) >+ >+/* >+ * Calculate the week day of the first day of a year. Valid for >+ * the Gregorian calendar, which began Sept 14, 1752 in the UK >+ * and its colonies. Ref: >+ * http://en.wikipedia.org/wiki/Calculating_the_day_of_the_week/ >+ */ >+ >+static int >+first_wday_of(int year) >+{ >+ return (((2 * (3 - (year / 100) % 4)) + (year % 100) + >+ ((year % 100) / 4) + (isleap(year) ? 6 : 0) + 1) % 7); >+} >+ > static char * > _strptime(const char *buf, const char *fmt, struct tm *tm, int *GMTp, > locale_t locale) >@@ -66,10 +88,18 @@ > { > char c; > const char *ptr; >+ int day_offset = -1, wday_offset; > int i, len; >+ int flags; > int Ealternative, Oalternative; >- struct lc_time_T *tptr = __get_current_time_locale(locale); >+ const struct lc_time_T *tptr = __get_current_time_locale(locale); >+ static int start_of_month[2][13] = { >+ {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365}, >+ {0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366} >+ }; > >+ flags = FLAG_NONE; >+ > ptr = fmt; > while (*ptr != 0) { > if (*buf == 0) >@@ -119,7 +149,9 @@ > if (i < 19) > return (NULL); > >- tm->tm_year = i * 100 - 1900; >+ tm->tm_year = i * 100 - TM_YEAR_BASE; >+ flags |= FLAG_YEAR; >+ > break; > > case 'c': >@@ -197,6 +229,8 @@ > return (NULL); > > tm->tm_yday = i - 1; >+ flags |= FLAG_YDAY; >+ > break; > > case 'M': >@@ -303,7 +337,32 @@ > return (NULL); > > tm->tm_wday = i; >+ if (day_offset >= 0 && (i - day_offset) != 0) { >+ tm->tm_yday += i - day_offset; >+ i = 0; >+ while (tm->tm_yday >= >+ start_of_month[isleap(tm->tm_year + >+ TM_YEAR_BASE)][i]) >+ i++; >+ if (i > 12) >+ { >+ i = 1; >+ tm->tm_yday -= >+ start_of_month[isleap(tm->tm_year + >+ TM_YEAR_BASE)] >+ [12]; >+ tm->tm_year++; >+ } >+ tm->tm_mon = i - 1; >+ tm->tm_mday = tm->tm_yday - >+ start_of_month[isleap(tm->tm_year + >+ TM_YEAR_BASE)] >+ [i - 1] + 1; >+ } > buf += len; >+ flags |= FLAG_YEAR | FLAG_MONTH | FLAG_YDAY | >+ FLAG_MDAY | FLAG_WDAY; >+ > break; > > case 'U': >@@ -313,6 +372,8 @@ > * information present in the tm structure at this > * point to calculate a real value, so just check the > * range for now. >+ * We expect that the year has already been >+ * parsed. > */ > if (!isdigit_l((unsigned char)*buf, locale)) > return (NULL); >@@ -327,6 +388,45 @@ > if (i > 53) > return (NULL); > >+ /* Week numbers are 1-origin. So that we can always >+ * return the date of a Sunday (or Monday), treat week >+ * 0 as week 1. >+ */ >+ >+ if (i == 0) >+ i = 1; >+ >+ if (c == 'U') >+ day_offset = TM_SUNDAY; >+ else >+ day_offset = TM_MONDAY; >+ >+ /* Set the date to the first Sunday (or Monday) >+ * of the specified week of the year. >+ */ >+ >+ tm->tm_yday = (7 - first_wday_of(tm->tm_year + >+ TM_YEAR_BASE) + day_offset) % 7 + (i - 1) * 7; >+ i = 0; >+ while (tm->tm_yday >= >+ start_of_month[isleap(tm->tm_year + TM_YEAR_BASE)][i]) >+ i++; >+ if (i > 12) >+ { >+ i = 1; >+ tm->tm_yday -= >+ start_of_month[isleap(tm->tm_year + >+ TM_YEAR_BASE)][12]; >+ tm->tm_year++; >+ } >+ tm->tm_mon = i - 1; >+ tm->tm_mday = tm->tm_yday - >+ start_of_month[isleap(tm->tm_year + TM_YEAR_BASE)] >+ [i - 1] + 1; >+ tm->tm_wday = day_offset; >+ flags |= FLAG_YEAR | FLAG_MONTH | FLAG_YDAY | >+ FLAG_MDAY | FLAG_WDAY; >+ > break; > > case 'w': >@@ -338,6 +438,7 @@ > return (NULL); > > tm->tm_wday = i; >+ flags |= FLAG_WDAY; > > break; > >@@ -374,6 +475,7 @@ > return (NULL); > > tm->tm_mday = i; >+ flags |= FLAG_MDAY; > > break; > >@@ -413,6 +515,8 @@ > > tm->tm_mon = i; > buf += len; >+ flags |= FLAG_MONTH; >+ > break; > > case 'm': >@@ -430,6 +534,7 @@ > return (NULL); > > tm->tm_mon = i - 1; >+ flags |= FLAG_MONTH; > > break; > >@@ -471,7 +576,7 @@ > len--; > } > if (c == 'Y') >- i -= 1900; >+ i -= TM_YEAR_BASE; > if (c == 'y' && i < 69) > i += 100; > if (i < 0) >@@ -478,6 +583,7 @@ > return (NULL); > > tm->tm_year = i; >+ flags |= FLAG_YEAR; > > break; > >@@ -543,10 +649,25 @@ > break; > } > } >+ >+ if (flags & (FLAG_YEAR | FLAG_MONTH)) { >+ if (!(flags & FLAG_YDAY) && (flags & FLAG_MDAY)) >+ tm->tm_yday = start_of_month[isleap(tm->tm_year >+ + TM_YEAR_BASE)][tm->tm_mon] + (tm->tm_mday - 1); >+ if (!(flags & FLAG_WDAY)) { >+ i = 0; >+ wday_offset = first_wday_of(tm->tm_year); >+ while (i++ <= tm->tm_yday) >+ if (wday_offset++ >= 6) >+ wday_offset = 0; >+ >+ tm->tm_wday = wday_offset; >+ } >+ } >+ > return ((char *)buf); > } > >- > char * > strptime_l(const char * __restrict buf, const char * __restrict fmt, > struct tm * __restrict tm, locale_t loc)
You cannot view the attachment while viewing its details because your browser does not support IFRAMEs.
View the attachment on a separate page
.
View Attachment As Diff
View Attachment As Raw
Actions:
View
|
Diff
Attachments on
bug 137307
:
98401
|
98402
|
144894
|
145108
|
145481
|
145482
|
147573
|
147680
|
147740