579 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			579 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*-
 | 
						|
 * Copyright (c) 1991, 1993
 | 
						|
 *	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.
 | 
						|
 * 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
 | 
						|
#if 0
 | 
						|
static char sccsid[] = "@(#)input.c	8.3 (Berkeley) 6/9/95";
 | 
						|
#endif
 | 
						|
#endif /* not lint */
 | 
						|
#include <sys/cdefs.h>
 | 
						|
/*
 | 
						|
__FBSDID("$FreeBSD: src/bin/sh/input.c,v 1.22 2004/04/06 20:06:51 markm Exp $");
 | 
						|
*/
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
#include <stdio.h>	/* defines BUFSIZ */
 | 
						|
#include <fcntl.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
 | 
						|
/*
 | 
						|
 * This file implements the input routines used by the parser.
 | 
						|
 */
 | 
						|
 | 
						|
#include "shell.h"
 | 
						|
#include "redir.h"
 | 
						|
#include "syntax.h"
 | 
						|
#include "input.h"
 | 
						|
#include "output.h"
 | 
						|
#include "options.h"
 | 
						|
#include "memalloc.h"
 | 
						|
#include "error.h"
 | 
						|
#include "alias.h"
 | 
						|
#include "parser.h"
 | 
						|
#ifdef EDITLINE
 | 
						|
#ifdef __minix_vmd
 | 
						|
#include <readline/readline.h>
 | 
						|
#else
 | 
						|
/* What about other systems? */
 | 
						|
char *readline(char *prompt);
 | 
						|
#endif
 | 
						|
#else
 | 
						|
#include "myhistedit.h"
 | 
						|
#endif
 | 
						|
#include "redir.h"
 | 
						|
#include "trap.h"
 | 
						|
 | 
						|
static void popstring(void);
 | 
						|
 | 
						|
#define EOF_NLEFT -99		/* value of parsenleft when EOF pushed back */
 | 
						|
 | 
						|
MKINIT
 | 
						|
