#include "sh.h"
RCSID("$Id: tc.sched.c,v 1.1.1.2 2001/06/28 23:10:54 bbraun Exp $")
#include "ed.h"
#include "tc.h"
extern int just_signaled;
struct sched_event {
struct sched_event *t_next;
time_t t_when;
Char **t_lex;
};
static struct sched_event *sched_ptr = NULL;
time_t
sched_next()
{
if (sched_ptr)
return (sched_ptr->t_when);
return ((time_t) - 1);
}
void
dosched(v, c)
register Char **v;
struct command *c;
{
register struct sched_event *tp, *tp1, *tp2;
time_t cur_time;
int count, hours, minutes, dif_hour, dif_min;
Char *cp;
bool relative;
struct tm *ltp;
USE(c);
#if defined(_MINIX) && !defined(_MINIX_VMD)
char kludge[10];
extern char *sprintf();
sprintf(kludge, CGETS(24, 1, "kludge"));
#endif
v++;
cp = *v++;
if (cp == NULL) {
Char *fmt;
if ((fmt = varval(STRsched)) == STRNULL)
fmt = str2short("%h\t%T\t%R\n");
for (count = 1, tp = sched_ptr; tp; count++, tp = tp->t_next) {
Char buf[BUFSIZE], sbuf[BUFSIZE];
blkexpand(tp->t_lex, buf);
tprintf(FMT_SCHED, sbuf, fmt, sizeof(sbuf),
short2str(buf), tp->t_when, (ptr_t) &count);
for (cp = sbuf; *cp;)
xputchar(*cp++);
}
return;
}
if (*cp == '-') {
if (!sched_ptr)
stderror(ERR_NOSCHED);
if (*v)
stderror(ERR_SCHEDUSAGE);
count = atoi(short2str(++cp));
if (count <= 0)
stderror(ERR_SCHEDUSAGE);
tp = sched_ptr;
tp1 = 0;
while (--count) {
if (tp->t_next == 0)
break;
else {
tp1 = tp;
tp = tp->t_next;
}
}
if (count)
stderror(ERR_SCHEDEV);
if (tp1 == 0)
sched_ptr = tp->t_next;
else
tp1->t_next = tp->t_next;
blkfree(tp->t_lex);
xfree((ptr_t) tp);
return;
}
if (!*v)
stderror(ERR_SCHEDCOM);
relative = 0;
if (!Isdigit(*cp)) {
if (*cp != '+')
stderror(ERR_SCHEDUSAGE);
cp++, relative++;
}
minutes = 0;
hours = atoi(short2str(cp));
while (*cp && *cp != ':' && *cp != 'a' && *cp != 'p')
cp++;
if (*cp && *cp == ':')
minutes = atoi(short2str(++cp));
if ((hours < 0) || (minutes < 0) ||
(hours > 23) || (minutes > 59))
stderror(ERR_SCHEDTIME);
while (*cp && *cp != 'p' && *cp != 'a')
cp++;
if (*cp && relative)
stderror(ERR_SCHEDREL);
if (*cp == 'p')
hours += 12;
(void) time(&cur_time);
ltp = localtime(&cur_time);
if (relative) {
dif_hour = hours;
dif_min = minutes;
}
else {
if ((dif_hour = hours - ltp->tm_hour) < 0)
dif_hour += 24;
if ((dif_min = minutes - ltp->tm_min) < 0) {
dif_min += 60;
if ((--dif_hour) < 0)
dif_hour = 23;
}
}
tp = (struct sched_event *) xcalloc(1, sizeof *tp);
#ifdef _SX
tp->t_when = cur_time - ltp->tm_sec + dif_hour * 3600 + dif_min * 60;
#else
tp->t_when = cur_time - ltp->tm_sec + dif_hour * 3600L + dif_min * 60L;
#endif
if (!sched_ptr || tp->t_when < sched_ptr->t_when) {
tp->t_next = sched_ptr;
sched_ptr = tp;
}
else {
tp1 = sched_ptr->t_next;
tp2 = sched_ptr;
while (tp1 && tp->t_when >= tp1->t_when) {
tp2 = tp1;
tp1 = tp1->t_next;
}
tp->t_next = tp1;
tp2->t_next = tp;
}
tp->t_lex = saveblk(v);
}
void
sched_run(n)
int n;
{
time_t cur_time;
register struct sched_event *tp, *tp1;
struct wordent cmd, *nextword, *lastword;
struct command *t;
Char **v, *cp;
extern Char GettingInput;
#ifdef BSDSIGS
sigmask_t omask;
omask = sigblock(sigmask(SIGINT)) & ~sigmask(SIGINT);
#else
(void) sighold(SIGINT);
#endif
USE(n);
(void) time(&cur_time);
tp = sched_ptr;
if (!(tp && tp->t_when < cur_time)) {
#ifdef BSDSIGS
(void) sigsetmask(omask);
#else
(void) sigrelse(SIGINT);
#endif
return;
}
if (GettingInput)
(void) Cookedmode();
while (tp && tp->t_when < cur_time) {
if (seterr) {
xfree((ptr_t) seterr);
seterr = NULL;
}
cmd.word = STRNULL;
lastword = &cmd;
v = tp->t_lex;
for (cp = *v; cp; cp = *++v) {
nextword = (struct wordent *) xcalloc(1, sizeof cmd);
nextword->word = Strsave(cp);
lastword->next = nextword;
nextword->prev = lastword;
lastword = nextword;
}
lastword->next = &cmd;
cmd.prev = lastword;
tp1 = tp;
sched_ptr = tp = tp1->t_next;
blkfree(tp1->t_lex);
xfree((ptr_t) tp1);
alias(&cmd);
t = syntax(cmd.next, &cmd, 0);
if (seterr)
stderror(ERR_OLD);
execute(t, -1, NULL, NULL);
freelex(&cmd), freesyn(t);
}
if (GettingInput && !just_signaled) {
(void) Rawmode();
ClearLines();
ClearDisp();
Refresh();
}
just_signaled = 0;
#ifdef BSDSIGS
(void) sigsetmask(omask);
#else
(void) sigrelse(SIGINT);
#endif
}