/*- * Copyright (c) 1990 The Regents of the University of California. * All rights reserved. * * This code is derived from software contributed to Berkeley by * John B. Roll Jr. * * 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. * 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 * SUCH DAMAGE. */ #ifndef lint char copyright[] = "@(#) Copyright (c) 1990 The Regents of the University of California.\n\ All rights reserved.\n"; #endif /* not lint */ #ifndef lint static char sccsid[] = "@(#)xargs.c 5.11 (Berkeley) 6/19/91"; #endif /* not lint */ #include #include #include #include #include #include #include #include #include #include #if __minix #define _PATH_ECHO "/bin/echo" #else #include "pathnames.h" #endif #ifndef ARG_MAX #define ARG_MAX (sizeof(int) == 2 ? 4096 : 128 * 1024) #endif int exit_status = 0; int tflag; void err(const char *, ...); void run(char **argv); void usage(void); int main(int argc, char **argv) { extern int optind; extern char *optarg; register int ch; register char *p, *bbp, *ebp, **bxp, **exp, **xp; int cnt, indouble, insingle, nargs, nflag, nline, xflag, zflag; char **av, *argp; /* * POSIX.2 limits the exec line length to ARG_MAX - 2K. Running that * caused some E2BIG errors, so it was changed to ARG_MAX - 4K. Given * that the smallest argument is 2 bytes in length, this means that * the number of arguments is limited to: * * (ARG_MAX - 4K - LENGTH(utility + arguments)) / 2. * * We arbitrarily limit the number of arguments to 5000. This is * allowed by POSIX.2 as long as the resulting minimum exec line is * at least LINE_MAX. Realloc'ing as necessary is possible, but * probably not worthwhile. */ #if !__minix || __minix_vmd nargs = 5000; nline = ARG_MAX - 4 * 1024; #else /* Things are more cramped under standard Minix. */ nargs = 80 * sizeof(int); nline = ARG_MAX - 512 * sizeof(int); #endif nflag = xflag = zflag = 0; while ((ch = getopt(argc, argv, "n:s:tx0")) != EOF) switch(ch) { case 'n': nflag = 1; if ((nargs = atoi(optarg)) <= 0) err("illegal argument count"); break; case 's': nline = atoi(optarg); break; case 't': tflag = 1; break; case 'x': xflag = 1; break; case '0': zflag = 1; break; case '?': default: usage(); } argc -= optind; argv += optind; if (xflag && !nflag) usage(); /* * Allocate pointers for the utility name, the utility arguments, * the maximum arguments to be read from stdin and the trailing * NULL. */ if (!(av = bxp = malloc((u_int)(1 + argc + nargs + 1) * sizeof(char **)))) err("%s", strerror(errno)); /* * Use the user's name for the utility as argv[0], just like the * shell. Echo is the default. Set up pointers for the user's * arguments. */ if (!*argv) cnt = strlen(*bxp++ = _PATH_ECHO); else { cnt = 0; do { cnt += strlen(*bxp++ = *argv) + 1; } while (*++argv); } /* * Set up begin/end/traversing pointers into the array. The -n * count doesn't include the trailing NULL pointer, so the malloc * added in an extra slot. */ exp = (xp = bxp) + nargs; /* * Allocate buffer space for the arguments read from stdin and the * trailing NULL. Buffer space is defined as the default or specified * space, minus the length of the utility name and arguments. Set up * begin/end/traversing pointers into the array. The -s count does * include the trailing NULL, so the malloc didn't add in an extra * slot. */ nline -= cnt; if (nline <= 0) err("insufficient space for command"); if (!(bbp = malloc((u_int)nline + 1))) err("%s", strerror(errno)); ebp = (argp = p = bbp) + nline - 1; if (zflag) { /* Read pathnames terminated by null bytes as produced by * find ... -print0. No comments in this code, see further * below. */ for (;;) switch(ch = getchar()) { case EOF: if (p == bbp) exit(exit_status); if (argp == p) { *xp = NULL; run(av); exit(exit_status); } /*FALL THROUGH*/ case '\0': if (argp == p) continue; *p = '\0'; *xp++ = argp; if (xp == exp || p == ebp || ch == EOF) { if (xflag && xp != exp && p == ebp) err( "insufficient space for arguments"); *xp = NULL; run(av); if (ch == EOF) exit(exit_status); p = bbp; xp = bxp; } else ++p; argp = p; break; default: if (p < ebp) { *p++ = ch; break; } if (bxp == xp) err("insufficient space for argument"); if (xflag) err("insufficient space for arguments"); *xp = NULL; run(av); xp = bxp; cnt = ebp - argp; bcopy(argp, bbp, cnt); p = (argp = bbp) + cnt; *p++ = ch; break; } /* NOTREACHED */ } for (insingle = indouble = 0;;) switch(ch = getchar()) { case EOF: /* No arguments since last exec. */ if (p == bbp) exit(exit_status); /* Nothing since end of last argument. */ if (argp == p) { *xp = NULL; run(av); exit(exit_status); } goto arg1; case ' ': case '\t': /* Quotes escape tabs and spaces. */ if (insingle || indouble) goto addch; goto arg2; case '\n': /* Empty lines are skipped. */ if (argp == p) continue; /* Quotes do not escape newlines. */ arg1: if (insingle || indouble) err("unterminated quote"); arg2: *p = '\0'; *xp++ = argp; /* * If max'd out on args or buffer, or reached EOF, * run the command. If xflag and max'd out on buffer * but not on args, object. */ if (xp == exp || p == ebp || ch == EOF) { if (xflag && xp != exp && p == ebp) err("insufficient space for arguments"); *xp = NULL; run(av); if (ch == EOF) exit(exit_status); p = bbp; xp = bxp; } else ++p; argp = p; break; case '\'': if (indouble) goto addch; insingle = !insingle; break; case '"': if (insingle) goto addch; indouble = !indouble; break; case '\\': /* Backslash escapes anything, is escaped by quotes. */ if (!insingle && !indouble && (ch = getchar()) == EOF) err("backslash at EOF"); /* FALLTHROUGH */ default: addch: if (p < ebp) { *p++ = ch; break; } /* If only one argument, not enough buffer space. */ if (bxp == xp) err("insufficient space for argument"); /* Didn't hit argument limit, so if xflag object. */ if (xflag) err("insufficient space for arguments"); *xp = NULL; run(av); xp = bxp; cnt = ebp - argp; bcopy(argp, bbp, cnt); p = (argp = bbp) + cnt; *p++ = ch; break; } /* NOTREACHED */ } void run(char **argv) { register char **p; pid_t pid; int noinvoke; int status; int pfd[2]; if (tflag) { (void)fprintf(stderr, "%s", *argv); for (p = argv + 1; *p; ++p) (void)fprintf(stderr, " %s", *p); (void)fprintf(stderr, "\n"); (void)fflush(stderr); } if (pipe(pfd) < 0) err("pipe: %s", strerror(errno)); switch(pid = fork()) { case -1: err("fork: %s", strerror(errno)); case 0: close(pfd[0]); fcntl(pfd[1], F_SETFD, fcntl(pfd[1], F_GETFD) | FD_CLOEXEC); execvp(argv[0], argv); noinvoke = (errno == ENOENT) ? 127 : 126; (void)fprintf(stderr, "xargs: %s exec failed: %s.\n", argv[0], strerror(errno)); /* Modern way of returning noinvoke instead of a dirty vfork() * trick: (kjb) */ write(pfd[1], &noinvoke, sizeof(noinvoke)); _exit(-1); } close(pfd[1]); if (read(pfd[0], &noinvoke, sizeof(noinvoke)) < sizeof(noinvoke)) noinvoke = 0; close(pfd[0]); pid = waitpid(pid, &status, 0); if (pid == -1) err("waitpid: %s", strerror(errno)); /* * If we couldn't invoke the utility or the utility didn't exit * properly, quit with 127 or 126 respectively. */ if (noinvoke) exit(noinvoke); /* * According to POSIX, we have to exit if the utility exits with * a 255 status, or is interrupted by a signal. xargs is allowed * to return any exit status between 1 and 125 in these cases, but * we'll use 124 and 125, the same values used by GNU xargs. */ if (WIFEXITED(status)) { if (WEXITSTATUS (status) == 255) { fprintf (stderr, "xargs: %s exited with status 255\n", argv[0]); exit(124); } else if (WEXITSTATUS (status) != 0) { exit_status = 123; } } else if (WIFSTOPPED (status)) { fprintf (stderr, "xargs: %s terminated by signal %d\n", argv[0], WSTOPSIG (status)); exit(125); } else if (WIFSIGNALED (status)) { fprintf (stderr, "xargs: %s terminated by signal %d\n", argv[0], WTERMSIG (status)); exit(125); } } void usage(void) { (void)fprintf(stderr, "usage: xargs [-t0] [[-x] -n number] [-s size] [utility [argument ...]]\n"); exit(1); } void err(const char *fmt, ...) { va_list ap; va_start(ap, fmt); (void)fprintf(stderr, "xargs: "); (void)vfprintf(stderr, fmt, ap); va_end(ap); (void)fprintf(stderr, "\n"); exit(1); /* NOTREACHED */ }