ack/lang/cem/libcc.ansi/misc/popen.c
1989-12-18 14:40:54 +00:00

108 lines
2 KiB
C

/*
* popen - open a pipe
*/
/* $Header$ */
#include <errno.h>
#include <stdio.h>
#include <signal.h>
#if defined(__BSD4_2)
#if _EM_WSIZE != 4
#error union wait is too small
#endif
union wait {
int w_status;
};
typedef union wait wait_arg;
#else
typedef int wait_arg;
#endif /* __BSD4_2 */
#include "../stdio/loc_incl.h"
int close(int d);
int dup2(int oldd, int newd); /* not present in System 5 */
int execl(const char *name, ... );
int fork(void);
int pipe(int fildes[2]);
int wait(wait_arg *status);
void _exit(int status);
static int pids[20];
FILE *
popen(const char *command, const char *type)
{
int piped[2];
int Xtype = *type == 'r' ? 0 : *type == 'w' ? 1 : 2;
int pid;
if (Xtype == 2 ||
pipe(piped) < 0 ||
(pid = fork()) < 0) return 0;
if (pid == 0) {
/* child */
register int *p;
for (p = pids; p < &pids[ FOPEN_MAX]; p++) {
if (*p) close(p - pids);
}
close(piped[Xtype]);
dup2(piped[!Xtype], !Xtype);
close(piped[!Xtype]);
execl("/bin/sh", "sh", "-c", command, (char *) 0);
_exit(127); /* like system() ??? */
}
pids[piped[Xtype]] = pid;
close(piped[!Xtype]);
return fdopen(piped[Xtype], type);
}
#if defined(__BSD4_2)
#define ret_val status.w_status
#else
#define ret_val status
#endif
int
pclose(FILE *stream)
{
int fd = fileno(stream);
wait_arg status;
int wret;
void (*intsave)(int) = signal(SIGINT, SIG_IGN);
void (*quitsave)(int) = signal(SIGQUIT, SIG_IGN);
fclose(stream);
while ((wret = wait(&status)) != -1) {
if (wret == pids[fd]) break;
}
if (wret == -1) ret_val = -1;
signal(SIGINT, intsave);
signal(SIGQUIT, quitsave);
pids[fd] = 0;
return ret_val;
}
#if defined(__USG)
int
dup2(int oldd, int newd)
{
int i = 0, fd, tmp;
int fdbuf[_NFILES];
/* ignore the error on the close() */
tmp = errno; (void) close(newd); errno = tmp;
while ((fd = dup(oldd)) != newd) {
if (fd == -1) break;
fdbuf[i++] = fd;
}
tmp = errno;
while (--i >= 0) {
close(fdbuf[i]);
}
errno = tmp;
return -(fd == -1);
}
#endif /* __USG */