]>
diplodocus.org Git - nmh/blob - sbr/arglist.c
3 * arglist.c -- Routines for handling argument lists for execvp() and friends
5 * This code is Copyright (c) 2013, by the authors of nmh. See the
6 * COPYRIGHT file in the root directory of the nmh distribution for
7 * complete copyright information.
14 * Split up a command into an appropriate array to pass to execvp()
15 * or similar function. Returns an allocated argv[] array.
19 * command - String to split up
20 * file - the first argument to "command", suitable for the first argument
21 * to execvp(). Returns allocated memory that must be free()d.
22 * argp - Index to last element (NULL) of returned argv[] array.
24 * Our basic algorithm is this:
26 * - If there are no spaces or shell metacharacters in "command", then
28 * - If there are spaces in command, space-split command string.
29 * - If we have shell metacharacters, run the command using
30 * /bin/sh -c 'command "$@"'. In this case, any additional arguments
31 * appended to the arglist will be expanded by "$@".
33 * In all cases additional arguments can be added to the argv[] array.
36 /* Shell metacharacters we use to trigger a call to the shell */
38 #define METACHARS "$&*(){}[]'\";\\|?<>~`\n"
41 argsplit(char *command
, char **file
, int *argp
)
44 int space
= 0, metachar
= 0, i
;
46 for (p
= command
; *p
; p
++) {
47 if (*p
== ' ' || *p
== '\t') {
49 } else if (strchr(METACHARS
, *p
)) {
55 argvarray
= (char **) mh_xmalloc((sizeof(char **) * (MAXARGS
+ 5)));
58 * The simple case - no spaces or shell metacharacters
61 if (!space
&& !metachar
) {
62 argvarray
[0] = getcpy(r1bindex(command
, '/'));
64 *file
= mh_xstrdup(command
);
71 * Spaces, but no shell metacharacters; space-split into separate
75 if (space
&& !metachar
) {
77 p
= mh_xstrdup(command
);
78 split
= brkstring(p
, " \t", NULL
);
79 if (split
[0] == NULL
) {
80 adios(NULL
, "Invalid blank command found");
82 argvarray
[0] = mh_xstrdup(r1bindex(split
[0], '/'));
83 for (i
= 1; split
[i
] != NULL
; i
++) {
85 adios(NULL
, "Command exceeded argument limit");
87 argvarray
[i
] = mh_xstrdup(split
[i
]);
90 *file
= mh_xstrdup(split
[0]);
98 * Remaining option - pass to the shell.
102 * - The command passed to "sh -c" is actually:
105 * If there are additional arguments they will be expanded by the
106 * shell, otherwise "$@" expands to nothing.
108 * - Every argument after the -c string gets put into positional
109 * parameters starting at $0, but $@ starts expanding with $1.
110 * So we put in a dummy argument (we just use /bin/sh)
113 *file
= mh_xstrdup("/bin/sh");
114 argvarray
[0] = mh_xstrdup("sh");
115 argvarray
[1] = mh_xstrdup("-c");
116 argvarray
[2] = mh_xstrdup(command
);
117 argvarray
[2] = add(" \"$@\"", argvarray
[2]);
118 argvarray
[3] = mh_xstrdup("/bin/sh");
128 * Free our argument array
132 arglist_free(char *command
, char **argvarray
)
138 if (argvarray
!= NULL
) {
139 for (i
= 0; argvarray
[i
] != NULL
; i
++)
146 * Similar in functionality to argsplit, but is designed to deal with
151 argsplit_msgarg(struct msgs_array
*msgs
, char *command
, char **program
)
156 vec
= argsplit(command
, program
, &argp
);
159 * As usual, there is lousy memory management in nmh. Nothing ever
160 * free's the msgs_array, and a lot of the arguments are allocated
161 * from static memory. I could have app_msgarg() allocate new
162 * memory for each pointer, but for now I decided to stick with
163 * the existing interface; maybe that will be revisited later.
164 * So we'll just copy over our pointers and free the pointer list
165 * (not the actual pointers themselves). Note that we don't
166 * include a trailing NULL, since we are expecting the application
167 * to take care of that.
170 for (i
= 0; i
< argp
; i
++) {
171 app_msgarg(msgs
, vec
[i
]);
178 * Insert a arglist vector into the beginning of an struct msgs array
180 * Uses by some programs (e.g., show) who want to decide which proc
181 * to use after the argument vector has been constructed
189 argsplit_insert(struct msgs_array
*msgs
, char *command
, char **program
)
194 vec
= argsplit(command
, program
, &argp
);
197 * Okay, we want to shuffle all of our arguments down so we have room
198 * for argp number of arguments. This means we need to know about
199 * msgs_array internals. If that changes, we need to change this
203 if (msgs
->size
+ argp
>= msgs
->max
) {
204 msgs
->max
+= MAXMSGS
> argp
? MAXMSGS
: argp
;
205 msgs
->msgs
= mh_xrealloc(msgs
->msgs
, msgs
->max
* sizeof(*msgs
->msgs
));
208 for (i
= msgs
->size
- 1; i
>= 0; i
--)
209 msgs
->msgs
[i
+ argp
] = msgs
->msgs
[i
];
214 * Now fill in the arguments at the beginning of the vector.
217 for (i
= 0; i
< argp
; i
++)
218 msgs
->msgs
[i
] = vec
[i
];