blob: 21bfb6bbee5ca6e0b3da66118cdebeb36801b533 [file] [log] [blame]
Eric Andersena37d5b72000-09-23 06:10:14 +00001/* vi: set sw=4 ts=4: */
Eric Andersen5b176932000-09-22 20:22:28 +00002/*
Eric Andersena37d5b72000-09-23 06:10:14 +00003 * Mini xargs implementation for busybox
Eric Andersen5b176932000-09-22 20:22:28 +00004 *
Eric Andersena37d5b72000-09-23 06:10:14 +00005 * Copyright (C) 2000 by Lineo, inc.
6 * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org>
Eric Andersen5b176932000-09-22 20:22:28 +00007 *
Eric Andersena37d5b72000-09-23 06:10:14 +00008 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
Eric Andersen5b176932000-09-22 20:22:28 +000012 *
Eric Andersena37d5b72000-09-23 06:10:14 +000013 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
Eric Andersen5b176932000-09-22 20:22:28 +000017 *
Eric Andersena37d5b72000-09-23 06:10:14 +000018 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 *
Eric Andersen5b176932000-09-22 20:22:28 +000022 */
Eric Andersen92a61c12000-09-22 20:01:23 +000023
Eric Andersena37d5b72000-09-23 06:10:14 +000024#include "internal.h"
Eric Andersen5b176932000-09-22 20:22:28 +000025#include <stdlib.h>
Eric Andersen5b176932000-09-22 20:22:28 +000026#include <stdio.h>
Eric Andersena37d5b72000-09-23 06:10:14 +000027#include <string.h>
28#include <errno.h>
Eric Andersen92a61c12000-09-22 20:01:23 +000029#include <getopt.h>
Eric Andersen6f283c22000-09-24 02:40:56 +000030#include <ctype.h>
Eric Andersen92a61c12000-09-22 20:01:23 +000031
Eric Andersenbf739092000-09-25 18:41:18 +000032/* get_sh_safe_line_from_file() - This function reads an entire line from a text file
33 * up to a newline. It returns a malloc'ed char * which must be stored and
34 * free'ed by the caller. */
35extern char *get_sh_safe_line_from_file(FILE *file)
36{
37 static const int GROWBY = 80; /* how large we will grow strings by */
38
39 char ch, last_ch = 0;
40 char tmp[]=" ";
41 int idx = 0;
42 char *linebuf = NULL;
43 int linebufsz = 0;
44
45 while (1) {
46 ch = fgetc(file);
47 if (ch == EOF)
48 break;
49
50 /* grow the line buffer as necessary */
51 while (idx > linebufsz-4)
52 linebuf = xrealloc(linebuf, linebufsz += GROWBY);
53
54 /* Remove any extra spaces */
55 if (last_ch == ' ' && ch == ' ')
56 continue;
57
58 /* Replace any tabs with spaces */
59 if (ch == '\t')
60 ch=' ';
61
62 /* Escape any characters that are treated specially by /bin/sh */
63 *tmp=ch;
64 if (strpbrk(tmp, "\\~`!$^&*()=|{}[];\"'<>?#") != NULL && last_ch!='\\') {
65 linebuf[idx++] = '\\';
66 }
67
68 linebuf[idx++] = ch;
69 last_ch=ch;
70
71 if (ch == '\n')
72 break;
73 }
74 if (idx == 0 && last_ch!=0)
75 linebuf[idx++]=' ';
76
77 if (idx == 0)
78 return NULL;
79
80 linebuf[idx] = 0;
81 return linebuf;
82}
83
84
Eric Andersen92a61c12000-09-22 20:01:23 +000085
Eric Andersena37d5b72000-09-23 06:10:14 +000086int xargs_main(int argc, char **argv)
Eric Andersen92a61c12000-09-22 20:01:23 +000087{
Eric Andersena37d5b72000-09-23 06:10:14 +000088 char *in_from_stdin = NULL;
89 char *args_from_cmdline = NULL;
90 char *cmd_to_be_executed = NULL;
91 char traceflag = 0;
92 int len_args_from_cmdline, len_cmd_to_be_executed, len, opt;
Eric Andersen92a61c12000-09-22 20:01:23 +000093
Eric Andersen9ae38382000-09-24 01:12:54 +000094 /* Note that we do not use getopt here, since
95 * we only want to interpret initial options,
96 * not options passed to commands */
97 while (--argc && **(++argv) == '-') {
98 while (*++(*argv)) {
99 switch (**argv) {
100 case 't':
101 traceflag=1;
102 break;
103 default:
104 fatalError(xargs_usage);
105 }
Eric Andersena37d5b72000-09-23 06:10:14 +0000106 }
107 }
Eric Andersen92a61c12000-09-22 20:01:23 +0000108
Eric Andersena37d5b72000-09-23 06:10:14 +0000109 /* Store the command and arguments to be executed (from the command line) */
Eric Andersenbf739092000-09-25 18:41:18 +0000110 if (argc == 0) {
Eric Andersena37d5b72000-09-23 06:10:14 +0000111 len_args_from_cmdline = 6;
112 args_from_cmdline = xmalloc(len_args_from_cmdline);
113 strcat(args_from_cmdline, "echo ");
114 } else {
Eric Andersen9ae38382000-09-24 01:12:54 +0000115 opt=strlen(*argv);
Eric Andersena37d5b72000-09-23 06:10:14 +0000116 len_args_from_cmdline = (opt > 10)? opt : 10;
117 args_from_cmdline = xcalloc(len_args_from_cmdline, sizeof(char));
Eric Andersen9ae38382000-09-24 01:12:54 +0000118 while (argc-- > 0) {
119 if (strlen(*argv) + strlen(args_from_cmdline) >
Eric Andersena37d5b72000-09-23 06:10:14 +0000120 len_args_from_cmdline) {
Eric Andersen9ae38382000-09-24 01:12:54 +0000121 len_args_from_cmdline += strlen(*argv);
Eric Andersena37d5b72000-09-23 06:10:14 +0000122 args_from_cmdline =
123 xrealloc(args_from_cmdline,
124 len_args_from_cmdline+1);
125 }
Eric Andersen9ae38382000-09-24 01:12:54 +0000126 strcat(args_from_cmdline, *argv);
Eric Andersena37d5b72000-09-23 06:10:14 +0000127 strcat(args_from_cmdline, " ");
Eric Andersenbf739092000-09-25 18:41:18 +0000128 ++argv;
Eric Andersena37d5b72000-09-23 06:10:14 +0000129 }
130 }
Eric Andersen92a61c12000-09-22 20:01:23 +0000131
Eric Andersena37d5b72000-09-23 06:10:14 +0000132 /* Set up some space for the command to be executed to be held in */
133 len_cmd_to_be_executed=10;
134 cmd_to_be_executed = xcalloc(len_cmd_to_be_executed, sizeof(char));
135 strcpy(cmd_to_be_executed, args_from_cmdline);
Eric Andersen92a61c12000-09-22 20:01:23 +0000136
Eric Andersena37d5b72000-09-23 06:10:14 +0000137 /* Now, read in one line at a time from stdin, and run command+args on it */
Eric Andersenbf739092000-09-25 18:41:18 +0000138 in_from_stdin = get_sh_safe_line_from_file(stdin);
Eric Andersena37d5b72000-09-23 06:10:14 +0000139 for (;in_from_stdin!=NULL;) {
Eric Andersen96bdde92000-09-23 19:53:31 +0000140 char *tmp;
Eric Andersen6f283c22000-09-24 02:40:56 +0000141 opt = strlen(in_from_stdin);
142 len = opt + len_args_from_cmdline;
Eric Andersen96bdde92000-09-23 19:53:31 +0000143 len_cmd_to_be_executed+=len+3;
144 cmd_to_be_executed=xrealloc(cmd_to_be_executed, len_cmd_to_be_executed);
Eric Andersenbf739092000-09-25 18:41:18 +0000145
Eric Andersen6f283c22000-09-24 02:40:56 +0000146 /* Strip out the final \n */
147 in_from_stdin[opt-1]=' ';
Eric Andersenbf739092000-09-25 18:41:18 +0000148
Eric Andersen6f283c22000-09-24 02:40:56 +0000149 /* trim trailing whitespace */
150 opt = strlen(in_from_stdin) - 1;
151 while (isspace(in_from_stdin[opt]))
152 opt--;
153 in_from_stdin[++opt] = 0;
154
155 /* Strip out any leading whitespace */
156 tmp=in_from_stdin;
157 while(isspace(*tmp))
158 tmp++;
159
160 strcat(cmd_to_be_executed, tmp);
Eric Andersena37d5b72000-09-23 06:10:14 +0000161 strcat(cmd_to_be_executed, " ");
162
163 free(in_from_stdin);
Eric Andersenbf739092000-09-25 18:41:18 +0000164 in_from_stdin = get_sh_safe_line_from_file(stdin);
Eric Andersena37d5b72000-09-23 06:10:14 +0000165 }
166
167 if (traceflag==1)
168 fputs(cmd_to_be_executed, stderr);
169
170 if ((system(cmd_to_be_executed) != 0) && (errno != 0))
171 fatalError("%s", strerror(errno));
172
173
174#ifdef BB_FEATURE_CLEAN_UP
175 free(args_from_cmdline);
176 free(cmd_to_be_executed);
177#endif
178
179 return 0;
Eric Andersen92a61c12000-09-22 20:01:23 +0000180}
Eric Andersena37d5b72000-09-23 06:10:14 +0000181/*
182Local Variables:
183c-file-style: "linux"
184c-basic-offset: 4
185tab-width: 4
186End:
187*/
188