2011-02-12 00:22:29 -05:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <langinfo.h>
|
|
|
|
#include <time.h>
|
2011-08-16 10:38:33 -04:00
|
|
|
#include <ctype.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <strings.h>
|
2011-02-12 00:22:29 -05:00
|
|
|
|
|
|
|
char *strptime(const char *s, const char *f, struct tm *tm)
|
|
|
|
{
|
2011-08-16 10:38:33 -04:00
|
|
|
int i, w, neg, adj, min, range, *dest;
|
|
|
|
const char *ex;
|
|
|
|
size_t len;
|
|
|
|
while (*f) {
|
|
|
|
if (*f != '%') {
|
|
|
|
if (isspace(*f)) for (; *s && isspace(*s); s++);
|
|
|
|
else if (*s != *f) return 0;
|
|
|
|
else s++;
|
|
|
|
f++;
|
|
|
|
}
|
|
|
|
f++;
|
|
|
|
if (*f == '+') f++;
|
|
|
|
if (isdigit(*f)) w=strtoul(f, (void *)&f, 10);
|
|
|
|
else w=-1;
|
|
|
|
adj=0;
|
|
|
|
switch (*f++) {
|
|
|
|
case 'a': case 'A':
|
|
|
|
dest = &tm->tm_wday;
|
|
|
|
min = ABDAY_1;
|
|
|
|
range = 7;
|
|
|
|
goto symbolic_range;
|
|
|
|
case 'b': case 'B': case 'h':
|
|
|
|
dest = &tm->tm_mon;
|
|
|
|
min = ABMON_1;
|
|
|
|
range = 12;
|
|
|
|
goto symbolic_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'c':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, nl_langinfo(D_T_FMT), tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'C':
|
2011-08-16 10:38:33 -04:00
|
|
|
case 'd': case 'e':
|
|
|
|
dest = &tm->tm_mday;
|
|
|
|
min = 1;
|
|
|
|
range = 31;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'D':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, "%m/%d/%y", tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'H':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_hour;
|
|
|
|
min = 0;
|
|
|
|
range = 24;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'I':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_hour;
|
|
|
|
min = 1;
|
|
|
|
range = 12;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'j':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_yday;
|
|
|
|
min = 1;
|
|
|
|
range = 366;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'm':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_mon;
|
|
|
|
min = 1;
|
|
|
|
range = 12;
|
|
|
|
adj = 1;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'M':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_min;
|
|
|
|
min = 0;
|
|
|
|
range = 60;
|
|
|
|
goto numeric_range;
|
|
|
|
case 'n': case 't':
|
|
|
|
for (; *s && isspace(*s); s++);
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'p':
|
2011-08-16 10:38:33 -04:00
|
|
|
ex = nl_langinfo(AM_STR);
|
|
|
|
len = strlen(ex);
|
|
|
|
if (!strncasecmp(s, ex, len)) {
|
|
|
|
tm->tm_hour %= 12;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ex = nl_langinfo(PM_STR);
|
|
|
|
len = strlen(ex);
|
|
|
|
if (!strncasecmp(s, ex, len)) {
|
|
|
|
tm->tm_hour %= 12;
|
|
|
|
tm->tm_hour += 12;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'r':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, nl_langinfo(T_FMT_AMPM), tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'R':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, "%H:%M", tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'S':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_sec;
|
|
|
|
min = 0;
|
|
|
|
range = 61;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'T':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, "%H:%M:%S", tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'U':
|
|
|
|
case 'W':
|
2011-08-16 10:38:33 -04:00
|
|
|
//FIXME
|
|
|
|
return 0;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'w':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_wday;
|
|
|
|
min = 0;
|
|
|
|
range = 7;
|
|
|
|
goto numeric_range;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'x':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, nl_langinfo(D_FMT), tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'X':
|
2011-08-16 10:38:33 -04:00
|
|
|
s = strptime(s, nl_langinfo(T_FMT), tm);
|
|
|
|
if (!s) return 0;
|
|
|
|
break;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'y':
|
2011-08-16 10:38:33 -04:00
|
|
|
//FIXME
|
|
|
|
return 0;
|
2011-02-12 00:22:29 -05:00
|
|
|
case 'Y':
|
2011-08-16 10:38:33 -04:00
|
|
|
dest = &tm->tm_year;
|
|
|
|
if (w<0) w=4;
|
|
|
|
adj = 1900;
|
|
|
|
goto numeric_digits;
|
|
|
|
case '%':
|
|
|
|
if (*s++ != '%') return 0;
|
|
|
|
break;
|
|
|
|
numeric_range:
|
|
|
|
if (!isdigit(*s)) return 0;
|
|
|
|
*dest = 0;
|
|
|
|
for (i=1; i<=min+range && isdigit(*s); i*=10)
|
|
|
|
*dest = *dest * 10 + *s++ - '0';
|
|
|
|
if (*dest - min >= (unsigned)range) return 0;
|
|
|
|
*dest -= adj;
|
|
|
|
switch((char *)dest - (char *)tm) {
|
|
|
|
case offsetof(struct tm, tm_yday):
|
|
|
|
;
|
|
|
|
}
|
|
|
|
goto update;
|
|
|
|
numeric_digits:
|
|
|
|
neg = 0;
|
|
|
|
if (*s == '+') s++;
|
|
|
|
else if (*s == '-') neg=1, s++;
|
|
|
|
if (!isdigit(*s)) return 0;
|
|
|
|
for (i=0; i<w && isdigit(*s); i++)
|
|
|
|
*dest = *dest * 10 + *s++ - '0';
|
|
|
|
if (neg) *dest = -*dest;
|
|
|
|
*dest -= adj;
|
|
|
|
goto update;
|
|
|
|
symbolic_range:
|
|
|
|
for (i=2*range-1; i>=0; i--) {
|
|
|
|
ex = nl_langinfo(min+i);
|
|
|
|
len = strlen(ex);
|
|
|
|
if (strncasecmp(s, ex, len)) continue;
|
|
|
|
*dest = i % range;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i<0) return 0;
|
|
|
|
goto update;
|
|
|
|
update:
|
|
|
|
//FIXME
|
|
|
|
;
|
2011-02-12 00:22:29 -05:00
|
|
|
}
|
|
|
|
}
|
2011-08-16 10:38:33 -04:00
|
|
|
return (char *)s;
|
2011-02-12 00:22:29 -05:00
|
|
|
}
|