NetBSD/usr.sbin/sup/source/run.c
2009-03-04 18:22:14 +00:00

298 lines
7.1 KiB
C

/* $NetBSD: run.c,v 1.14 2009/03/04 18:22:14 christos Exp $ */
/*
* Copyright (c) 1991 Carnegie Mellon University
* All Rights Reserved.
*
* Permission to use, copy, modify and distribute this software and its
* documentation is hereby granted, provided that both the copyright
* notice and this permission notice appear in all copies of the
* software, derivative works or modified versions, and any portions
* thereof, and that both notices appear in supporting documentation.
*
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
*
* Carnegie Mellon requests users of this software to return to
*
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
* School of Computer Science
* Carnegie Mellon University
* Pittsburgh PA 15213-3890
*
* any improvements or extensions that they make and grant Carnegie the rights
* to redistribute these changes.
*/
/* run, runv, runp, runvp -- execute process and wait for it to exit
*
* Usage:
* i = run (file, arg1, arg2, ..., argn, 0);
* i = runv (file, arglist);
* i = runp (file, arg1, arg2, ..., argn, 0);
* i = runvp (file, arglist);
* i = runio (argv, in, out, err);
*
* Run, runv, runp and runvp have argument lists exactly like the
* corresponding routines, execl, execv, execlp, execvp. The run
* routines perform a fork, then:
* IN THE NEW PROCESS, an execl[p] or execv[p] is performed with the
* specified arguments. The process returns with a -1 code if the
* exec was not successful.
* IN THE PARENT PROCESS, the signals SIGQUIT and SIGINT are disabled,
* the process waits until the newly forked process exits, the
* signals are restored to their original status, and the return
* status of the process is analyzed.
* All run routines return: -1 if the exec failed or if the child was
* terminated abnormally; otherwise, the exit code of the child is
* returned.
*
**********************************************************************
* HISTORY
* Revision 1.1 89/10/14 19:53:39 rvb
* Initial revision
*
* Revision 1.2 89/08/03 14:36:46 mja
* Update run() and runp() to use <varargs.h>.
* [89/04/19 mja]
*
* 23-Sep-86 Glenn Marcy (gm0w) at Carnegie-Mellon University
* Merged old runv and runvp modules.
*
* 22-Nov-85 Glenn Marcy (gm0w) at Carnegie-Mellon University
* Added check and kill if child process was stopped.
*
* 30-Apr-85 Steven Shafer (sas) at Carnegie-Mellon University
* Adapted for 4.2 BSD UNIX: Conforms to new signals and wait.
*
* 15-July-82 Mike Accetta (mja) and Neal Friedman (naf)
* at Carnegie-Mellon University
* Added a return(-1) if vfork fails. This should only happen
* if there are no more processes available.
*
* 28-Jan-80 Steven Shafer (sas) at Carnegie-Mellon University
* Added setuid and setgid for system programs' use.
*
* 21-Jan-80 Steven Shafer (sas) at Carnegie-Mellon University
* Changed fork to vfork.
*
* 20-Nov-79 Steven Shafer (sas) at Carnegie-Mellon University
* Created for VAX. The proper way to fork-and-execute a system
* program is now by "runvp" or "runp", with the program name
* (rather than an absolute pathname) as the first argument;
* that way, the "PATH" variable in the environment does the right
* thing. Too bad execvp and execlp (hence runvp and runp) don't
* accept a pathlist as an explicit argument.
*
**********************************************************************
*/
#include <stdio.h>
#include <signal.h>
#include <fcntl.h>
#include <sys/wait.h>
#include "supcdefs.h"
#include "supextern.h"
static int dorun(char *, char **, int);
static char **makearglist(va_list);
static char **
makearglist(va_list ap)
{
static size_t ns = 0;
static char **np = NULL;
char **nnp;
size_t i = 0;
do {
if (i >= ns) {
nnp = realloc(np, (ns + 20) * sizeof(*np));
if (nnp == NULL) {
free(np);
return NULL;
}
np = nnp;
ns += 20;
}
np[i] = va_arg(ap, char *);
}
while (np[i++] != NULL);
return np;
}
int
run(char *name, ...)
{
int val;
va_list ap;
char **argv;
va_start(ap, name);
if ((argv = makearglist(ap)) == NULL) {
va_end(ap);
return -1;
}
val = runv(name, argv);
va_end(ap);
return (val);
}
int
runv(char *name, char **argv)
{
return (dorun(name, argv, 0));
}
int
runp(char *name, ...)
{
int val;
va_list ap;
char **argv;
va_start(ap, name);
if ((argv = makearglist(ap)) == NULL) {
va_end(ap);
return -1;
}
val = runvp(name, argv);
va_end(ap);
return (val);
}
int
runvp(char *name, char **argv)
{
return (dorun(name, argv, 1));
}
static int
dorun(char *name, char **argv, int usepath)
{
int wpid;
int pid;
struct sigaction ignoresig, intsig, quitsig;
int status;
if ((pid = vfork()) == -1)
return (-1); /* no more processes, so exit with error */
if (pid == 0) { /* child process */
setgid(getgid());
setuid(getuid());
if (usepath)
execvp(name, argv);
else
execv(name, argv);
fprintf(stderr, "run: can't exec %s (%s)\n", name,
strerror(errno));
_exit(0377);
}
ignoresig.sa_handler = SIG_IGN; /* ignore INT and QUIT signals */
sigemptyset(&ignoresig.sa_mask);
ignoresig.sa_flags = 0;
sigaction(SIGINT, &ignoresig, &intsig);
sigaction(SIGQUIT, &ignoresig, &quitsig);
do {
wpid = wait3(&status, WUNTRACED, 0);
if (WIFSTOPPED(status)) {
kill(0, SIGTSTP);
wpid = 0;
}
} while (wpid != pid && wpid != -1);
sigaction(SIGINT, &intsig, 0); /* restore signals */
sigaction(SIGQUIT, &quitsig, 0);
if (WIFSIGNALED(status) || WEXITSTATUS(status) == 0377)
return (-1);
return (WEXITSTATUS(status));
}
/*
* Like system(3), but with an argument list and explicit redirections
* that does not use the shell
*/
int
runio(char *const * argv, const char *infile, const char *outfile,
const char *errfile)
{
int fd;
pid_t pid;
int status;
switch ((pid = fork())) {
case -1:
return -1;
case 0:
if (infile) {
(void) close(0);
if ((fd = open(infile, O_RDONLY)) == -1)
exit(1);
if (fd != 0)
(void) dup2(fd, 0);
}
if (outfile) {
(void) close(1);
if ((fd = open(outfile, O_RDWR | O_CREAT | O_TRUNC,
0666)) == -1)
exit(1);
if (fd != 1)
(void) dup2(fd, 1);
}
if (errfile) {
(void) close(2);
if ((fd = open(errfile, O_RDWR | O_CREAT | O_TRUNC,
0666)) == -1)
exit(1);
if (fd != 2)
(void) dup2(fd, 2);
}
execvp(argv[0], argv);
exit(1);
/* NOTREACHED */
return 0;
default:
if (waitpid(pid, &status, 0) == -1)
return -1;
return status;
}
}
/*
* Like runio, but works with filedescriptors instead of filenames
*/
int
runiofd(char *const * argv, const int infile, const int outfile,
const int errfile)
{
pid_t pid;
int status;
switch ((pid = fork())) {
case -1:
return -1;
case 0:
if (infile != 0)
dup2(infile, 0);
if (outfile != 1)
dup2(outfile, 1);
if (errfile != 2)
dup2(errfile, 2);
execvp(argv[0], argv);
exit(1);
/* NOTREACHED */
return 0;
default:
if (waitpid(pid, &status, 0) == -1)
return -1;
return status;
}
}