diff options
Diffstat (limited to 'libc/stdio')
| -rw-r--r-- | libc/stdio/popen.c | 70 | 
1 files changed, 45 insertions, 25 deletions
| diff --git a/libc/stdio/popen.c b/libc/stdio/popen.c index 7a00e570f..bc91bb73b 100644 --- a/libc/stdio/popen.c +++ b/libc/stdio/popen.c @@ -1,46 +1,66 @@ +/* + * Modified     3/03/2001       Manuel Novoa III + * + * Added check for legal mode arg. + * Call fdopen and check return value before forking. + * Reduced code size by using variables pr and pnr instead of array refs. + */ +  #include <stdio.h>  #include <unistd.h>  #include <sys/types.h>  #include <sys/wait.h> +#include <errno.h> - -FILE *popen (const char *command, const char *modes) +FILE *popen (const char *command, const char *mode)  { +	FILE *fp;  	int pipe_fd[2];  	int pid, reading; +	int pr, pnr; -	if (pipe(pipe_fd) < 0) -		return NULL; -	reading = (modes[0] == 'r'); - -	pid = vfork(); -	if (pid < 0) { -		close(pipe_fd[0]); -		close(pipe_fd[1]); -		return NULL; -	} -	if (pid == 0) { -		close(pipe_fd[!reading]); -		close(reading); -		if (pipe_fd[reading] != reading) { -			dup2(pipe_fd[reading], reading); -			close(pipe_fd[reading]); +	reading = (mode[0] == 'r'); +	if ((!reading && (mode[0] != 'w')) || mode[1]) { +		errno = EINVAL;			/* Invalid mode arg. */ +	} else if (pipe(pipe_fd) == 0) { +		pr = pipe_fd[reading]; +		pnr = pipe_fd[1-reading]; +		if ((fp = fdopen(pnr, mode)) != NULL) { +			if ((pid = vfork()) == 0) {	/* vfork -- child */ +				close(pnr); +				close(reading); +				if (pr != reading) { +					dup2(pr, reading); +					close(pr); +				} +				execl("/bin/sh", "sh", "-c", command, (char *) 0); +				_exit(255);		/* execl failed! */ +			} else {			/* vfork -- parent or failed */ +				close(pr); +				if (pid > 0) {	/* vfork -- parent */ +					return fp; +				} else {		/* vfork -- failed! */ +					fclose(fp); +				} +			} +		} else {				/* fdopen failed */ +			close(pr); +			close(pnr);  		} - -		execl("/bin/sh", "sh", "-c", command, (char *) 0); -		_exit(255);  	} - -	close(pipe_fd[reading]); -	return fdopen(pipe_fd[!reading], modes); +	return NULL;  }  int pclose(FILE *fd)  {  	int waitstat; -	if (fclose(fd) != 0) +	if (fclose(fd) != 0) {  		return EOF; +	}  	wait(&waitstat);  	return waitstat;  } + + + | 
