sbase/xargs.c

260 lines
4.1 KiB
C
Raw Normal View History

2014-01-03 11:52:47 +00:00
/* See LICENSE file for copyright and license details. */
2015-02-14 20:02:41 +00:00
#include <sys/wait.h>
#include <errno.h>
2014-01-03 11:52:47 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
2014-01-03 11:52:47 +00:00
#include "util.h"
enum {
NARGS = 10000
2014-01-03 11:52:47 +00:00
};
static int inputc(void);
static void deinputc(int);
2014-01-07 14:56:05 +00:00
static void fillargbuf(int);
static int eatspace(void);
2014-01-03 11:52:47 +00:00
static int parsequote(int);
static int parseescape(void);
2014-01-03 11:52:47 +00:00
static char *poparg(void);
static void waitchld(void);
2014-01-08 20:27:27 +00:00
static void spawn(void);
2014-01-03 11:52:47 +00:00
2014-01-16 11:51:19 +00:00
static char *cmd[NARGS];
2014-01-03 11:52:47 +00:00
static char *argb;
2014-01-08 20:29:34 +00:00
static size_t argbsz;
2014-01-03 11:52:47 +00:00
static size_t argbpos;
static long maxargs = 0;
static int nerrors = 0;
2014-01-07 11:53:55 +00:00
static char *eofstr;
static int rflag = 0, nflag = 0;
2014-01-03 11:52:47 +00:00
static int
inputc(void)
{
int ch;
ch = getc(stdin);
if (ch == EOF && ferror(stdin))
eprintf("stdin: read error:");
return ch;
}
static void
deinputc(int ch)
{
ungetc(ch, stdin);
}
static void
2014-01-07 14:56:05 +00:00
fillargbuf(int ch)
2014-01-03 11:52:47 +00:00
{
if (argbpos >= argbsz) {
2014-01-08 20:29:34 +00:00
argbsz = argbpos == 0 ? 1 : argbsz * 2;
argb = erealloc(argb, argbsz);
2014-01-03 11:52:47 +00:00
}
argb[argbpos] = ch;
}
static int
2014-01-03 11:52:47 +00:00
eatspace(void)
{
int ch;
while ((ch = inputc()) != EOF) {
switch (ch) {
case ' ': case '\t': case '\n':
break;
default:
deinputc(ch);
return ch;
2014-01-03 11:52:47 +00:00
}
}
return -1;
2014-01-03 11:52:47 +00:00
}
static int
parsequote(int q)
{
int ch;
while ((ch = inputc()) != EOF) {
2014-01-08 20:22:10 +00:00
if (ch == q)
2014-01-03 11:52:47 +00:00
return 0;
if (ch != '\n') {
2014-01-07 14:56:05 +00:00
fillargbuf(ch);
2014-01-03 11:52:47 +00:00
argbpos++;
}
}
return -1;
}
static int
2014-01-03 11:52:47 +00:00
parseescape(void)
{
int ch;
if ((ch = inputc()) != EOF) {
2014-01-07 14:56:05 +00:00
fillargbuf(ch);
2014-01-03 11:52:47 +00:00
argbpos++;
return ch;
2014-01-03 11:52:47 +00:00
}
return -1;
2014-01-03 11:52:47 +00:00
}
static char *
poparg(void)
{
int ch;
argbpos = 0;
2014-11-19 19:59:37 +00:00
if (eatspace() < 0)
return NULL;
2014-01-03 11:52:47 +00:00
while ((ch = inputc()) != EOF) {
switch (ch) {
case ' ': case '\t': case '\n':
2014-01-07 11:53:55 +00:00
goto out;
2014-01-03 11:52:47 +00:00
case '\'':
2014-11-19 19:59:37 +00:00
if (parsequote('\'') < 0)
eprintf("unterminated single quote\n");
2014-01-03 11:52:47 +00:00
break;
case '\"':
2014-11-19 19:59:37 +00:00
if (parsequote('\"') < 0)
eprintf("unterminated double quote\n");
2014-01-03 11:52:47 +00:00
break;
case '\\':
2014-11-19 19:59:37 +00:00
if (parseescape() < 0)
eprintf("backslash at EOF\n");
2014-01-03 11:52:47 +00:00
break;
default:
2014-01-07 14:56:05 +00:00
fillargbuf(ch);
2014-01-03 11:52:47 +00:00
argbpos++;
break;
}
}
2014-01-07 11:53:55 +00:00
out:
fillargbuf('\0');
if (eofstr && strcmp(argb, eofstr) == 0)
return NULL;
return argb;
2014-01-03 11:52:47 +00:00
}
static void
waitchld(void)
2014-01-03 11:52:47 +00:00
{
int status;
2014-01-03 11:52:47 +00:00
wait(&status);
if (WIFEXITED(status)) {
if (WEXITSTATUS(status) == 255)
exit(124);
if (WEXITSTATUS(status) == 127 ||
WEXITSTATUS(status) == 126)
exit(WEXITSTATUS(status));
if (status != 0)
nerrors++;
}
if (WIFSIGNALED(status))
exit(125);
2014-01-03 11:52:47 +00:00
}
static void
2014-01-08 20:27:27 +00:00
spawn(void)
{
2014-01-27 15:16:43 +00:00
int savederrno;
switch (fork()) {
case -1:
eprintf("fork:");
case 0:
execvp(*cmd, cmd);
2014-01-27 15:16:43 +00:00
savederrno = errno;
weprintf("execvp %s:", *cmd);
_exit(126 + (savederrno == ENOENT));
}
waitchld();
}
2015-03-07 12:36:40 +00:00
static void
usage(void)
{
eprintf("usage: %s [-n maxargs] [-r] [-E eofstr] [cmd [arg...]]\n", argv0);
}
int
main(int argc, char *argv[])
{
int leftover = 0;
long argsz, argmaxsz;
char *arg = "";
int i, a;
ARGBEGIN {
case 'n':
nflag = 1;
if ((maxargs = strtol(EARGF(usage()), NULL, 10)) <= 0)
eprintf("%s: value for -n option should be >= 1\n", argv0);
break;
case 'r':
rflag = 1;
break;
case 'E':
eofstr = EARGF(usage());
break;
default:
usage();
} ARGEND;
argmaxsz = sysconf(_SC_ARG_MAX);
if (argmaxsz < 0)
eprintf("sysconf:");
/* Leave some room for environment variables */
argmaxsz -= 4 * 1024;
do {
argsz = 0; i = 0; a = 0;
if (argc > 0) {
for (; i < argc; i++) {
cmd[i] = estrdup(argv[i]);
argsz += strlen(cmd[i]) + 1;
}
} else {
cmd[i] = estrdup("/bin/echo");
argsz += strlen(cmd[i]) + 1;
i++;
}
while (leftover == 1 || (arg = poparg())) {
if (argsz + strlen(arg) + 1 > argmaxsz ||
i >= NARGS - 1) {
if (strlen(arg) + 1 > argmaxsz)
eprintf("insufficient argument space\n");
leftover = 1;
break;
}
cmd[i] = estrdup(arg);
argsz += strlen(cmd[i]) + 1;
i++;
a++;
leftover = 0;
if (nflag == 1 && a >= maxargs)
break;
}
cmd[i] = NULL;
if (a >= maxargs && nflag == 1)
spawn();
else if (!a || (i == 1 && rflag == 1))
;
else
spawn();
for (; i >= 0; i--)
free(cmd[i]);
} while (arg);
free(argb);
return nerrors > 0 ? 123 : 0;
}