struct strpush {
 | 
						|
	struct strpush *prev;	/* preceding string on stack */
 | 
						|
	char *prevstring;
 | 
						|
	int prevnleft;
 | 
						|
	int prevlleft;
 | 
						|
	struct alias *ap;	/* if push was associated with an alias */
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * The parsefile structure pointed to by the global variable parsefile
 | 
						|
 * contains information about the current file being read.
 | 
						|
 */
 | 
						|
 | 
						|
MKINIT
 | 
						|
struct parsefile {
 | 
						|
	struct parsefile *prev;	/* preceding file on stack */
 | 
						|
	int linno;		/* current line */
 | 
						|
	int fd;			/* file descriptor (or -1 if string) */
 | 
						|
	int nleft;		/* number of chars left in this line */
 | 
						|
	int lleft;		/* number of lines left in this buffer */
 | 
						|
	char *nextc;		/* next char in buffer */
 | 
						|
	char *buf;		/* input buffer */
 | 
						|
	struct strpush *strpush; /* for pushing strings at this level */
 | 
						|
	struct strpush basestrpush; /* so pushing one is fast */
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
int plinno = 1;			/* input line number */
 | 
						|
MKINIT int parsenleft;		/* copy of parsefile->nleft */
 | 
						|
MKINIT int parselleft;		/* copy of parsefile->lleft */
 | 
						|
char *parsenextc;		/* copy of parsefile->nextc */
 | 
						|
MKINIT struct parsefile basepf;	/* top level input file */
 | 
						|
char basebuf[BUFSIZ];		/* buffer for top level input file */
 | 
						|
STATIC struct parsefile *parsefile = &basepf;	/* current input file */
 | 
						|
int init_editline = 0;		/* editline library initialized? */
 | 
						|
int whichprompt;		/* -1 == PSE, 1 == PS1, 2 == PS2 */
 | 
						|
 | 
						|
#ifndef EDITLINE
 | 
						|
EditLine *el;			/* cookie for editline package */
 | 
						|
#endif
 | 
						|
 | 
						|
STATIC void pushfile(void);
 | 
						|
static int preadfd(void);
 | 
						|
 | 
						|
#ifdef mkinit
 | 
						|
INCLUDE "input.h"
 | 
						|
INCLUDE "error.h"
 | 
						|
 | 
						|
INIT {
 | 
						|
	extern char basebuf[];
 | 
						|
 | 
						|
	basepf.nextc = basepf.buf = basebuf;
 | 
						|
}
 | 
						|
 | 
						|
RESET {
 | 
						|
	if (exception != EXSHELLPROC)
 | 
						|
		parselleft = parsenleft = 0;	/* clear input buffer */
 | 
						|
	popallfiles();
 | 
						|
}
 | 
						|
 | 
						|
SHELLPROC {
 | 
						|
	popallfiles();
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Read a line from the script.
 | 
						|
 */
 | 
						|
 | 
						|
char *
 | 
						|
pfgets(char *line, int len)
 | 
						|
{
 | 
						|
	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(void)
 | 
						|
{
 | 
						|
	return pgetc_macro();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static int
 | 
						|
preadfd(void)
 | 
						|
{
 | 
						|
	int nr;
 | 
						|
	parsenextc = parsefile->buf;
 | 
						|
 | 
						|
#if !defined(NO_HISTORY) && !defined(EDITLINE)
 | 
						|
	if (el != NULL && gotwinch) {
 | 
						|
		gotwinch = 0;
 | 
						|
		el_resize(el);
 | 
						|
	}
 | 
						|
#endif
 | 
						|
retry:
 | 
						|
#ifndef NO_HISTORY
 | 
						|
#ifdef EDITLINE
 | 
						|
	if (parsefile->fd == 0 && editable) {
 | 
						|
		static const char *rl_cp= NULL;
 | 
						|
		static size_t rl_off= 0;
 | 
						|
 | 
						|
		if (!rl_cp)
 | 
						|
		{
 | 
						|
			rl_cp = readline(getprompt(NULL));
 | 
						|
			if (rl_cp == NULL)
 | 
						|
				nr = 0;
 | 
						|
		}
 | 
						|
		if (rl_cp)
 | 
						|
		{
 | 
						|
			nr= strlen(rl_cp+rl_off);
 | 
						|
			if (nr >= BUFSIZ-1)
 | 
						|
			{
 | 
						|
				nr= BUFSIZ-1;
 | 
						|
				(void) memcpy(parsenextc, rl_cp+rl_off, nr);
 | 
						|
				rl_off += nr;
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				(void) memcpy(parsenextc, rl_cp+rl_off, nr);
 | 
						|
				parsenextc[nr++]= '\n';
 | 
						|
				free(rl_cp);
 | 
						|
				rl_cp= NULL;
 | 
						|
				rl_off= 0;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	} else
 | 
						|
#else /* !EDITLINE */
 | 
						|
	if (parsefile->fd == 0 && el) {
 | 
						|
		const char *rl_cp;
 | 
						|
 | 
						|
		rl_cp = el_gets(el, &nr);
 | 
						|
		if (rl_cp == NULL)
 | 
						|
			nr = 0;
 | 
						|
		else {
 | 
						|
			/* XXX - BUFSIZE should redesign so not necessary */
 | 
						|
			(void) strcpy(parsenextc, rl_cp);
 | 
						|
		}
 | 
						|
	} else
 | 
						|
#endif /* !EDITLINE */
 | 
						|
#endif
 | 
						|
		nr = read(parsefile->fd, parsenextc, BUFSIZ - 1);
 | 
						|
 | 
						|
	if (nr <= 0) {
 | 
						|
                if (nr < 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 /* EWOULDBLOCK */
 | 
						|
                }
 | 
						|
                nr = -1;
 | 
						|
	}
 | 
						|
	return nr;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Refill the input buffer and return the next input character:
 | 
						|
 *
 | 
						|
 * 1) If a string was pushed back on the input, pop it;
 | 
						|
 * 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) If there is more in this buffer, use it else call read to fill it.
 | 
						|
 * 4) Process input up to the next newline, deleting nul characters.
 | 
						|
 */
 | 
						|
 | 
						|
int
 | 
						|
preadbuffer(void)
 | 
						|
{
 | 
						|
	char *p, *q;
 | 
						|
	int more;
 | 
						|
	int something;
 | 
						|
	char savec;
 | 
						|
 | 
						|
	if (parsefile->strpush) {
 | 
						|
		popstring();
 | 
						|
		if (--parsenleft >= 0)
 | 
						|
			return (*parsenextc++);
 | 
						|
	}
 | 
						|
	if (parsenleft == EOF_NLEFT || parsefile->buf == NULL)
 | 
						|
		return PEOF;
 | 
						|
	flushout(&output);
 | 
						|
	flushout(&errout);
 | 
						|
 | 
						|
again:
 | 
						|
	if (parselleft <= 0) {
 | 
						|
		if ((parselleft = preadfd()) == -1) {
 | 
						|
			parselleft = parsenleft = EOF_NLEFT;
 | 
						|
			return PEOF;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	q = p = parsenextc;
 | 
						|
 | 
						|
	/* delete nul characters */
 | 
						|
	something = 0;
 | 
						|
	for (more = 1; more;) {
 | 
						|
		switch (*p) {
 | 
						|
		case '\0':
 | 
						|
			p++;	/* Skip nul */
 | 
						|
			goto check;
 | 
						|
 | 
						|
		case '\t':
 | 
						|
		case ' ':
 | 
						|
			break;
 | 
						|
 | 
						|
		case '\n':
 | 
						|
			parsenleft = q - parsenextc;
 | 
						|
			more = 0; /* Stop processing here */
 | 
						|
			break;
 | 
						|
 | 
						|
		default:
 | 
						|
			something = 1;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		*q++ = *p++;
 | 
						|
check:
 | 
						|
		if (--parselleft <= 0) {
 | 
						|
			parsenleft = q - parsenextc - 1;
 | 
						|
			if (parsenleft < 0)
 | 
						|
				goto again;
 | 
						|
			*q = '\0';
 | 
						|
			more = 0;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	savec = *q;
 | 
						|
	*q = '\0';
 | 
						|
 | 
						|
#if !defined(NO_HISTORY) && !defined(EDITLINE)
 | 
						|
	if (parsefile->fd == 0 && hist && something) {
 | 
						|
		HistEvent he;
 | 
						|
		INTOFF;
 | 
						|
		history(hist, &he, whichprompt == 1 ? H_ENTER : H_ADD,
 | 
						|
		    parsenextc);
 | 
						|
		INTON;
 | 
						|
	}
 | 
						|
#endif
 | 
						|
 | 
						|
	if (vflag) {
 | 
						|
		out2str(parsenextc);
 | 
						|
		flushout(out2);
 | 
						|
	}
 | 
						|
 | 
						|
	*q = savec;
 | 
						|
 | 
						|
	return *parsenextc++;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Undo the last call to pgetc.  Only one character may be pushed back.
 | 
						|
 * PEOF may be pushed back.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
pungetc(void)
 | 
						|
{
 | 
						|
	parsenleft++;
 | 
						|
	parsenextc--;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Push a string back onto the input at this current parsefile level.
 | 
						|
 * We handle aliases this way.
 | 
						|
 */
 | 
						|
void
 | 
						|
pushstring(char *s, int len, void *ap)
 | 
						|
{
 | 
						|
	struct strpush *sp;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
/*dprintf("*** calling pushstring: %s, %d\n", s, len);*/
 | 
						|
	if (parsefile->strpush) {
 | 
						|
		sp = ckmalloc(sizeof (struct strpush));
 | 
						|
		sp->prev = parsefile->strpush;
 | 
						|
		parsefile->strpush = sp;
 | 
						|
	} else
 | 
						|
		sp = parsefile->strpush = &(parsefile->basestrpush);
 | 
						|
	sp->prevstring = parsenextc;
 | 
						|
	sp->prevnleft = parsenleft;
 | 
						|
	sp->prevlleft = parselleft;
 | 
						|
	sp->ap = (struct alias *)ap;
 | 
						|
	if (ap)
 | 
						|
		((struct alias *)ap)->flag |= ALIASINUSE;
 | 
						|
	parsenextc = s;
 | 
						|
	parsenleft = len;
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
popstring(void)
 | 
						|
{
 | 
						|
	struct strpush *sp = parsefile->strpush;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
	parsenextc = sp->prevstring;
 | 
						|
	parsenleft = sp->prevnleft;
 | 
						|
	parselleft = sp->prevlleft;
 | 
						|
/*dprintf("*** calling popstring: restoring to '%s'\n", parsenextc);*/
 | 
						|
	if (sp->ap)
 | 
						|
		sp->ap->flag &= ~ALIASINUSE;
 | 
						|
	parsefile->strpush = sp->prev;
 | 
						|
	if (sp != &(parsefile->basestrpush))
 | 
						|
		ckfree(sp);
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Set the input to take input from a file.  If push is set, push the
 | 
						|
 * old input onto the stack first.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
setinputfile(char *fname, int push)
 | 
						|
{
 | 
						|
	int fd;
 | 
						|
	int fd2;
 | 
						|
	struct stat statbuf;
 | 
						|
	int saved_errno;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
	if ((fd = open(fname, O_RDONLY)) < 0)
 | 
						|
		error("Can't open %s: %s", fname, strerror(errno));
 | 
						|
	if (fstat(fd, &statbuf) < 0) {
 | 
						|
		saved_errno = errno;
 | 
						|
		close(fd);
 | 
						|
		error("Can't stat %s: %s", fname, strerror(saved_errno));
 | 
						|
	}
 | 
						|
	if (!S_ISREG(statbuf.st_mode)) {
 | 
						|
		close(fd);
 | 
						|
		error("Can't open %s: %s", fname, strerror(ENOEXEC));
 | 
						|
	}
 | 
						|
	if (fd < 10) {
 | 
						|
		fd2 = fcntl(fd, F_DUPFD, 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(int fd, int push)
 | 
						|
{
 | 
						|
	(void)fcntl(fd, F_SETFD, 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);
 | 
						|
	parselleft = parsenleft = 0;
 | 
						|
	plinno = 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Like setinputfile, but takes input from a string.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
setinputstring(char *string, int push)
 | 
						|
{
 | 
						|
	INTOFF;
 | 
						|
	if (push)
 | 
						|
		pushfile();
 | 
						|
	parsenextc = string;
 | 
						|
	parselleft = 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(void)
 | 
						|
{
 | 
						|
	struct parsefile *pf;
 | 
						|
 | 
						|
	parsefile->nleft = parsenleft;
 | 
						|
	parsefile->lleft = parselleft;
 | 
						|
	parsefile->nextc = parsenextc;
 | 
						|
	parsefile->linno = plinno;
 | 
						|
	pf = (struct parsefile *)ckmalloc(sizeof (struct parsefile));
 | 
						|
	pf->prev = parsefile;
 | 
						|
	pf->fd = -1;
 | 
						|
	pf->strpush = NULL;
 | 
						|
	pf->basestrpush.prev = NULL;
 | 
						|
	parsefile = pf;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void
 | 
						|
popfile(void)
 | 
						|
{
 | 
						|
	struct parsefile *pf = parsefile;
 | 
						|
 | 
						|
	INTOFF;
 | 
						|
	if (pf->fd >= 0)
 | 
						|
		close(pf->fd);
 | 
						|
	if (pf->buf)
 | 
						|
		ckfree(pf->buf);
 | 
						|
	while (pf->strpush)
 | 
						|
		popstring();
 | 
						|
	parsefile = pf->prev;
 | 
						|
	ckfree(pf);
 | 
						|
	parsenleft = parsefile->nleft;
 | 
						|
	parselleft = parsefile->lleft;
 | 
						|
	parsenextc = parsefile->nextc;
 | 
						|
	plinno = parsefile->linno;
 | 
						|
	INTON;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Return to top level.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
popallfiles(void)
 | 
						|
{
 | 
						|
	while (parsefile != &basepf)
 | 
						|
		popfile();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Close the file(s) that the shell is reading commands from.  Called
 | 
						|
 * after a fork is done.
 | 
						|
 */
 | 
						|
 | 
						|
void
 | 
						|
closescript(void)
 | 
						|
{
 | 
						|
	popallfiles();
 | 
						|
	if (parsefile->fd > 0) {
 | 
						|
		close(parsefile->fd);
 | 
						|
		parsefile->fd = 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * $PchId: input.c,v 1.7 2006/05/29 13:09:38 philip Exp $
 | 
						|
 */
 |