415 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			415 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/*-
 | 
						|
 * Copyright (c) 1991 The Regents of the University of California.
 | 
						|
 * All rights reserved.
 | 
						|
 *
 | 
						|
 * This code is derived from software contributed to Berkeley by
 | 
						|
 * Kenneth Almquist.
 | 
						|
 *
 | 
						|
 * Redistribution and use in source and binary forms, with or without
 | 
						|
 * modification, are permitted provided that the following conditions
 | 
						|
 * are met:
 | 
						|
 * 1. Redistributions of source code must retain the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer.
 | 
						|
 * 2. Redistributions in binary form must reproduce the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer in the
 | 
						|
 *    documentation and/or other materials provided with the distribution.
 | 
						|
 * 3. All advertising materials mentioning features or use of this software
 | 
						|
 *    must display the following acknowledgement:
 | 
						|
 *	This product includes software developed by the University of
 | 
						|
 *	California, Berkeley and its contributors.
 | 
						|
 * 4. Neither the name of the University nor the names of its contributors
 | 
						|
 *    may be used to endorse or promote products derived from this software
 | 
						|
 *    without specific prior written permission.
 | 
						|
 *
 | 
						|
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 | 
						|
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
						|
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | 
						|
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 | 
						|
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | 
						|
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | 
						|
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | 
						|
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | 
						|
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | 
						|
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | 
						|
 * SUCH DAMAGE.
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef lint
 | 
						|
static char sccsid[] = "@(#)input.c	5.4 (Berkeley) 7/1/91";
 | 
						|
#endif /* not lint */
 | 
						|
 | 
						|
/*
 | 
						|
 * This file implements the input routines used by the parser.
 | 
						|
 */
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
#include <stdio.h>	/* defines BUFSIZ */
 | 
						|
#include "shell.h"
 | 
						|
#include <fcntl.h>
 | 
						|
#include <errno.h>
 | 
						|
#include "syntax.h"
 | 
						|
#include "input.h"
 | 
						|
#include "output.h"
 | 
						|
#include "memalloc.h"
 | 
						|
#include "error.h"
 | 
						|
 | 
						|
#define EOF_NLEFT -99		/* value of parsenleft when EOF pushed back */
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * The parsefile structure pointed to by the global variable parsefile
 | 
						|
 * contains information about the current file being read.
 | 
						|
 */
 | 
						|
 | 
						|
MKINIT
 | 
						|
