tc.sched.c revision 69408
1/* $Header: /src/pub/tcsh/tc.sched.c,v 3.17 2000/07/04 19:46:23 christos Exp $ */
2/*
3 * tc.sched.c: Scheduled command execution
4 *
5 * Karl Kleinpaste: Computer Consoles Inc. 1984
6 */
7/*-
8 * Copyright (c) 1980, 1991 The Regents of the University of California.
9 * All rights reserved.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 *    notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 *    notice, this list of conditions and the following disclaimer in the
18 *    documentation and/or other materials provided with the distribution.
19 * 3. All advertising materials mentioning features or use of this software
20 *    must display the following acknowledgement:
21 *	This product includes software developed by the University of
22 *	California, Berkeley and its contributors.
23 * 4. Neither the name of the University nor the names of its contributors
24 *    may be used to endorse or promote products derived from this software
25 *    without specific prior written permission.
26 *
27 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
28 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
31 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
36 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 * SUCH DAMAGE.
38 */
39#include "sh.h"
40
41RCSID("$Id: tc.sched.c,v 3.17 2000/07/04 19:46:23 christos Exp $")
42
43#include "ed.h"
44#include "tc.h"
45
46extern int just_signaled;
47
48struct sched_event {
49    struct sched_event *t_next;
50    time_t t_when;
51    Char  **t_lex;
52};
53static struct sched_event *sched_ptr = NULL;
54
55
56time_t
57sched_next()
58{
59    if (sched_ptr)
60	return (sched_ptr->t_when);
61    return ((time_t) - 1);
62}
63
64/*ARGSUSED*/
65void
66dosched(v, c)
67    register Char **v;
68    struct command *c;
69{
70    register struct sched_event *tp, *tp1, *tp2;
71    time_t  cur_time;
72    int     count, hours, minutes, dif_hour, dif_min;
73    Char   *cp;
74    bool    relative;		/* time specified as +hh:mm */
75    struct tm *ltp;
76
77    USE(c);
78/* This is a major kludge because of a gcc linker  */
79/* Problem.  It may or may not be needed for you   */
80#if defined(_MINIX) && !defined(_MINIX_VMD)
81    char kludge[10];
82    extern char *sprintf();
83    sprintf(kludge, CGETS(24, 1, "kludge"));
84#endif /* _MINIX && !_MINIX_VMD */
85
86    v++;
87    cp = *v++;
88    if (cp == NULL) {
89	Char   *fmt;
90	if ((fmt = varval(STRsched)) == STRNULL)
91	    fmt = str2short("%h\t%T\t%R\n");
92	/* print list of scheduled events */
93	for (count = 1, tp = sched_ptr; tp; count++, tp = tp->t_next) {
94	    Char buf[BUFSIZE], sbuf[BUFSIZE];
95	    blkexpand(tp->t_lex, buf);
96	    tprintf(FMT_SCHED, sbuf, fmt, sizeof(sbuf),
97		    short2str(buf), tp->t_when, (ptr_t) &count);
98	    for (cp = sbuf; *cp;)
99		xputchar(*cp++);
100	}
101	return;
102    }
103
104    if (*cp == '-') {
105	/* remove item from list */
106	if (!sched_ptr)
107	    stderror(ERR_NOSCHED);
108	if (*v)
109	    stderror(ERR_SCHEDUSAGE);
110	count = atoi(short2str(++cp));
111	if (count <= 0)
112	    stderror(ERR_SCHEDUSAGE);
113	tp = sched_ptr;
114	tp1 = 0;
115	while (--count) {
116	    if (tp->t_next == 0)
117		break;
118	    else {
119		tp1 = tp;
120		tp = tp->t_next;
121	    }
122	}
123	if (count)
124	    stderror(ERR_SCHEDEV);
125	if (tp1 == 0)
126	    sched_ptr = tp->t_next;
127	else
128	    tp1->t_next = tp->t_next;
129	blkfree(tp->t_lex);
130	xfree((ptr_t) tp);
131	return;
132    }
133
134    /* else, add an item to the list */
135    if (!*v)
136	stderror(ERR_SCHEDCOM);
137    relative = 0;
138    if (!Isdigit(*cp)) {	/* not abs. time */
139	if (*cp != '+')
140	    stderror(ERR_SCHEDUSAGE);
141	cp++, relative++;
142    }
143    minutes = 0;
144    hours = atoi(short2str(cp));
145    while (*cp && *cp != ':' && *cp != 'a' && *cp != 'p')
146	cp++;
147    if (*cp && *cp == ':')
148	minutes = atoi(short2str(++cp));
149    if ((hours < 0) || (minutes < 0) ||
150	(hours > 23) || (minutes > 59))
151	stderror(ERR_SCHEDTIME);
152    while (*cp && *cp != 'p' && *cp != 'a')
153	cp++;
154    if (*cp && relative)
155	stderror(ERR_SCHEDREL);
156    if (*cp == 'p')
157	hours += 12;
158    (void) time(&cur_time);
159    ltp = localtime(&cur_time);
160    if (relative) {
161	dif_hour = hours;
162	dif_min = minutes;
163    }
164    else {
165	if ((dif_hour = hours - ltp->tm_hour) < 0)
166	    dif_hour += 24;
167	if ((dif_min = minutes - ltp->tm_min) < 0) {
168	    dif_min += 60;
169	    if ((--dif_hour) < 0)
170		dif_hour = 23;
171	}
172    }
173    tp = (struct sched_event *) xcalloc(1, sizeof *tp);
174#ifdef _SX
175    tp->t_when = cur_time - ltp->tm_sec + dif_hour * 3600 + dif_min * 60;
176#else	/* _SX */
177    tp->t_when = cur_time - ltp->tm_sec + dif_hour * 3600L + dif_min * 60L;
178#endif /* _SX */
179    /* use of tm_sec: get to beginning of minute. */
180    if (!sched_ptr || tp->t_when < sched_ptr->t_when) {
181	tp->t_next = sched_ptr;
182	sched_ptr = tp;
183    }
184    else {
185	tp1 = sched_ptr->t_next;
186	tp2 = sched_ptr;
187	while (tp1 && tp->t_when >= tp1->t_when) {
188	    tp2 = tp1;
189	    tp1 = tp1->t_next;
190	}
191	tp->t_next = tp1;
192	tp2->t_next = tp;
193    }
194    tp->t_lex = saveblk(v);
195}
196
197/*
198 * Execute scheduled events
199 */
200/*ARGSUSED*/
201void
202sched_run(n)
203    int n;
204{
205    time_t   cur_time;
206    register struct sched_event *tp, *tp1;
207    struct wordent cmd, *nextword, *lastword;
208    struct command *t;
209    Char  **v, *cp;
210    extern Char GettingInput;
211#ifdef BSDSIGS
212    sigmask_t omask;
213
214    omask = sigblock(sigmask(SIGINT)) & ~sigmask(SIGINT);
215#else
216    (void) sighold(SIGINT);
217#endif
218
219    USE(n);
220
221    (void) time(&cur_time);
222    tp = sched_ptr;
223
224    /* bugfix by: Justin Bur at Universite de Montreal */
225    /*
226     * this test wouldn't be necessary if this routine were not called before
227     * each prompt (in sh.c).  But it is, to catch missed alarms.  Someone
228     * ought to fix it all up.  -jbb
229     */
230    if (!(tp && tp->t_when < cur_time)) {
231#ifdef BSDSIGS
232	(void) sigsetmask(omask);
233#else
234	(void) sigrelse(SIGINT);
235#endif
236	return;
237    }
238
239    if (GettingInput)
240	(void) Cookedmode();
241
242    while (tp && tp->t_when < cur_time) {
243	if (seterr) {
244	    xfree((ptr_t) seterr);
245	    seterr = NULL;
246	}
247	cmd.word = STRNULL;
248	lastword = &cmd;
249	v = tp->t_lex;
250	for (cp = *v; cp; cp = *++v) {
251	    nextword = (struct wordent *) xcalloc(1, sizeof cmd);
252	    nextword->word = Strsave(cp);
253	    lastword->next = nextword;
254	    nextword->prev = lastword;
255	    lastword = nextword;
256	}
257	lastword->next = &cmd;
258	cmd.prev = lastword;
259	tp1 = tp;
260	sched_ptr = tp = tp1->t_next;	/* looping termination cond: */
261	blkfree(tp1->t_lex);	/* straighten out in case of */
262	xfree((ptr_t) tp1);	/* command blow-up. */
263
264	/* expand aliases like process() does. */
265	alias(&cmd);
266	/* build a syntax tree for the command. */
267	t = syntax(cmd.next, &cmd, 0);
268	if (seterr)
269	    stderror(ERR_OLD);
270	/* execute the parse tree. */
271	execute(t, -1, NULL, NULL);
272	/* done. free the lex list and parse tree. */
273	freelex(&cmd), freesyn(t);
274    }
275    if (GettingInput && !just_signaled) {	/* PWP */
276	(void) Rawmode();
277	ClearLines();		/* do a real refresh since something may */
278	ClearDisp();		/* have printed to the screen */
279	Refresh();
280    }
281    just_signaled = 0;
282
283#ifdef BSDSIGS
284    (void) sigsetmask(omask);
285#else
286    (void) sigrelse(SIGINT);
287#endif
288}
289