Annotation of 43BSDReno/usr.bin/xargs/xargs.c, revision 1.1

1.1     ! root        1: /*-
        !             2:  * Copyright (c) 1990 The Regents of the University of California.
        !             3:  * All rights reserved.
        !             4:  *
        !             5:  * This code is derived from software contributed to Berkeley by
        !             6:  * John B. Roll Jr.
        !             7:  *
        !             8:  * Redistribution and use in source and binary forms are permitted provided
        !             9:  * that: (1) source distributions retain this entire copyright notice and
        !            10:  * comment, and (2) distributions including binaries display the following
        !            11:  * acknowledgement:  ``This product includes software developed by the
        !            12:  * University of California, Berkeley and its contributors'' in the
        !            13:  * documentation or other materials provided with the distribution and in
        !            14:  * all advertising materials mentioning features or use of this software.
        !            15:  * Neither the name of the University nor the names of its contributors may
        !            16:  * be used to endorse or promote products derived from this software without
        !            17:  * specific prior written permission.
        !            18:  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
        !            19:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
        !            20:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
        !            21:  */
        !            22: 
        !            23: #ifndef lint
        !            24: char copyright[] =
        !            25: "@(#) Copyright (c) 1990 The Regents of the University of California.\n\
        !            26:  All rights reserved.\n";
        !            27: #endif /* not lint */
        !            28: 
        !            29: #ifndef lint
        !            30: static char sccsid[] = "@(#)xargs.c    5.4 (Berkeley) 6/24/90";
        !            31: #endif /* not lint */
        !            32: 
        !            33: #include <sys/types.h>
        !            34: #include <sys/wait.h>
        !            35: #include <errno.h>
        !            36: #include <stdio.h>
        !            37: #include <string.h>
        !            38: #include <limits.h>
        !            39: #include "pathnames.h"
        !            40: 
        !            41: #define        DEF_ARGC        255
        !            42: 
        !            43: int tflag;
        !            44: 
        !            45: main(argc, argv)
        !            46:        int argc;
        !            47:        char **argv;
        !            48: {
        !            49:        extern int errno, optind;
        !            50:        extern char *optarg;
        !            51:        register int ch;
        !            52:        register char *p, *bp, *endbp, **bxp, **endxp, **xp;
        !            53:        int cnt, indouble, insingle, nargs, nline;
        !            54:        char *mark, *prog, **xargs, *malloc();
        !            55: 
        !            56:        nargs = DEF_ARGC;
        !            57:        nline = _BSD_LINE_MAX;
        !            58: 
        !            59:        while ((ch = getopt(argc, argv, "n:s:t")) != EOF)
        !            60:                switch(ch) {
        !            61:                case 'n':
        !            62:                        if ((nargs = atoi(optarg)) <= 0) {
        !            63:                                (void)fprintf(stderr,
        !            64:                                    "xargs: bad argument count.\n");
        !            65:                                exit(1);
        !            66:                        }
        !            67:                        break;
        !            68:                case 's':
        !            69:                        if ((nline = atoi(optarg)) <= 0) {
        !            70:                                (void)fprintf(stderr,
        !            71:                                    "xargs: bad command length.\n");
        !            72:                                exit(1);
        !            73:                        }
        !            74:                        break;
        !            75:                case 't':
        !            76:                        tflag = 1;
        !            77:                        break;
        !            78:                case '?':
        !            79:                default:
        !            80:                        usage();
        !            81:        }
        !            82:        argc -= optind;
        !            83:        argv += optind;
        !            84: 
        !            85:        /* room for the command, leftover arguments and trailing NULL */
        !            86:        if (!(xargs =
        !            87:            (char **)malloc((u_int)(nargs + argc + 2) * sizeof(char **))))
        !            88:                enomem();
        !            89: 
        !            90:        if (!(bp = malloc((u_int)nline + 1)))
        !            91:                enomem();
        !            92: 
        !            93:        xp = xargs + 1;
        !            94:        if (!*argv)
        !            95:                prog = _PATH_ECHO;
        !            96:        else {
        !            97:                prog = *argv;
        !            98:                while (*++argv)
        !            99:                        *xp++ = *argv;
        !           100:        }
        !           101: 
        !           102:        if (xargs[0] = rindex(prog, '/'))
        !           103:                ++xargs[0];
        !           104:        else
        !           105:                xargs[0] = prog;
        !           106: 
        !           107:        /* set up the pointers into the buffer and the arguments */
        !           108:        *(endxp = (bxp = xp) + nargs) = NULL;
        !           109:        endbp = (mark = p = bp) + nline;
        !           110: 
        !           111:        insingle = indouble = 0;
        !           112:        for (;;)
        !           113:                switch(ch = getchar()) {
        !           114:                case EOF:
        !           115:                        if (p == bp)            /* nothing to display */
        !           116:                                exit(0);
        !           117:                        if (mark == p) {        /* nothing since last arg end */
        !           118:                                run(prog, xargs);
        !           119:                                exit(0);
        !           120:                        }
        !           121:                        goto addarg;
        !           122:                case ' ':
        !           123:                case '\t':
        !           124:                        if (insingle || indouble)
        !           125:                                goto addch;
        !           126:                        goto addarg;
        !           127:                case '\n':
        !           128:                        if (mark == p)                  /* empty line */
        !           129:                                continue;
        !           130: addarg:                        *xp++ = mark;
        !           131:                        *p++ = '\0';
        !           132:                        if (xp == endxp || p >= endbp || ch == EOF) {
        !           133:                                if (insingle || indouble) {
        !           134:                                        (void)fprintf(stderr,
        !           135:                                           "xargs: unterminated quote.\n");
        !           136:                                        exit(1);
        !           137:                                }
        !           138:                                run(prog, xargs);
        !           139:                                if (ch == EOF)
        !           140:                                        exit(0);
        !           141:                                p = bp;
        !           142:                                xp = bxp;
        !           143:                        }
        !           144:                        mark = p;
        !           145:                        break;
        !           146:                case '\'':
        !           147:                        if (indouble)
        !           148:                                goto addch;
        !           149:                        insingle = !insingle;
        !           150:                        break;
        !           151:                case '"':
        !           152:                        if (insingle)
        !           153:                                goto addch;
        !           154:                        indouble = !indouble;
        !           155:                        break;
        !           156:                case '\\':
        !           157:                        if ((ch = getchar()) == EOF)
        !           158:                                ch = '\\';
        !           159:                        if (ch == '\n') {
        !           160:                                (void)fprintf(stderr,
        !           161:                                    "xargs: newline may not be escaped.\n");
        !           162:                                exit(1);
        !           163:                        }
        !           164:                        /* FALLTHROUGH */
        !           165:                default:
        !           166: addch:                 if (p != endbp) {
        !           167:                                *p++ = ch;
        !           168:                                continue;
        !           169:                        }
        !           170:                        if (xp == bxp) {
        !           171:                                (void)fprintf(stderr,
        !           172:                                    "xargs: argument too large.\n");
        !           173:                                exit(1);
        !           174:                        }
        !           175:                        *xp = NULL;
        !           176:                        run(prog, xargs);
        !           177:                        cnt = endbp - mark;
        !           178:                        bcopy(mark, bp, cnt);
        !           179:                        p = (mark = bp) + cnt;
        !           180:                        *p++ = ch;
        !           181:                        xp = bxp;
        !           182:                        break;
        !           183:                }
        !           184:        /* NOTREACHED */
        !           185: }
        !           186: 
        !           187: run(prog, argv)
        !           188:        char *prog, **argv;
        !           189: {
        !           190:        union wait pstat;
        !           191:        pid_t pid, waitpid();
        !           192:        char **p;
        !           193: 
        !           194:        if (tflag) {
        !           195:                (void)fprintf(stderr, "%s", *argv);
        !           196:                for (p = argv + 1; *p; ++p)
        !           197:                        (void)fprintf(stderr, " %s", *p);
        !           198:                (void)fprintf(stderr, "\n");
        !           199:                (void)fflush(stderr);
        !           200:        }
        !           201:        switch(pid = vfork()) {
        !           202:        case -1:
        !           203:                (void)fprintf(stderr,
        !           204:                   "xargs: vfork: %s.\n", strerror(errno));
        !           205:                exit(1);
        !           206:        case 0:
        !           207:                execvp(prog, argv);
        !           208:                (void)fprintf(stderr,
        !           209:                   "xargs: %s: %s.\n", prog, strerror(errno));
        !           210:                _exit(1);
        !           211:        }
        !           212:        pid = waitpid(pid, &pstat, 0);
        !           213:        if (pid == -1) {
        !           214:                (void)fprintf(stderr,
        !           215:                   "xargs: waitpid: %s.\n", strerror(errno));
        !           216:                exit(1);
        !           217:        }
        !           218:        if (pstat.w_status)
        !           219:                exit(1);
        !           220: }
        !           221: 
        !           222: enomem()
        !           223: {
        !           224:        (void)fprintf(stderr, "xargs: %s.\n", strerror(ENOMEM));
        !           225:        exit(1);
        !           226: }
        !           227: 
        !           228: usage()
        !           229: {
        !           230:        (void)fprintf(stderr,
        !           231:            "xargs: [-t] [-n number] [-s size] [utility [argument ...]]\n");
        !           232:        exit(1);
        !           233: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.