struct parsefile {
 | 
						|
	int linno;		/* current line */
 | 
						|
	int fd;			/* file descriptor (or -1 if string) */
 | 
						|
	int nleft;		/* number of chars left in buffer */
 | 
						|
	char *nextc;		/* next char in buffer */
 | 
						|
	struct parsefile *prev;	/* preceding file on stack */
 | 
						|
	char *buf;		/* input buffer */
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
int plinno = 1;			/* input line number */
 | 
						|
MKINIT int parsenleft;		/* copy of parsefile->nleft */
 | 
						|
char *parsenextc;		/* copy of parsefile->nextc */
 | 
						|
MKINIT struct parsefile basepf;	/* top level input file */
 | 
						|
char basebuf[BUFSIZ];		/* buffer for top level input file */
 | 
						|
struct parsefile *parsefile = &basepf;	/* current input file */
 | 
						|
char *pushedstring;		/* copy of parsenextc when text pushed back */
 | 
						|
int pushednleft;		/* copy of parsenleft when text pushed back */
 | 
						|
 | 
						|
#if READLINE
 | 
						|
char *readline __P((const char *prompt));
 | 
						|
char *r_use_prompt = NULL;	/* the prompt to use with readline */
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef __STDC__
 | 
						|
STATIC void pushfile(void);
 | 
						|
#else
 | 
						|
STATIC void pushfile();
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
 | 
						|
#ifdef mkinit
 | 
						|
INCLUDE "input.h"
 | 
						|
INCLUDE "error.h"
 | 
						|
 | 
						|
INIT {
 | 
						|
	extern char basebuf[];
 | 
						|
 | 
						|
	basepf.nextc = basepf.buf = basebuf;
 | 
						|
}
 | 
						|
 | 
						|
RESET {
 | 
						|
	if (exception != EXSHELLPROC)
 | 
						|
		parsenleft = 0;            /* clear input buffer */
 | 
						|
	popallfiles();
 | 
						|
}
 | 
						|
 | 
						|
SHELLPROC {
 | 
						|
	popallfiles();
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Read a line from the script.
 | 
						|
 */
 | 
						|
 | 
						|
char *
 | 
						|
pfgets(line, len)
 | 
						|
	char *line;
 | 
						|
	{
 | 
						|
	register char *p = line;
 | 
						|
	int nleft = len;
 | 
						|
	int c;
 | 
						|
 | 
						|
	while (--nleft > 0) {
 | 
						|
		c = pgetc_macro();
 | 
						|
		if (c == PEOF) {
 | 
						|
			if (p == line)
 | 
						|
				return NULL;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		*p++ = c;
 | 
						|
		if (c == '\n')
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	*p = '\0';
 | 
						|
	return line;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Read a character from the script, returning PEOF on end of file.
 | 
						|
 * Nul characters in the input are silently discarded.
 | 
						|
 */
 | 
						|
 | 
						|
int
 | 
						|
pgetc() {
 | 
						|
	return pgetc_macro();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Refill the input buffer and return the next input character:
 | 
						|
 *
 | 
						|
 * 1) If a string was pushed back on the input, switch back to the regular
 | 
						|
 *    buffer.
 | 
						|
 * 2) If an EOF was pushed back (parsenleft == EOF_NLEFT) or we are reading
 | 
						|
 *    from a string so we can't refill the buffer, return EOF.
 | 
						|
 * 3) Call read to read in the characters.
 | 
						|
 * 4) Delete all nul characters from the buffer.
 | 
						|
 */
 | 
						|
 | 
						|
int
 | 
						|
preadbuffer() {
 | 
						|
	register char *p, *q;
 | 
						|
	register int i;
 | 
						|
 | 
						|
	if (pushedstring) {
 | 
						|
		parsenextc = pushedstring;
 | 
						|
		pushedstring = NULL;
 | 
						|
		parsenleft = pushednleft;
 | 
						|
		if (--parsenleft >= 0)
 | 
						|
			return *parsenextc++;
 | 
						|
	}
 | 
						|
	if (parsenleft == EOF_NLEFT || parsefile->buf == NULL)
 | 
						|
		return PEOF;
 | 
						|
	flushout(&output);
 | 
						|
	flushout(&errout);
 | 
						|
#if READLINE
 | 
						|
    /* Use the readline() call if a prompt is to be printed (interactive). */
 | 
						|
    if (r_use_prompt != NULL) {
 | 
						|
	char *prompt;
 | 
						|
	char *line;
 | 
						|
 | 
						|
	p = parsenextc = parsefile->buf;
 | 
						|
 | 
						|
	prompt = r_use_prompt;
 | 
						|
	r_use_prompt = NULL;
 | 
						|
 | 
						|
	if ((line = readline(prompt)) == NULL) {
 | 
						|
                parsenleft = EOF_NLEFT;
 | 
						|
                return PEOF;
 | 
						|
	}
 | 
						|
	strcpy(p, line);
 | 
						|
	free(line);
 | 
						|
	i = strlen(p);
 | 
						|
	p[i++] = '\n';
 | 
						|
    } else {
 | 
						|
#endif
 | 
						|
retry:
 | 
						|
	p = parsenextc = parsefile->buf;
 | 
						|
	i = read(parsefile->fd, p, BUFSIZ);
 | 
						|
	if (i <= 0) {
 | 
						|
                if (i < 0) {
 | 
						|
                        if (errno == EINTR)
 | 
						|
                                goto retry;
 | 
						|
#ifdef EWOULDBLOCK
 | 
						|
                        if (parsefile->fd == 0 && errno == EWOULDBLOCK) {
 | 
						|
                                int flags = fcntl(0, F_GETFL, 0);
 | 
						|
                                if (flags >= 0 && flags & O_NONBLOCK) {
 | 
						|
                                        flags &=~ O_NONBLOCK;
 | 
						|
                                        if (fcntl(0, F_SETFL, flags) >= 0) {
 | 
						|
						out2str("sh: turning off NDELAY mode\n");
 | 
						|
                                                goto retry;
 | 
						|
                                        }
 | 
						|
                                }
 | 
						|
                        }
 | 
						|
#endif
 | 
						|
                }
 | 
						|
                parsenleft = EOF_NLEFT;
 | 
						|
                return PEOF;
 | 
						|
	}
 | 
						|
#if READLINE
 | 
						|
    }
 | 
						|
#endif
 | 
						|
	parsenleft = i - 1;
 | 
						|
 | 
						|
	/* delete nul characters */
 | 
						|
	for (;;) {
 | 
						|
		if (*p++ == '\0')
 | 
						|
			break;
 | 
						|
		if (--i <= 0)
 | 
						|
			return *parsenextc++;		/* no nul characters */
 | 
						|
	}
 | 
						|
	q = p - 1;
 | 
						|
	while (--i > 0) {
 | 
						|
		if (*p != '\0')
 | 
						|
			*q++ = *p;
 | 
						|
		p++;
 | 
						|
	}
 | 
						|
	if (q == parsefile->buf)
 | 
						|
		goto retry;			/* buffer contained nothing but nuls */
 | 
						|
	parsenleft = q - parsefile->buf - 1;
 | 
						|
	return *parsenextc++;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Undo the last call to pgetc.  Only one character may be pushed back.
 | 
						|
 * PEOF may be pushed back.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
pungetc() {
 | 
						|
	parsenleft++;
 | 
						|
	parsenextc--;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Push a string back onto the input.  This code doesn't work if the user
 | 
						|
 * tries to push back more than one string at once.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
ppushback(string, length)
 | 
						|
	char *string;
 | 
						|
	{
 | 
						|
	pushedstring = parsenextc;
 | 
						|
	pushednleft = parsenleft;
 | 
						|
	parsenextc = string;
 | 
						|
	parsenleft = length;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Set the input to take input from a file.  If push is set, push the
 | 
						|
 * old input onto the stack first.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
setinputfile(fname, push)
 | 
						|
	char *fname;
 | 
						|
	{
 | 
						|
	int fd;
 | 
						|
	int fd2;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
	if ((fd = open(fname, O_RDONLY)) < 0)
 | 
						|
		error("Can't open %s", fname);
 | 
						|
	if (fd < 10) {
 | 
						|
		fd2 = copyfd(fd, 10);
 | 
						|
		close(fd);
 | 
						|
		if (fd2 < 0)
 | 
						|
			error("Out of file descriptors");
 | 
						|
		fd = fd2;
 | 
						|
	}
 | 
						|
	setinputfd(fd, push);
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Like setinputfile, but takes an open file descriptor.  Call this with
 | 
						|
 * interrupts off.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
setinputfd(fd, push) {
 | 
						|
	(void) fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC);
 | 
						|
	if (push) {
 | 
						|
		pushfile();
 | 
						|
		parsefile->buf = ckmalloc(BUFSIZ);
 | 
						|
	}
 | 
						|
	if (parsefile->fd > 0)
 | 
						|
		close(parsefile->fd);
 | 
						|
	parsefile->fd = fd;
 | 
						|
	if (parsefile->buf == NULL)
 | 
						|
		parsefile->buf = ckmalloc(BUFSIZ);
 | 
						|
	parsenleft = 0;
 | 
						|
	plinno = 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Like setinputfile, but takes input from a string.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
setinputstring(string, push)
 | 
						|
	char *string;
 | 
						|
	{
 | 
						|
	INTOFF;
 | 
						|
	if (push)
 | 
						|
		pushfile();
 | 
						|
	parsenextc = string;
 | 
						|
	parsenleft = strlen(string);
 | 
						|
	parsefile->buf = NULL;
 | 
						|
	plinno = 1;
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * To handle the "." command, a stack of input files is used.  Pushfile
 | 
						|
 * adds a new entry to the stack and popfile restores the previous level.
 | 
						|
 */
 | 
						|
 | 
						|
STATIC void
 | 
						|
pushfile() {
 | 
						|
	struct parsefile *pf;
 | 
						|
 | 
						|
	parsefile->nleft = parsenleft;
 | 
						|
	parsefile->nextc = parsenextc;
 | 
						|
	parsefile->linno = plinno;
 | 
						|
	pf = (struct parsefile *)ckmalloc(sizeof (struct parsefile));
 | 
						|
	pf->prev = parsefile;
 | 
						|
	pf->fd = -1;
 | 
						|
	parsefile = pf;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void
 | 
						|
popfile() {
 | 
						|
	struct parsefile *pf = parsefile;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
	if (pf->fd >= 0)
 | 
						|
		close(pf->fd);
 | 
						|
	if (pf->buf)
 | 
						|
		ckfree(pf->buf);
 | 
						|
	parsefile = pf->prev;
 | 
						|
	ckfree(pf);
 | 
						|
	parsenleft = parsefile->nleft;
 | 
						|
	parsenextc = parsefile->nextc;
 | 
						|
	plinno = parsefile->linno;
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Return to top level.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
popallfiles() {
 | 
						|
	while (parsefile != &basepf)
 | 
						|
		popfile();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Close the file(s) that the shell is reading commands from.  Called
 | 
						|
 * after a fork is done.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
closescript() {
 | 
						|
	popallfiles();
 | 
						|
	if (parsefile->fd > 0) {
 | 
						|
		close(parsefile->fd);
 | 
						|
		parsefile->fd = 0;
 | 
						|
	}
 | 
						|
}
 |