#include <sys/cdefs.h>
__RCSID("$FreeBSD: src/usr.bin/calendar/day.c,v 1.20 2002/06/13 21:20:56 grog Exp $");
#include <sys/types.h>
#include <sys/uio.h>
#include <ctype.h>
#include <err.h>
#include <locale.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include "pathnames.h"
#include "calendar.h"
struct tm *tp;
int *cumdays, offset, yrdays;
char dayname[10];
int daytab[][14] = {
{ 0, -1, 30, 58, 89, 119, 150, 180, 211, 242, 272, 303, 333, 364 },
{ 0, -1, 30, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 },
};
static char const *days[] = {
"sun", "mon", "tue", "wed", "thu", "fri", "sat", NULL,
};
static const char *months[] = {
"jan", "feb", "mar", "apr", "may", "jun",
"jul", "aug", "sep", "oct", "nov", "dec", NULL,
};
static struct fixs fndays[8];
static struct fixs ndays[8];
static struct fixs fnmonths[13];
static struct fixs nmonths[13];
void setnnames(void)
{
char buf[80];
int i, l;
struct tm tm;
for (i = 0; i < 7; i++) {
tm.tm_wday = i;
strftime(buf, sizeof(buf), "%a", &tm);
for (l = strlen(buf);
l > 0 && isspace((unsigned char)buf[l - 1]);
l--)
;
buf[l] = '\0';
if (ndays[i].name != NULL)
free(ndays[i].name);
if ((ndays[i].name = strdup(buf)) == NULL)
errx(1, "cannot allocate memory");
ndays[i].len = strlen(buf);
strftime(buf, sizeof(buf), "%A", &tm);
for (l = strlen(buf);
l > 0 && isspace((unsigned char)buf[l - 1]);
l--)
;
buf[l] = '\0';
if (fndays[i].name != NULL)
free(fndays[i].name);
if ((fndays[i].name = strdup(buf)) == NULL)
errx(1, "cannot allocate memory");
fndays[i].len = strlen(buf);
}
for (i = 0; i < 12; i++) {
tm.tm_mon = i;
strftime(buf, sizeof(buf), "%b", &tm);
for (l = strlen(buf);
l > 0 && isspace((unsigned char)buf[l - 1]);
l--)
;
buf[l] = '\0';
if (nmonths[i].name != NULL)
free(nmonths[i].name);
if ((nmonths[i].name = strdup(buf)) == NULL)
errx(1, "cannot allocate memory");
nmonths[i].len = strlen(buf);
strftime(buf, sizeof(buf), "%B", &tm);
for (l = strlen(buf);
l > 0 && isspace((unsigned char)buf[l - 1]);
l--)
;
buf[l] = '\0';
if (fnmonths[i].name != NULL)
free(fnmonths[i].name);
if ((fnmonths[i].name = strdup(buf)) == NULL)
errx(1, "cannot allocate memory");
fnmonths[i].len = strlen(buf);
}
}
#ifndef __APPLE__
void
settime(now)
time_t now;
#else
void settime(time_t now, short lookahead, short weekend)
#endif
{
char *oldl, *lbufp;
tp = localtime(&now);
if ( isleap(tp->tm_year + 1900) ) {
yrdays = 366;
cumdays = daytab[1];
} else {
yrdays = 365;
cumdays = daytab[0];
}
#ifndef __APPLE__
offset = tp->tm_wday == Friday ? 3 : 1;
#else
offset = tp->tm_wday == Friday ? lookahead + weekend : lookahead;
#endif
header[5].iov_base = dayname;
oldl = NULL;
lbufp = setlocale(LC_TIME, NULL);
if (lbufp != NULL && (oldl = strdup(lbufp)) == NULL)
errx(1, "cannot allocate memory");
(void) setlocale(LC_TIME, "C");
header[5].iov_len = strftime(dayname, sizeof(dayname), "%A", tp);
(void) setlocale(LC_TIME, (oldl != NULL ? oldl : ""));
if (oldl != NULL)
free(oldl);
setnnames();
}
time_t Mktime (dp)
char *dp;
{
time_t t;
int d, m, y;
struct tm tm;
(void)time(&t);
tp = localtime(&t);
tm.tm_sec = 0;
tm.tm_min = 0;
tm.tm_hour = 0;
tm.tm_wday = 0;
tm.tm_mday = tp->tm_mday;
tm.tm_mon = tp->tm_mon;
tm.tm_year = tp->tm_year;
switch (sscanf(dp, "%d.%d.%d", &d, &m, &y)) {
case 3:
if (y > 1900)
y -= 1900;
tm.tm_year = y;
case 2:
tm.tm_mon = m - 1;
case 1:
tm.tm_mday = d;
}
#ifdef DEBUG
fprintf(stderr, "Mktime: %d %d %s\n", (int)mktime(&tm), (int)t,
asctime(&tm));
#endif
return(mktime(&tm));
}
int
isnow(endp, monthp, dayp, varp)
char *endp;
int *monthp;
int *dayp;
int *varp;
{
int day, flags, month = 0, v1, v2;
flags = 0;
if (!(v1 = getfield(endp, &endp, &flags)))
return (0);
if (flags & F_EASTER)
day = v1 - 1;
else if (flags & F_ISDAY || v1 > 12) {
day = v1;
if (!(month = getfield(endp, &endp, &flags)))
month = tp->tm_mon + 1;
}
else if (flags & F_ISMONTH) {
month = v1;
if (!(day = getfield(endp, &endp, &flags)))
day = 1;
}
else {
v2 = getfield(endp, &endp, &flags);
if (flags & F_ISMONTH) {
day = v1;
month = v2;
*varp = 0;
}
else {
month = v1;
day = v2 ? v2 : 1;
*varp = 0;
}
}
if (flags & F_ISDAY) {
#ifdef DEBUG
fprintf(stderr, "\nday: %d %s month %d\n", day, endp, month);
#endif
*varp = 1;
if (day < 0 || day >= 10) {
if (day < 0) {
v1 = day/10 - 1;
day = 10 + (day % 10);
v2 = tp->tm_mday + (((day - 1) - tp->tm_wday + 7) % 7);
if (cumdays[month+1] - cumdays[month] >= v2
&& ((int)((cumdays[month+1] -
cumdays[month] - v2) / 7) + 1) == -v1)
day = v2;
else {
day = tp->tm_mday - 1;
if (day == 0)
return (0);
}
}
else {
v1 = day/10;
day = day % 10;
v2 = tp->tm_mday + (((day - 1) - tp->tm_wday + 7) % 7);
if ( ((v2 - 1 + 7) / 7) == v1 )
day = v2;
else {
day = tp->tm_mday - 1;
if (day == 0)
return (0);
}
}
}
else {
day = tp->tm_mday + (((day - 1) - tp->tm_wday + 7) % 7);
*varp = 1;
}
}
if (!(flags & F_EASTER)) {
*monthp = month;
*dayp = day;
day = cumdays[month] + day;
}
else {
for (v1 = 0; day > cumdays[v1]; v1++)
;
*monthp = v1 - 1;
*dayp = day - cumdays[v1 - 1];
*varp = 1;
}
#ifdef DEBUG
fprintf(stderr, "day2: day %d(%d-%d) yday %d\n", *dayp, day, cumdays[month], tp->tm_yday);
#endif
if (day >= tp->tm_yday - f_dayBefore &&
day <= tp->tm_yday + offset + f_dayAfter)
return (1);
if (yrdays - tp->tm_yday + day <= offset + f_dayAfter ||
tp->tm_yday + day - f_dayBefore < 0
)
return (1);
return (0);
}
int
getmonth(s)
char *s;
{
const char **p;
struct fixs *n;
for (n = fnmonths; n->name; ++n)
if (!strncasecmp(s, n->name, n->len))
return ((n - fnmonths) + 1);
for (n = nmonths; n->name; ++n)
if (!strncasecmp(s, n->name, n->len))
return ((n - nmonths) + 1);
for (p = months; *p; ++p)
if (!strncasecmp(s, *p, 3))
return ((p - months) + 1);
return (0);
}
int
getday(s)
char *s;
{
const char **p;
struct fixs *n;
for (n = fndays; n->name; ++n)
if (!strncasecmp(s, n->name, n->len))
return ((n - fndays) + 1);
for (n = ndays; n->name; ++n)
if (!strncasecmp(s, n->name, n->len))
return ((n - ndays) + 1);
for (p = days; *p; ++p)
if (!strncasecmp(s, *p, 3))
return ((p - days) + 1);
return (0);
}
int
getdayvar(s)
char *s;
{
int offs;
offs = strlen(s);
switch(*(s + offs - 2)) {
case '-':
return(-(atoi(s + offs - 1)));
break;
case '+':
return(atoi(s + offs - 1));
break;
}
if (offs > 4 && !strcasecmp(s + offs - 4, "last"))
return(-1);
else if (offs > 5 && !strcasecmp(s + offs - 5, "first"))
return(+1);
else if (offs > 6 && !strcasecmp(s + offs - 6, "second"))
return(+2);
else if (offs > 5 && !strcasecmp(s + offs - 5, "third"))
return(+3);
else if (offs > 6 && !strcasecmp(s + offs - 6, "fourth"))
return(+4);
return(0);
}