* Copyright (c) 1991 The Regents of the University of California.
* This code is derived from software contributed to Berkeley by
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* 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
/*static char sccsid[] = "from: @(#)input.c 5.4 (Berkeley) 7/1/91";*/
static char rcsid
[] = "input.c,v 1.4 1993/08/01 18:58:15 mycroft Exp";
* This file implements the input routines used by the parser.
#include <stdio.h> /* defines BUFSIZ */
#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.
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 */
STATIC
void pushfile(void);
basepf
.nextc
= basepf
.buf
= basebuf
;
if (exception
!= EXSHELLPROC
)
parsenleft
= 0; /* clear input buffer */
* Read a line from the script.
* Read a character from the script, returning PEOF on end of file.
* Nul characters in the input are silently discarded.
* 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
* 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.
parsenextc
= pushedstring
;
parsenleft
= pushednleft
;
if (parsenleft
== EOF_NLEFT
|| parsefile
->buf
== NULL
)
p
= parsenextc
= parsefile
->buf
;
i
= read(parsefile
->fd
, p
, BUFSIZ
);
if (parsefile
->fd
== 0 && errno
== EWOULDBLOCK
) {
int flags
= fcntl(0, F_GETFL
, 0);
if (flags
>= 0 && flags
& O_NONBLOCK
) {
if (fcntl(0, F_SETFL
, flags
) >= 0) {
out2str("sh: turning off NDELAY mode\n");
/* delete nul characters */
return *parsenextc
++; /* no nul characters */
goto retry
; /* buffer contained nothing but nuls */
parsenleft
= q
- parsefile
->buf
- 1;
* Undo the last call to pgetc. Only one character may be pushed back.
* PEOF may be pushed back.
* 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.
ppushback(string
, length
)
pushedstring
= parsenextc
;
pushednleft
= parsenleft
;
* Set the input to take input from a file. If push is set, push the
* old input onto the stack first.
setinputfile(fname
, push
)
if ((fd
= open(fname
, O_RDONLY
)) < 0)
error("Can't open %s", fname
);
error("Out of file descriptors");
* Like setinputfile, but takes an open file descriptor. Call this with
parsefile
->buf
= ckmalloc(BUFSIZ
);
if (parsefile
->buf
== NULL
)
parsefile
->buf
= ckmalloc(BUFSIZ
);
* Like setinputfile, but takes input from a string.
setinputstring(string
, push
)
parsenleft
= strlen(string
);
* 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.
parsefile
->nleft
= parsenleft
;
parsefile
->nextc
= parsenextc
;
parsefile
->linno
= plinno
;
pf
= (struct parsefile
*)ckmalloc(sizeof (struct parsefile
));
struct parsefile
*pf
= parsefile
;
parsenleft
= parsefile
->nleft
;
parsenextc
= parsefile
->nextc
;
plinno
= parsefile
->linno
;
while (parsefile
!= &basepf
)
* Close the file(s) that the shell is reading commands from. Called