xref: /illumos-gate/usr/src/cmd/csh/sh.c (revision fe0e7ec4d916b05b52d8c7cc8a3e6a1b28e77b6f)
1 /*
2  * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
3  * Use is subject to license terms.
4  */
5 
6 /*	Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T	*/
7 /*	  All Rights Reserved  	*/
8 
9 /*
10  * Copyright (c) 1980 Regents of the University of California.
11  * All rights reserved.  The Berkeley Software License Agreement
12  * specifies the terms and conditions for redistribution.
13  */
14 
15 #pragma ident	"%Z%%M%	%I%	%E% SMI"
16 
17 #include <locale.h>
18 #include "sh.h"
19 /* #include <sys/ioctl.h> */
20 #include <fcntl.h>
21 #include <sys/filio.h>
22 #include "sh.tconst.h"
23 #include <pwd.h>
24 #include <stdlib.h>
25 #include "sh_policy.h"		/* for pfcsh */
26 #ifdef	TRACE
27 #include <stdio.h>
28 #endif
29 
30 /*
31  * We use these csh(1) private versions of the select macros, (see select(3C))
32  * so as not to be limited by the size of struct fd_set (ie 1024).
33  */
34 #define	CSH_FD_SET(n, p)   ((*((p) + ((n)/NFDBITS))) |= (1 << ((n) % NFDBITS)))
35 #define	CSH_FD_CLR(n, p)   ((*((p) + ((n)/NFDBITS))) &= ~(1 << ((n) % NFDBITS)))
36 #define	CSH_FD_ISSET(n, p) ((*((p) + ((n)/NFDBITS))) & (1 << ((n) % NFDBITS)))
37 #define	CSH_FD_ZERO(p, n)  memset((void *)(p), 0,  (n))
38 
39 tchar *pathlist[] =	{ S_usrbin /* "/usr/bin" */, S_DOT /* "." */, 0 };
40 tchar *dumphist[] =	{ S_history /* "history" */, S_h /* "-h" */, 0, 0 };
41 tchar *loadhist[] =	{ S_source /* "source" */, S_h /* "-h" */, S_NDOThistory /* "~/.history" */, 0 };
42 tchar HIST = '!';
43 tchar HISTSUB = '^';
44 int	nofile;
45 bool	reenter;
46 bool	nverbose;
47 bool	nexececho;
48 bool	quitit;
49 bool	fast;
50 bool	batch;
51 bool	prompt = 1;
52 bool	enterhist = 0;
53 
54 extern	gid_t getegid(), getgid();
55 extern	uid_t geteuid(), getuid();
56 extern tchar **strblktotsblk(/* char **, int */);
57 
58 void	importpath(tchar *);
59 void	srccat(tchar *, tchar *);
60 void	srccat_inlogin(tchar *, tchar *);
61 void	srcunit(int, bool, bool);
62 void	rechist(void);
63 void	goodbye(void);
64 void	pintr1(bool);
65 void	process(bool);
66 void	dosource(tchar **);
67 void	mailchk(void);
68 void	printprompt(void);
69 void	sigwaiting(void);
70 void	siglwp(void);
71 void	initdesc(int, char *[]);
72 void	initdesc_x(int, char *[], int);
73 void	closem(void);
74 void	unsetfd(int);
75 void	secpolicy_print(int, const char *);
76 void	phup(void);
77 
78 #ifdef	TRACE
79 FILE *trace;
80 /*
81  * Trace routines
82  */
83 #define	TRACEFILE	"/tmp/trace.XXXXXX"
84 
85 /*
86  * Initialize trace file.
87  * Called from main.
88  */
89 void
90 trace_init(void)
91 {
92 	char name[128];
93 	char *p;
94 
95 	strcpy(name, TRACEFILE);
96 	p = mktemp(name);
97 	trace = fopen(p, "w");
98 }
99 
100 /*
101  * write message to trace file
102  */
103 /*VARARGS1*/
104 void
105 tprintf(fmt, a, b, c, d, e, f, g, h, i, j)
106 	char *fmt;
107 {
108 	if (trace) {
109 		fprintf(trace, fmt, a, b, c, d, e, f, g, h, i, j);
110 		fflush(trace);
111 	}
112 }
113 #endif
114 
115 int
116 main(int c, char **av)
117 {
118 	tchar **v, *cp, *p, *q, *r;
119 	int f;
120 	struct sigvec osv;
121 	struct sigaction sa;
122 	tchar s_prompt[MAXHOSTNAMELEN+3];
123 
124 	pfcshflag = 0;
125 
126 	/*
127 	 * set up the error exit, if there is an error before
128 	 * this is done, it will core dump, and we don't
129 	 * tolerate core dumps
130 	 */
131 	haderr = 0;
132 	setexit();
133 	if (haderr) {
134 		/*
135 		 *  if were here, there was an error in the csh
136 		 *  startup so just punt
137 		 */
138 		printf("csh startup error, csh exiting...\n");
139 		flush();
140 		exitstat();
141 	}
142 
143 
144 	(void) setlocale(LC_ALL, "");
145 #if !defined(TEXT_DOMAIN)	/* Should be defined by cc -D */
146 #define	TEXT_DOMAIN "SYS_TEST"	/* Use this only if it weren't */
147 #endif
148 	(void) textdomain(TEXT_DOMAIN);
149 
150 	/*
151 	 * This is a profile shell if the simple name of argv[0] is
152 	 * pfcsh or -pfcsh
153 	 */
154 	p = strtots(NOSTR, "pfcsh");
155 	r = strtots(NOSTR, "-pfcsh");
156 	if ((p != NOSTR) && (r != NOSTR) &&
157 	    ((q = strtots(NOSTR, *av)) != NOSTR)) {
158 		if (c > 0 && (eq(p, simple(q)) || eq(r, simple(q)))) {
159 			pfcshflag = 1;
160 		}
161 		xfree(q);
162 	}
163 
164 	if (p != NOSTR)
165 		xfree(p);
166 	if (r != NOSTR)
167 		xfree(r);
168 
169 	if (pfcshflag == 1) {
170 		secpolicy_init();
171 	}
172 
173 	/* Copy arguments */
174 	v = strblktotsblk(av, c);
175 
176 	/*
177 	 * Initialize paraml list
178 	 */
179 	paraml.next = paraml.prev = &paraml;
180 
181 	settimes();			/* Immed. estab. timing base */
182 
183 	if (eq(v[0], S_aout /* "a.out" */))	/* A.out's are quittable */
184 		quitit = 1;
185 	uid = getuid();
186 	loginsh = **v == '-';
187 	if (loginsh)
188 		(void) time(&chktim);
189 
190 	/*
191 	 * Move the descriptors to safe places.
192 	 * The variable didfds is 0 while we have only FSH* to work with.
193 	 * When didfds is true, we have 0,1,2 and prefer to use these.
194 	 *
195 	 * Also, setup data for csh internal file descriptor book keeping.
196 	 */
197 	initdesc(c, av);
198 
199 	/*
200 	 * Initialize the shell variables.
201 	 * ARGV and PROMPT are initialized later.
202 	 * STATUS is also munged in several places.
203 	 * CHILD is munged when forking/waiting
204 	 */
205 
206 	/* don't do globbing here, just set exact copies */
207 	setNS(S_noglob);
208 
209 	set(S_status /* "status" */, S_0 /* "0" */);
210 	dinit(cp = getenvs_("HOME"));	/* dinit thinks that HOME==cwd in a */
211 					/* login shell */
212 	if (cp == NOSTR)
213 		fast++;			/* No home -> can't read scripts */
214 	else {
215 		if (strlen_(cp) >= BUFSIZ - 10) {
216 			cp = NOSTR;
217 			fast++;
218 			printf("%s\n", gettext("Pathname too long"));
219 			set(S_home /* "home" */, savestr(cp));
220 			local_setenv(S_HOME, savestr(cp));
221 		}
222 		set(S_home /* "home" */, savestr(cp));
223 	}
224 	/*
225 	 * Grab other useful things from the environment.
226 	 * Should we grab everything??
227 	 */
228 	if ((cp = getenvs_("USER")) != NOSTR)
229 		set(S_user /* "user" */, savestr(cp));
230 	else {
231 		/*
232 		 * If USER is not defined, set it here.
233 		 */
234 		struct passwd *pw;
235 		pw = getpwuid(getuid());
236 
237 		if (pw != NULL) {
238 			set(S_user, strtots((tchar *)0, pw->pw_name));
239 			local_setenv(S_USER, strtots((tchar *)0, pw->pw_name));
240 		}
241 		else if (loginsh) { /* Give up setting USER variable. */
242 	printf("Warning: USER environment variable could not be set.\n");
243 		}
244 	}
245 	if ((cp = getenvs_("TERM")) != NOSTR)
246 		set(S_term /* "term" */, savestr(cp));
247 	/*
248 	 * Re-initialize path if set in environment
249 	 */
250 	if ((cp = getenvs_("PATH")) == NOSTR)
251 		set1(S_path /* "path" */, saveblk(pathlist), &shvhed);
252 	else
253 		importpath(cp);
254 	set(S_shell /* "shell" */, S_SHELLPATH);
255 
256 	doldol = putn(getpid());		/* For $$ */
257 
258 	/* restore globbing until the user says otherwise */
259 	unsetv(S_noglob);
260 
261 	/*
262 	 * Record the interrupt states from the parent process.
263 	 * If the parent is non-interruptible our hand must be forced
264 	 * or we (and our children) won't be either.
265 	 * Our children inherit termination from our parent.
266 	 * We catch it only if we are the login shell.
267 	 */
268 		/* parents interruptibility */
269 	(void) sigvec(SIGINT, (struct sigvec *)0, &osv);
270 	parintr = osv.sv_handler;
271 		/* parents terminability */
272 	(void) sigvec(SIGTERM, (struct sigvec *)0, &osv);
273 	parterm = osv.sv_handler;
274 
275 	_signal(SIGLWP, siglwp);
276 	_signal(SIGWAITING, sigwaiting);
277 	if (loginsh) {
278 		(void) signal(SIGHUP, phup);	/* exit processing on HUP */
279 		(void) signal(SIGXCPU, phup);	/* ...and on XCPU */
280 		(void) signal(SIGXFSZ, phup);	/* ...and on XFSZ */
281 	}
282 
283 	/*
284 	 * Process the arguments.
285 	 *
286 	 * Note that processing of -v/-x is actually delayed till after
287 	 * script processing.
288 	 */
289 	c--, v++;
290 	while (c > 0 && (cp = v[0])[0] == '-' && *++cp != '\0' && !batch) {
291 		do switch (*cp++) {
292 
293 		case 'b':		/* -b	Next arg is input file */
294 			batch++;
295 			break;
296 
297 		case 'c':		/* -c	Command input from arg */
298 			if (c == 1)
299 				exit(0);
300 			c--, v++;
301 			arginp = v[0];
302 			prompt = 0;
303 			nofile++;
304 			cflg++;
305 			break;
306 
307 		case 'e':		/* -e	Exit on any error */
308 			exiterr++;
309 			break;
310 
311 		case 'f':		/* -f	Fast start */
312 			fast++;
313 			break;
314 
315 		case 'i':		/* -i	Interactive, even if !intty */
316 			intact++;
317 			nofile++;
318 			break;
319 
320 		case 'n':		/* -n	Don't execute */
321 			noexec++;
322 			break;
323 
324 		case 'q':		/* -q	(Undoc'd) ... die on quit */
325 			quitit = 1;
326 			break;
327 
328 		case 's':		/* -s	Read from std input */
329 			nofile++;
330 			break;
331 
332 		case 't':		/* -t	Read one line from input */
333 			onelflg = 2;
334 			prompt = 0;
335 			nofile++;
336 			break;
337 #ifdef TRACE
338 		case 'T':		/* -T 	trace switch on */
339 			trace_init();
340 			break;
341 #endif
342 
343 		case 'v':		/* -v	Echo hist expanded input */
344 			nverbose = 1;			/* ... later */
345 			break;
346 
347 		case 'x':		/* -x	Echo just before execution */
348 			nexececho = 1;			/* ... later */
349 			break;
350 
351 		case 'V':		/* -V	Echo hist expanded input */
352 			setNS(S_verbose /* "verbose" */);	/* NOW! */
353 			break;
354 
355 		case 'X':		/* -X	Echo just before execution */
356 			setNS(S_echo /* "echo" */);		/* NOW! */
357 			break;
358 
359 		} while (*cp);
360 		v++, c--;
361 	}
362 
363 	if (quitit)			/* With all due haste, for debugging */
364 		(void) signal(SIGQUIT, SIG_DFL);
365 
366 	/*
367 	 * Unless prevented by -c, -i, -s, or -t, if there
368 	 * are remaining arguments the first of them is the name
369 	 * of a shell file from which to read commands.
370 	 */
371 	if (!batch && (uid != geteuid() || getgid() != getegid())) {
372 		errno = EACCES;
373 		child++;			/* So this ... */
374 		Perror(S_csh /* "csh" */);	/* ... doesn't return */
375 	}
376 
377 	if (nofile == 0 && c > 0) {
378 		nofile = open_(v[0], 0);
379 		if (nofile < 0) {
380 			child++;		/* So this ... */
381 			Perror(v[0]);		/* ... doesn't return */
382 		}
383 		file = v[0];
384 		SHIN = dmove(nofile, FSHIN);	/* Replace FSHIN */
385 		(void) fcntl(SHIN, F_SETFD, 1);
386 		prompt = 0;
387 		c--, v++;
388 	}
389 
390 	/*
391 	 * Consider input a tty if it really is or we are interactive.
392 	 */
393 	intty = intact || isatty(SHIN);
394 
395 	/*
396 	 * Decide whether we should play with signals or not.
397 	 * If we are explicitly told (via -i, or -) or we are a login
398 	 * shell (arg0 starts with -) or the input and output are both
399 	 * the ttys("csh", or "csh</dev/ttyx>/dev/ttyx")
400 	 * Note that in only the login shell is it likely that parent
401 	 * may have set signals to be ignored
402 	 */
403 	if (loginsh || intact || intty && isatty(SHOUT))
404 		setintr = 1;
405 #ifdef TELL
406 	settell();
407 #endif
408 	/*
409 	 * Save the remaining arguments in argv.
410 	 */
411 	setq(S_argv /* "argv" */, copyblk(v), &shvhed);
412 
413 	/*
414 	 * Set up the prompt.
415 	 */
416 	if (prompt) {
417 		gethostname_(s_prompt, MAXHOSTNAMELEN);
418 		strcat_(s_prompt, uid == 0 ? S_SHARPSP /* "# " */ : S_PERSENTSP /* "% " */);
419 		set(S_prompt /* "prompt" */, s_prompt);
420 	}
421 
422 	/*
423 	 * If we are an interactive shell, then start fiddling
424 	 * with the signals; this is a tricky game.
425 	 */
426 	shpgrp = getpgid(0);
427 	opgrp = tpgrp = -1;
428 	if (setintr) {
429 		**av = '-';
430 		if (!quitit)		/* Wary! */
431 			(void) signal(SIGQUIT, SIG_IGN);
432 		(void) signal(SIGINT, pintr);
433 		(void) sigblock(sigmask(SIGINT));
434 		(void) signal(SIGTERM, SIG_IGN);
435 		if (quitit == 0 && arginp == 0) {
436 			(void) signal(SIGTSTP, SIG_IGN);
437 			(void) signal(SIGTTIN, SIG_IGN);
438 			(void) signal(SIGTTOU, SIG_IGN);
439 			/*
440 			 * Wait till in foreground, in case someone
441 			 * stupidly runs
442 			 *	csh &
443 			 * dont want to try to grab away the tty.
444 			 */
445 			if (isatty(FSHDIAG))
446 				f = FSHDIAG;
447 			else if (isatty(FSHOUT))
448 				f = FSHOUT;
449 			else if (isatty(OLDSTD))
450 				f = OLDSTD;
451 			else
452 				f = -1;
453 retry:
454 			if (ioctl(f, TIOCGPGRP,  (char *)&tpgrp) == 0 &&
455 			    tpgrp != -1) {
456 				if (tpgrp != shpgrp) {
457 					void (*old)() = (void (*)())signal(SIGTTIN, SIG_DFL);
458 					(void) kill(0, SIGTTIN);
459 					(void) signal(SIGTTIN, old);
460 					goto retry;
461 				}
462 				opgrp = shpgrp;
463 				shpgrp = getpid();
464 				tpgrp = shpgrp;
465 				(void) setpgid(0, shpgrp);
466 				(void) ioctl(f, TIOCSPGRP,  (char *)&shpgrp);
467 				(void) fcntl(dcopy(f, FSHTTY), F_SETFD, 1);
468 			} else {
469 notty:
470 printf("Warning: no access to tty; thus no job control in this shell...\n");
471 				tpgrp = -1;
472 			}
473 		}
474 	}
475 	if (setintr == 0 && parintr == SIG_DFL)
476 		setintr++;
477 
478 	/*
479 	 * Set SIGCHLD handler, making sure that reads restart after it runs.
480 	 */
481 	sigemptyset(&sa.sa_mask);
482 	sa.sa_handler = pchild;
483 	sa.sa_flags = SA_RESTART;
484 	(void) sigaction(SIGCHLD, &sa, (struct sigaction *)NULL);
485 
486 	/*
487 	 * Set an exit here in case of an interrupt or error reading
488 	 * the shell start-up scripts.
489 	 */
490 	setexit();
491 	haderr = 0;		/* In case second time through */
492 	if (!fast && reenter == 0) {
493 		reenter++;
494 
495 		/*
496 		 * If this is a login csh, and /etc/.login exists,
497 		 * source /etc/.login first.
498 		 */
499 		if (loginsh) {
500 			tchar tmp_etc[4+1];	/* strlen("/etc")+1 */
501 			tchar tmp_login[7+1];	/* strlen("/.login")+1 */
502 
503 			strtots(tmp_etc, "/etc");
504 			strtots(tmp_login, "/.login");
505 			srccat_inlogin(tmp_etc, tmp_login);
506 		}
507 
508 		/* Will have value("home") here because set fast if don't */
509 		srccat(value(S_home /* "home" */), S_SLADOTcshrc /* "/.cshrc" */);
510 
511 		/* Hash path */
512 		if (!fast && !arginp && !onelflg && !havhash)
513 			dohash(xhash);
514 
515 
516 		/*
517 		 * Reconstruct the history list now, so that it's
518 		 * available from within .login.
519 		 */
520 		dosource(loadhist);
521 		if (loginsh) {
522 			srccat_inlogin(value(S_home /* "home" */), S_SLADOTlogin /* "/.login" */);
523 		}
524 
525 		/*
526 		 * To get cdpath hashing $cdpath must have a
527 		 * value, not $CDPATH.  So if after reading
528 		 * the startup files ( .cshrc ), and
529 		 * user has specified a value for cdpath, then
530 		 * cache $cdpath paths. xhash2 is global array
531 		 * for $cdpath caching.
532 		 */
533 		if (!fast && !arginp && !onelflg && !havhash2)
534 				dohash(xhash2);
535 	}
536 
537 	/*
538 	 * Now are ready for the -v and -x flags
539 	 */
540 	if (nverbose)
541 		setNS(S_verbose /* "verbose" */);
542 	if (nexececho)
543 		setNS(S_echo /* "echo" */);
544 
545 	/*
546 	 * All the rest of the world is inside this call.
547 	 * The argument to process indicates whether it should
548 	 * catch "error unwinds".  Thus if we are a interactive shell
549 	 * our call here will never return by being blown past on an error.
550 	 */
551 	process(setintr);
552 
553 	/*
554 	 * Mop-up.
555 	 */
556 	if (loginsh) {
557 		printf("logout\n");
558 		(void) close(SHIN);	/* No need for unsetfd(). */
559 		child++;
560 		goodbye();
561 	}
562 	rechist();
563 	exitstat();
564 }
565 
566 void
567 untty(void)
568 {
569 
570 	if (tpgrp > 0) {
571 		(void) setpgid(0, opgrp);
572 		(void) ioctl(FSHTTY, TIOCSPGRP,  (char *)&opgrp);
573 	}
574 }
575 
576 void
577 importpath(tchar *cp)
578 {
579 	int i = 0;
580 	tchar *dp;
581 	tchar **pv;
582 	int c;
583 	static tchar dot[2] = {'.', 0};
584 
585 	for (dp = cp; *dp; dp++)
586 		if (*dp == ':')
587 			i++;
588 	/*
589 	 * i+2 where i is the number of colons in the path.
590 	 * There are i+1 directories in the path plus we need
591 	 * room for a zero terminator.
592 	 */
593 	pv =  (tchar **)xcalloc((unsigned)(i + 2), sizeof (tchar **));
594 	dp = cp;
595 	i = 0;
596 	if (*dp)
597 	for (;;) {
598 		if ((c = *dp) == ':' || c == 0) {
599 			*dp = 0;
600 			pv[i++] = savestr(*cp ? cp : dot);
601 			if (c) {
602 				cp = dp + 1;
603 				*dp = ':';
604 			} else
605 				break;
606 		}
607 		dp++;
608 	}
609 	pv[i] = 0;
610 	set1(S_path /* "path" */, pv, &shvhed);
611 }
612 
613 /*
614  * Source to the file which is the catenation of the argument names.
615  */
616 void
617 srccat(tchar *cp, tchar *dp)
618 {
619 	tchar *ep = strspl(cp, dp);
620 	int unit = dmove(open_(ep, 0), -1);
621 
622 	(void) fcntl(unit, F_SETFD, 1);
623 	xfree(ep);
624 #ifdef INGRES
625 	srcunit(unit, 0, 0);
626 #else
627 	srcunit(unit, 1, 0);
628 #endif
629 }
630 
631 /*
632  * Source to the file which is the catenation of the argument names.
633  * 	This one does not check the ownership.
634  */
635 void
636 srccat_inlogin(tchar *cp, tchar *dp)
637 {
638 	tchar *ep = strspl(cp, dp);
639 	int unit = dmove(open_(ep, 0), -1);
640 
641 	(void) fcntl(unit, F_SETFD, 1);
642 	xfree(ep);
643 	srcunit(unit, 0, 0);
644 }
645 
646 /*
647  * Source to a unit.  If onlyown it must be our file or our group or
648  * we don't chance it.	This occurs on ".cshrc"s and the like.
649  */
650 void
651 srcunit(int unit, bool onlyown, bool hflg)
652 {
653 	/* We have to push down a lot of state here */
654 	/* All this could go into a structure */
655 	int oSHIN = -1, oldintty = intty;
656 	struct whyle *oldwhyl = whyles;
657 	tchar *ogointr = gointr, *oarginp = arginp;
658 	tchar *oevalp = evalp, **oevalvec = evalvec;
659 	int oonelflg = onelflg;
660 	bool oenterhist = enterhist;
661 	tchar OHIST = HIST;
662 #ifdef TELL
663 	bool otell = cantell;
664 #endif
665 	struct Bin saveB;
666 
667 	/* The (few) real local variables */
668 	jmp_buf oldexit;
669 	int reenter, omask;
670 
671 	if (unit < 0)
672 		return;
673 	if (didfds)
674 		donefds();
675 	if (onlyown) {
676 		struct stat stb;
677 
678 		if (fstat(unit, &stb) < 0 ||
679 		    (stb.st_uid != uid && stb.st_gid != getgid())) {
680 			(void) close(unit);
681 			unsetfd(unit);
682 			return;
683 		}
684 	}
685 
686 	/*
687 	 * There is a critical section here while we are pushing down the
688 	 * input stream since we have stuff in different structures.
689 	 * If we weren't careful an interrupt could corrupt SHIN's Bin
690 	 * structure and kill the shell.
691 	 *
692 	 * We could avoid the critical region by grouping all the stuff
693 	 * in a single structure and pointing at it to move it all at
694 	 * once.  This is less efficient globally on many variable references
695 	 * however.
696 	 */
697 	getexit(oldexit);
698 	reenter = 0;
699 	if (setintr)
700 		omask = sigblock(sigmask(SIGINT));
701 	setexit();
702 	reenter++;
703 	if (reenter == 1) {
704 		/* Setup the new values of the state stuff saved above */
705 		copy((char *)&saveB, (char *)&B, sizeof saveB);
706 		fbuf =  (tchar **) 0;
707 		fseekp = feobp = fblocks = 0;
708 		oSHIN = SHIN, SHIN = unit, arginp = 0, onelflg = 0;
709 		intty = isatty(SHIN), whyles = 0, gointr = 0;
710 		evalvec = 0; evalp = 0;
711 		enterhist = hflg;
712 		if (enterhist)
713 			HIST = '\0';
714 		/*
715 		 * Now if we are allowing commands to be interrupted,
716 		 * we let ourselves be interrupted.
717 		 */
718 		if (setintr)
719 			(void) sigsetmask(omask);
720 #ifdef TELL
721 		settell();
722 #endif
723 		process(0);		/* 0 -> blow away on errors */
724 	}
725 	if (setintr)
726 		(void) sigsetmask(omask);
727 	if (oSHIN >= 0) {
728 		int i;
729 
730 		/* We made it to the new state... free up its storage */
731 		/* This code could get run twice but xfree doesn't care */
732 		for (i = 0; i < fblocks; i++)
733 			xfree(fbuf[i]);
734 		xfree((char *)fbuf);
735 
736 		/* Reset input arena */
737 		copy((char *)&B, (char *)&saveB, sizeof B);
738 
739 		(void) close(SHIN), SHIN = oSHIN;
740 		unsetfd(SHIN);
741 		arginp = oarginp, onelflg = oonelflg;
742 		evalp = oevalp, evalvec = oevalvec;
743 		intty = oldintty, whyles = oldwhyl, gointr = ogointr;
744 		if (enterhist)
745 			HIST = OHIST;
746 		enterhist = oenterhist;
747 #ifdef TELL
748 		cantell = otell;
749 #endif
750 	}
751 
752 	resexit(oldexit);
753 	/*
754 	 * If process reset() (effectively an unwind) then
755 	 * we must also unwind.
756 	 */
757 	if (reenter >= 2)
758 		error(NULL);
759 }
760 
761 void
762 rechist(void)
763 {
764 	tchar buf[BUFSIZ];
765 	int fp, ftmp, oldidfds;
766 
767 	if (!fast) {
768 		if (value(S_savehist /* "savehist" */)[0] == '\0')
769 			return;
770 		(void) strcpy_(buf, value(S_home /* "home" */));
771 		(void) strcat_(buf, S_SLADOThistory /* "/.history" */);
772 		fp = creat_(buf, 0666);
773 		if (fp == -1)
774 			return;
775 		oldidfds = didfds;
776 		didfds = 0;
777 		ftmp = SHOUT;
778 		SHOUT = fp;
779 		(void) strcpy_(buf, value(S_savehist /* "savehist" */));
780 		dumphist[2] = buf;
781 		dohist(dumphist);
782 		(void) close(fp);
783 		unsetfd(fp);
784 		SHOUT = ftmp;
785 		didfds = oldidfds;
786 	}
787 }
788 
789 void
790 goodbye(void)
791 {
792 	if (loginsh) {
793 		(void) signal(SIGQUIT, SIG_IGN);
794 		(void) signal(SIGINT, SIG_IGN);
795 		(void) signal(SIGTERM, SIG_IGN);
796 		setintr = 0;		/* No interrupts after "logout" */
797 		if (adrof(S_home /* "home" */))
798 			srccat(value(S_home /* "home" */), S_SLADOTlogout /* "/.logout" */);
799 	}
800 	rechist();
801 	exitstat();
802 }
803 
804 void
805 exitstat(void)
806 {
807 
808 #ifdef PROF
809 	monitor(0);
810 #endif
811 	/*
812 	 * Note that if STATUS is corrupted (i.e. getn bombs)
813 	 * then error will exit directly because we poke child here.
814 	 * Otherwise we might continue unwarrantedly (sic).
815 	 */
816 	child++;
817 	untty();
818 	exit(getn(value(S_status /* "status" */)));
819 }
820 
821 /*
822  * in the event of a HUP we want to save the history
823  */
824 void
825 phup(void)
826 {
827 	rechist();
828 	exit(1);
829 }
830 
831 tchar *jobargv[2] = { S_jobs /* "jobs" */, 0 };
832 /*
833  * Catch an interrupt, e.g. during lexical input.
834  * If we are an interactive shell, we reset the interrupt catch
835  * immediately.  In any case we drain the shell output,
836  * and finally go through the normal error mechanism, which
837  * gets a chance to make the shell go away.
838  */
839 void
840 pintr(void)
841 {
842 	pintr1(1);
843 }
844 
845 void
846 pintr1(bool wantnl)
847 {
848 	tchar **v;
849 	int omask;
850 
851 	omask = sigblock(0);
852 	if (setintr) {
853 		(void) sigsetmask(omask & ~sigmask(SIGINT));
854 		if (pjobs) {
855 			pjobs = 0;
856 			printf("\n");
857 			dojobs(jobargv);
858 			bferr("Interrupted");
859 		}
860 	}
861 	(void) sigsetmask(omask & ~sigmask(SIGCHLD));
862 	draino();
863 
864 	/*
865 	 * If we have an active "onintr" then we search for the label.
866 	 * Note that if one does "onintr -" then we shan't be interruptible
867 	 * so we needn't worry about that here.
868 	 */
869 	if (gointr) {
870 		search(ZGOTO, 0, gointr);
871 		timflg = 0;
872 		if (v = pargv)
873 			pargv = 0, blkfree(v);
874 		if (v = gargv)
875 			gargv = 0, blkfree(v);
876 		reset();
877 	} else if (intty && wantnl)
878 		printf("\n");		/* Some like this, others don't */
879 	error(NULL);
880 }
881 
882 /*
883  * Process is the main driving routine for the shell.
884  * It runs all command processing, except for those within { ... }
885  * in expressions (which is run by a routine evalav in sh.exp.c which
886  * is a stripped down process), and `...` evaluation which is run
887  * also by a subset of this code in sh.glob.c in the routine backeval.
888  *
889  * The code here is a little strange because part of it is interruptible
890  * and hence freeing of structures appears to occur when none is necessary
891  * if this is ignored.
892  *
893  * Note that if catch is not set then we will unwind on any error.
894  * If an end-of-file occurs, we return.
895  */
896 void
897 process(bool catch)
898 {
899 	jmp_buf osetexit;
900 	struct command *t;
901 
902 	getexit(osetexit);
903 	for (;;) {
904 		pendjob();
905 		freelex(&paraml);
906 		paraml.next = paraml.prev = &paraml;
907 		paraml.word = S_ /* "" */;
908 		t = 0;
909 		setexit();
910 		justpr = enterhist;	/* execute if not entering history */
911 
912 		/*
913 		 * Interruptible during interactive reads
914 		 */
915 		if (setintr)
916 			(void) sigsetmask(sigblock(0) & ~sigmask(SIGINT));
917 
918 		/*
919 		 * For the sake of reset()
920 		 */
921 		freelex(&paraml), freesyn(t), t = 0;
922 
923 		if (haderr) {
924 			if (!catch) {
925 				/* unwind */
926 				doneinp = 0;
927 				resexit(osetexit);
928 				reset();
929 			}
930 			haderr = 0;
931 			/*
932 			 * Every error is eventually caught here or
933 			 * the shell dies.  It is at this
934 			 * point that we clean up any left-over open
935 			 * files, by closing all but a fixed number
936 			 * of pre-defined files.  Thus routines don't
937 			 * have to worry about leaving files open due
938 			 * to deeper errors... they will get closed here.
939 			 */
940 			closem();
941 			continue;
942 		}
943 		if (doneinp) {
944 			doneinp = 0;
945 			break;
946 		}
947 		if (chkstop)
948 			chkstop--;
949 		if (neednote)
950 			pnote();
951 		if (intty && prompt && evalvec == 0) {
952 			mailchk();
953 			/*
954 			 * If we are at the end of the input buffer
955 			 * then we are going to read fresh stuff.
956 			 * Otherwise, we are rereading input and don't
957 			 * need or want to prompt.
958 			 */
959 			if (fseekp == feobp)
960 				printprompt();
961 		}
962 		err = 0;
963 
964 		/*
965 		 * Echo not only on VERBOSE, but also with history expansion.
966 		 */
967 		if (lex(&paraml) && intty ||
968 		    adrof(S_verbose /* "verbose" */)) {
969 			haderr = 1;
970 			prlex(&paraml);
971 			haderr = 0;
972 		}
973 
974 		/*
975 		 * The parser may lose space if interrupted.
976 		 */
977 		if (setintr)
978 			(void) sigblock(sigmask(SIGINT));
979 
980 		/*
981 		 * Save input text on the history list if
982 		 * reading in old history, or it
983 		 * is from the terminal at the top level and not
984 		 * in a loop.
985 		 */
986 		if (enterhist || catch && intty && !whyles)
987 			savehist(&paraml);
988 
989 		/*
990 		 * Print lexical error messages, except when sourcing
991 		 * history lists.
992 		 */
993 		if (!enterhist && err)
994 			error("%s", gettext(err));
995 
996 		/*
997 		 * If had a history command :p modifier then
998 		 * this is as far as we should go
999 		 */
1000 		if (justpr)
1001 			reset();
1002 
1003 		alias(&paraml);
1004 
1005 		/*
1006 		 * Parse the words of the input into a parse tree.
1007 		 */
1008 		t = syntax(paraml.next, &paraml, 0);
1009 		if (err)
1010 			error("%s", gettext(err));
1011 
1012 		/*
1013 		 * Execute the parse tree
1014 		 */
1015 		{
1016 			/*
1017 			 * POSIX requires SIGCHLD to be held
1018 			 * until all processes have joined the
1019 			 * process group in order to avoid race
1020 			 * condition.
1021 			 */
1022 			int omask;
1023 
1024 			omask = sigblock(sigmask(SIGCHLD));
1025 			execute(t, tpgrp);
1026 			(void) sigsetmask(omask &~ sigmask(SIGCHLD));
1027 		}
1028 
1029 		if (err)
1030 			error("%s", gettext(err));
1031 		/*
1032 		 * Made it!
1033 		 */
1034 		freelex(&paraml), freesyn(t);
1035 	}
1036 	resexit(osetexit);
1037 }
1038 
1039 void
1040 dosource(tchar **t)
1041 {
1042 	tchar *f;
1043 	int u;
1044 	bool hflg = 0;
1045 	tchar buf[BUFSIZ];
1046 
1047 	t++;
1048 	if (*t && eq(*t, S_h /* "-h" */)) {
1049 		if (*++t == NOSTR)
1050 			bferr("Too few arguments.");
1051 		hflg++;
1052 	}
1053 	(void) strcpy_(buf, *t);
1054 	f = globone(buf);
1055 	u = dmove(open_(f, 0), -1);
1056 	xfree(f);
1057 	freelex(&paraml);
1058 	if (u < 0 && !hflg)
1059 		Perror(f);
1060 	(void) fcntl(u, F_SETFD, 1);
1061 	srcunit(u, 0, hflg);
1062 }
1063 
1064 /*
1065  * Check for mail.
1066  * If we are a login shell, then we don't want to tell
1067  * about any mail file unless its been modified
1068  * after the time we started.
1069  * This prevents us from telling the user things he already
1070  * knows, since the login program insists on saying
1071  * "You have mail."
1072  */
1073 void
1074 mailchk(void)
1075 {
1076 	struct varent *v;
1077 	tchar **vp;
1078 	time_t t;
1079 	int intvl, cnt;
1080 	struct stat stb;
1081 	bool new;
1082 
1083 	v = adrof(S_mail /* "mail" */);
1084 	if (v == 0)
1085 		return;
1086 	(void) time(&t);
1087 	vp = v->vec;
1088 	cnt = blklen(vp);
1089 	intvl = (cnt && number(*vp)) ? (--cnt, getn(*vp++)) : MAILINTVL;
1090 	if (intvl < 1)
1091 		intvl = 1;
1092 	if (chktim + intvl > t)
1093 		return;
1094 	for (; *vp; vp++) {
1095 		if (stat_(*vp, &stb) < 0)
1096 			continue;
1097 		new = stb.st_mtime > time0.tv_sec;
1098 		if (stb.st_size == 0 || stb.st_atime >= stb.st_mtime ||
1099 		    (stb.st_atime <= chktim && stb.st_mtime <= chktim) ||
1100 		    loginsh && !new)
1101 			continue;
1102 		if (cnt == 1)
1103 			printf("You have %smail.\n", new ? "new " : "");
1104 		else
1105 			printf("%s in %t.\n", new ? "New mail" : "Mail", *vp);
1106 	}
1107 	chktim = t;
1108 }
1109 
1110 /*
1111  * Extract a home directory from the password file
1112  * The argument points to a buffer where the name of the
1113  * user whose home directory is sought is currently.
1114  * We write the home directory of the user back there.
1115  */
1116 int
1117 gethdir(tchar *home)
1118 {
1119 	/* getpwname will not be modified, so we need temp. buffer */
1120 	char home_str[BUFSIZ];
1121 	tchar home_ts[BUFSIZ];
1122 	struct passwd *pp /* = getpwnam(home) */;
1123 
1124 	pp = getpwnam(tstostr(home_str, home));
1125 	if (pp == 0)
1126 		return (1);
1127 	(void) strcpy_(home, strtots(home_ts, pp->pw_dir));
1128 	return (0);
1129 }
1130 
1131 
1132 #if 0
1133 void
1134 #ifdef PROF
1135 done(int i)
1136 #else
1137 exit(int i)
1138 #endif
1139 {
1140 
1141 	untty();
1142 	_exit(i);
1143 }
1144 #endif
1145 
1146 void
1147 printprompt(void)
1148 {
1149 	tchar *cp;
1150 
1151 	if (!whyles) {
1152 		/*
1153 		 * Print the prompt string
1154 		 */
1155 		for (cp = value(S_prompt /* "prompt" */); *cp; cp++)
1156 			if (*cp == HIST)
1157 				printf("%d", eventno + 1);
1158 			else {
1159 				if (*cp == '\\' && cp[1] == HIST)
1160 					cp++;
1161 				Putchar(*cp | QUOTE);
1162 			}
1163 	} else
1164 		/*
1165 		 * Prompt for forward reading loop
1166 		 * body content.
1167 		 */
1168 		printf("? ");
1169 	flush();
1170 }
1171 
1172 /*
1173  * Save char * block.
1174  */
1175 tchar **
1176 strblktotsblk(char **v, int num)
1177 {
1178 	tchar **newv =
1179 		(tchar **)xcalloc((unsigned)(num+ 1), sizeof (tchar **));
1180 	tchar **onewv = newv;
1181 
1182 	while (*v && num--)
1183 		*newv++ = strtots(NOSTR, *v++);
1184 	*newv = 0;
1185 	return (onewv);
1186 }
1187 
1188 void
1189 sigwaiting(void)
1190 {
1191 	_signal(SIGWAITING, sigwaiting);
1192 }
1193 
1194 void
1195 siglwp(void)
1196 {
1197 	_signal(SIGLWP, siglwp);
1198 }
1199 
1200 
1201 /*
1202  * Following functions and data are used for csh to do its
1203  * file descriptors book keeping.
1204  */
1205 
1206 static int *fdinuse = NULL;	/* The list of files opened by csh */
1207 static int nbytesused = 0;	/* no of bytes allocated to fdinuse */
1208 static int max_fd = 0;		/* The maximum descriptor in fdinuse */
1209 static int my_pid;		/* The process id set in initdesc() */
1210 static int NoFile = NOFILE;	/* The number of files I can use. */
1211 
1212 /*
1213  * Get the number of files this csh can use.
1214  *
1215  * Move the initial descriptors to their eventual
1216  * resting places, closing all other units.
1217  *
1218  * Also, reserve 0/1/2, so NIS+ routines do not get
1219  * hold of them. And initialize fdinuse list and set
1220  * the current process id.
1221  *
1222  * If this csh was invoked from setuid'ed script file,
1223  * do not close the third argument passed. The file
1224  * must be one of /dev/fd/0,1,2,,,
1225  *	(execv() always passes three arguments when it execs a script
1226  *	 file in a form of #! /bin/csh -b.)
1227  *
1228  * If is_reinit is set in initdesc_x(), then we only close the file
1229  * descriptors that we actually opened (as recorded in fdinuse).
1230  */
1231 void
1232 initdesc(int argc, char *argv[])
1233 {
1234 	initdesc_x(argc, argv, 0);
1235 }
1236 
1237 void
1238 reinitdesc(int argc, char *argv[])
1239 {
1240 	initdesc_x(argc, argv, 1);
1241 }
1242 
1243 /*
1244  * Callback functions for closing all file descriptors.
1245  */
1246 static int
1247 close_except(void *cd, int fd)
1248 {
1249 	int script_fd = *(int *)cd;
1250 
1251 	if (fd >= 3 && fd < NoFile && fd != script_fd)
1252 		(void) close(fd);
1253 	return (0);
1254 }
1255 
1256 static int
1257 close_inuse(void *cd, int fd)
1258 {
1259 	int script_fd = *(int *)cd;
1260 
1261 	if (fd >= 3 && fd < NoFile && fd != script_fd &&
1262 	    CSH_FD_ISSET(fd, fdinuse)) {
1263 		(void) close(fd);
1264 		unsetfd(fd);
1265 	}
1266 	return (0);
1267 }
1268 
1269 void
1270 initdesc_x(int argc, char *argv[], int is_reinit)
1271 {
1272 
1273 	int script_fd = -1;
1274 	struct stat buf;
1275 	struct rlimit rlp;
1276 
1277 	/*
1278 	 * Get pid of this shell
1279 	 */
1280 	my_pid = getpid();
1281 
1282 	/*
1283 	 * Get the hard limit numbers of descriptors
1284 	 * this csh can use.
1285 	 */
1286 	if (getrlimit(RLIMIT_NOFILE, &rlp) == 0)
1287 		NoFile = rlp.rlim_cur;
1288 
1289 	/*
1290 	 * If this csh was invoked for executing setuid script file,
1291 	 * the third argument passed is the special file name
1292 	 * which should not be closed.  This special file name is
1293 	 * in the form /dev/fd/X.
1294 	 */
1295 	if (argc >= 3)
1296 	    if (sscanf(argv[2], "/dev/fd/%d", &script_fd) != 1)
1297 		script_fd = -1;
1298 	    else
1299 		fcntl(script_fd, F_SETFD, 1);	/* Make sure to close
1300 						 *  this file on exec.
1301 						 */
1302 
1303 	if (fdinuse == NULL) {
1304 		nbytesused = sizeof (int) * howmany(NoFile, sizeof (int) * NBBY);
1305 		fdinuse = (int *)xalloc(nbytesused);
1306 	}
1307 
1308 	/*
1309 	 * Close all files except 0/1/2 to get a clean
1310 	 * file descritor space.
1311 	 */
1312 	if (!is_reinit)
1313 		(void) fdwalk(close_except, &script_fd);
1314 	else
1315 		(void) fdwalk(close_inuse, &script_fd);
1316 
1317 	didfds = 0;			/* 0, 1, 2 aren't set up */
1318 
1319 	if (fstat(0, &buf) < 0)
1320 		open("/dev/null", 0);
1321 
1322 	(void) fcntl(SHIN = dcopy(0, FSHIN), F_SETFD,  1);
1323 	(void) fcntl(SHOUT = dcopy(1, FSHOUT), F_SETFD,  1);
1324 	(void) fcntl(SHDIAG = dcopy(2, FSHDIAG), F_SETFD,  1);
1325 	(void) fcntl(OLDSTD = dcopy(SHIN, FOLDSTD), F_SETFD,  1);
1326 
1327 	/*
1328 	 * Open 0/1/2 to avoid Nis+ functions to pick them up.
1329 	 *	Now, 0/1/2 are saved, close them and open them.
1330 	 */
1331 	close(0); close(1); close(2);
1332 	open("/dev/null", 0);
1333 	dup(0);
1334 	dup(0);
1335 
1336 	/*
1337 	 * Clear fd_set mask
1338 	 */
1339 	if (!is_reinit)
1340 		CSH_FD_ZERO(fdinuse, nbytesused);
1341 }
1342 
1343 /*
1344  * This routine is called after an error to close up
1345  * any units which may have been left open accidentally.
1346  *
1347  * You only need to remove files in fdinuse list.
1348  * After you have removed the files, you can clear the
1349  * list and max_fd.
1350  */
1351 void
1352 closem(void)
1353 {
1354 	int f;
1355 
1356 	for (f = 3; f <= max_fd; f++) {
1357 		if (CSH_FD_ISSET(f, fdinuse) &&
1358 		    f != SHIN && f != SHOUT && f != SHDIAG &&
1359 		    f != OLDSTD && f != FSHTTY)
1360 			close(f);
1361 	}
1362 	CSH_FD_ZERO(fdinuse, nbytesused);
1363 	max_fd = 0;
1364 }
1365 
1366 /*
1367  * Reset my_pid when a new process is created.  Only call this
1368  * if you want the process to affect fdinuse (e.g., fork, but
1369  * not vfork).
1370  */
1371 void
1372 new_process(void)
1373 {
1374 	my_pid = getpid();
1375 }
1376 
1377 
1378 /*
1379  * Whenever Csh open/create/dup/pipe a file or files,
1380  * Csh keeps track of its open files. The open files
1381  * are kept in "fdinuse, Fd In Use" list.
1382  *
1383  * When a file descriptor is newly allocated, setfd() is
1384  * used to mark the fact in "fdinuse" list.
1385  *	For example,
1386  *		fd = open("newfile", 0);
1387  *		setfd(fd);
1388  *
1389  * When a file is freed by close() function, unsetfd() is
1390  * used to remove the fd from "fdinuse" list.
1391  *	For example,
1392  *		close(fd);
1393  *		unsetfd(fd);
1394  */
1395 void
1396 setfd(int fd)
1397 {
1398 	/*
1399 	 * Because you want to avoid
1400 	 * conflict due to vfork().
1401 	 */
1402 	if (my_pid != getpid())
1403 		return;
1404 
1405 	if (fd >= NoFile || fd < 0)
1406 		return;
1407 
1408 	if (fd > max_fd)
1409 		max_fd = fd;
1410 	CSH_FD_SET(fd, fdinuse);
1411 }
1412 
1413 void
1414 unsetfd(int fd)
1415 {
1416 	int i;
1417 
1418 	/*
1419 	 * Because you want to avoid
1420 	 * conflict due to vfork().
1421 	 */
1422 	if (my_pid != getpid())
1423 		return;
1424 
1425 	if (fd >= NoFile || fd < 0)
1426 		return;
1427 
1428 	CSH_FD_CLR(fd, fdinuse);
1429 	if (fd == max_fd) {
1430 		for (i = max_fd-1; i >= 3; i--)
1431 			if (CSH_FD_ISSET(i, fdinuse)) {
1432 				max_fd = i;
1433 				return;
1434 			}
1435 		max_fd = 0;
1436 	}
1437 }
1438 
1439 /*
1440  * A generic call back routine to output error messages from the
1441  * policy backing functions called by pfcsh.
1442  */
1443 void
1444 secpolicy_print(int level, const char *msg)
1445 {
1446 	switch (level) {
1447 	case SECPOLICY_WARN:
1448 	default:
1449 		haderr = 1;
1450 		printf("%s: ", msg);	/* printf() does gettext() */
1451 		break;
1452 	case SECPOLICY_ERROR:
1453 		bferr((char *)msg);		/* bferr() does gettext() */
1454 		break;
1455 	}
1456 }
1457