1996-08-26 18:22:31 +00:00
|
|
|
|
/* braces.c -- code for doing word expansion in curly braces. */
|
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
/* Copyright (C) 1987-2003 Free Software Foundation, Inc.
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
This file is part of GNU Bash, the Bourne Again SHell.
|
|
|
|
|
|
|
|
|
|
Bash is free software; you can redistribute it and/or modify it
|
|
|
|
|
under the terms of the GNU General Public License as published by
|
2000-03-17 21:46:59 +00:00
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
|
|
Bash is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
|
or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
|
|
|
|
|
License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with Bash; see the file COPYING. If not, write to the Free
|
2000-03-17 21:46:59 +00:00
|
|
|
|
Software Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
1996-12-23 17:02:34 +00:00
|
|
|
|
/* Stuff in curly braces gets expanded before all other shell expansions. */
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
1996-12-23 17:02:34 +00:00
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
|
|
#if defined (BRACE_EXPANSION)
|
|
|
|
|
|
|
|
|
|
#if defined (HAVE_UNISTD_H)
|
1998-04-17 19:52:44 +00:00
|
|
|
|
# ifdef _MINIX
|
|
|
|
|
# include <sys/types.h>
|
|
|
|
|
# endif
|
1996-12-23 17:02:34 +00:00
|
|
|
|
# include <unistd.h>
|
|
|
|
|
#endif
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
1997-06-05 14:59:13 +00:00
|
|
|
|
#include "bashansi.h"
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
#if defined (SHELL)
|
1996-12-23 17:02:34 +00:00
|
|
|
|
# include "shell.h"
|
1996-08-26 18:22:31 +00:00
|
|
|
|
#endif /* SHELL */
|
|
|
|
|
|
|
|
|
|
#include "general.h"
|
2002-07-17 14:10:11 +00:00
|
|
|
|
#include "shmbutil.h"
|
2004-07-27 13:29:18 +00:00
|
|
|
|
#include "chartypes.h"
|
2002-07-17 14:10:11 +00:00
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
#define brace_whitespace(c) (!(c) || (c) == ' ' || (c) == '\t' || (c) == '\n')
|
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
#define BRACE_SEQ_SPECIFIER ".."
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* Basic idea:
|
|
|
|
|
|
|
|
|
|
Segregate the text into 3 sections: preamble (stuff before an open brace),
|
|
|
|
|
postamble (stuff after the matching close brace) and amble (stuff after
|
|
|
|
|
preamble, and before postamble). Expand amble, and then tack on the
|
|
|
|
|
expansions to preamble. Expand postamble, and tack on the expansions to
|
|
|
|
|
the result so far.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* The character which is used to separate arguments. */
|
|
|
|
|
int brace_arg_separator = ',';
|
|
|
|
|
|
2001-11-13 17:56:06 +00:00
|
|
|
|
#if defined (__P)
|
2002-07-17 14:10:11 +00:00
|
|
|
|
static int brace_gobbler __P((char *, size_t, int *, int));
|
2004-07-27 13:29:18 +00:00
|
|
|
|
static char **expand_amble __P((char *, size_t, int));
|
|
|
|
|
static char **expand_seqterm __P((char *, size_t));
|
2006-10-10 14:15:34 +00:00
|
|
|
|
static char **mkseq __P((int, int, int, int));
|
2001-11-13 17:56:06 +00:00
|
|
|
|
static char **array_concat __P((char **, char **));
|
|
|
|
|
#else
|
1996-08-26 18:22:31 +00:00
|
|
|
|
static int brace_gobbler ();
|
2001-11-13 17:56:06 +00:00
|
|
|
|
static char **expand_amble ();
|
2004-07-27 13:29:18 +00:00
|
|
|
|
static char **expand_seqterm ();
|
|
|
|
|
static char **mkseq();
|
2001-11-13 17:56:06 +00:00
|
|
|
|
static char **array_concat ();
|
|
|
|
|
#endif
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2006-10-10 14:15:34 +00:00
|
|
|
|
#if 0
|
|
|
|
|
static void
|
|
|
|
|
dump_result (a)
|
|
|
|
|
char **a;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; a[i]; i++)
|
|
|
|
|
printf ("dump_result: a[%d] = -%s-\n", i, a[i]);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* Return an array of strings; the brace expansion of TEXT. */
|
|
|
|
|
char **
|
|
|
|
|
brace_expand (text)
|
|
|
|
|
char *text;
|
|
|
|
|
{
|
|
|
|
|
register int start;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
size_t tlen;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
char *preamble, *postamble, *amble;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
size_t alen;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
char **tack, **result;
|
2006-10-10 14:15:34 +00:00
|
|
|
|
int i, j, c, c1;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
DECLARE_MBSTATE;
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* Find the text of the preamble. */
|
2002-07-17 14:10:11 +00:00
|
|
|
|
tlen = strlen (text);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
i = 0;
|
2006-10-10 14:15:34 +00:00
|
|
|
|
#if defined (CSH_BRACE_COMPAT)
|
|
|
|
|
c = brace_gobbler (text, tlen, &i, '{'); /* } */
|
|
|
|
|
#else
|
|
|
|
|
/* Make sure that when we exit this loop, c == 0 or text[i] begins a
|
|
|
|
|
valid brace expansion sequence. */
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
c = brace_gobbler (text, tlen, &i, '{'); /* } */
|
|
|
|
|
c1 = c;
|
|
|
|
|
/* Verify that c begins a valid brace expansion word. If it doesn't, we
|
|
|
|
|
go on. Loop stops when there are no more open braces in the word. */
|
|
|
|
|
if (c)
|
|
|
|
|
{
|
|
|
|
|
start = j = i + 1; /* { */
|
|
|
|
|
c = brace_gobbler (text, tlen, &j, '}');
|
|
|
|
|
if (c == 0) /* it's not */
|
|
|
|
|
{
|
|
|
|
|
i++;
|
|
|
|
|
c = c1;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else /* it is */
|
|
|
|
|
{
|
|
|
|
|
c = c1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
while (c);
|
|
|
|
|
#endif /* !CSH_BRACE_COMPAT */
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
preamble = (char *)xmalloc (i + 1);
|
|
|
|
|
strncpy (preamble, text, i);
|
|
|
|
|
preamble[i] = '\0';
|
|
|
|
|
|
|
|
|
|
result = (char **)xmalloc (2 * sizeof (char *));
|
|
|
|
|
result[0] = preamble;
|
|
|
|
|
result[1] = (char *)NULL;
|
1996-12-23 17:02:34 +00:00
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* Special case. If we never found an exciting character, then
|
|
|
|
|
the preamble is all of the text, so just return that. */
|
|
|
|
|
if (c != '{')
|
|
|
|
|
return (result);
|
|
|
|
|
|
|
|
|
|
/* Find the amble. This is the stuff inside this set of braces. */
|
|
|
|
|
start = ++i;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
c = brace_gobbler (text, tlen, &i, '}');
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
/* What if there isn't a matching close brace? */
|
1996-12-23 17:02:34 +00:00
|
|
|
|
if (c == 0)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
|
|
|
|
#if defined (NOTDEF)
|
|
|
|
|
/* Well, if we found an unquoted BRACE_ARG_SEPARATOR between START
|
|
|
|
|
and I, then this should be an error. Otherwise, it isn't. */
|
2002-07-17 14:10:11 +00:00
|
|
|
|
j = start;
|
|
|
|
|
while (j < i)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
|
|
|
|
if (text[j] == '\\')
|
|
|
|
|
{
|
|
|
|
|
j++;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (text, tlen, j);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (text[j] == brace_arg_separator)
|
2004-07-27 13:29:18 +00:00
|
|
|
|
{ /* { */
|
2002-07-17 14:10:11 +00:00
|
|
|
|
strvec_dispose (result);
|
2004-07-27 13:29:18 +00:00
|
|
|
|
report_error ("no closing `%c' in %s", '}', text);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
throw_to_top_level ();
|
|
|
|
|
}
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (text, tlen, j);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
free (preamble); /* Same as result[0]; see initialization. */
|
|
|
|
|
result[0] = savestring (text);
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-17 21:46:59 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
amble = substring (text, start, i);
|
2002-07-17 14:10:11 +00:00
|
|
|
|
alen = i - start;
|
2000-03-17 21:46:59 +00:00
|
|
|
|
#else
|
1996-08-26 18:22:31 +00:00
|
|
|
|
amble = (char *)xmalloc (1 + (i - start));
|
|
|
|
|
strncpy (amble, &text[start], (i - start));
|
2002-07-17 14:10:11 +00:00
|
|
|
|
alen = i - start;
|
|
|
|
|
amble[alen] = '\0';
|
2000-03-17 21:46:59 +00:00
|
|
|
|
#endif
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
#if defined (SHELL)
|
2002-07-17 14:10:11 +00:00
|
|
|
|
INITIALIZE_MBSTATE;
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* If the amble does not contain an unquoted BRACE_ARG_SEPARATOR, then
|
|
|
|
|
just return without doing any expansion. */
|
2002-07-17 14:10:11 +00:00
|
|
|
|
j = 0;
|
|
|
|
|
while (amble[j])
|
1996-12-23 17:02:34 +00:00
|
|
|
|
{
|
|
|
|
|
if (amble[j] == '\\')
|
|
|
|
|
{
|
|
|
|
|
j++;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (amble, alen, j);
|
1996-12-23 17:02:34 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2002-07-17 14:10:11 +00:00
|
|
|
|
|
1996-12-23 17:02:34 +00:00
|
|
|
|
if (amble[j] == brace_arg_separator)
|
|
|
|
|
break;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
|
|
|
|
|
ADVANCE_CHAR (amble, alen, j);
|
1996-12-23 17:02:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
if (amble[j] == 0)
|
1996-12-23 17:02:34 +00:00
|
|
|
|
{
|
2004-07-27 13:29:18 +00:00
|
|
|
|
tack = expand_seqterm (amble, alen);
|
|
|
|
|
if (tack)
|
|
|
|
|
goto add_tack;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
free (amble);
|
|
|
|
|
free (preamble);
|
|
|
|
|
result[0] = savestring (text);
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
1996-12-23 17:02:34 +00:00
|
|
|
|
}
|
1996-08-26 18:22:31 +00:00
|
|
|
|
#endif /* SHELL */
|
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
tack = expand_amble (amble, alen, 0);
|
|
|
|
|
add_tack:
|
1996-08-26 18:22:31 +00:00
|
|
|
|
result = array_concat (result, tack);
|
|
|
|
|
free (amble);
|
2002-07-17 14:10:11 +00:00
|
|
|
|
strvec_dispose (tack);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
postamble = text + i + 1;
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
tack = brace_expand (postamble);
|
|
|
|
|
result = array_concat (result, tack);
|
2002-07-17 14:10:11 +00:00
|
|
|
|
strvec_dispose (tack);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Expand the text found inside of braces. We simply try to split the
|
|
|
|
|
text at BRACE_ARG_SEPARATORs into separate strings. We then brace
|
|
|
|
|
expand each slot which needs it, until there are no more slots which
|
|
|
|
|
need it. */
|
|
|
|
|
static char **
|
2004-07-27 13:29:18 +00:00
|
|
|
|
expand_amble (text, tlen, flags)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
char *text;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
size_t tlen;
|
2004-07-27 13:29:18 +00:00
|
|
|
|
int flags;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
|
|
|
|
char **result, **partial;
|
|
|
|
|
char *tem;
|
|
|
|
|
int start, i, c;
|
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
DECLARE_MBSTATE;
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
result = (char **)NULL;
|
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
start = i = 0;
|
|
|
|
|
c = 1;
|
|
|
|
|
while (c)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
2002-07-17 14:10:11 +00:00
|
|
|
|
c = brace_gobbler (text, tlen, &i, brace_arg_separator);
|
2000-03-17 21:46:59 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
tem = substring (text, start, i);
|
|
|
|
|
#else
|
1996-08-26 18:22:31 +00:00
|
|
|
|
tem = (char *)xmalloc (1 + (i - start));
|
|
|
|
|
strncpy (tem, &text[start], (i - start));
|
|
|
|
|
tem[i- start] = '\0';
|
2000-03-17 21:46:59 +00:00
|
|
|
|
#endif
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
partial = brace_expand (tem);
|
|
|
|
|
|
|
|
|
|
if (!result)
|
|
|
|
|
result = partial;
|
|
|
|
|
else
|
|
|
|
|
{
|
2004-07-27 13:29:18 +00:00
|
|
|
|
register int lr, lp, j;
|
|
|
|
|
|
|
|
|
|
lr = strvec_len (result);
|
|
|
|
|
lp = strvec_len (partial);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
result = strvec_resize (result, lp + lr + 1);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
for (j = 0; j < lp; j++)
|
|
|
|
|
result[lr + j] = partial[j];
|
|
|
|
|
|
|
|
|
|
result[lr + j] = (char *)NULL;
|
|
|
|
|
free (partial);
|
|
|
|
|
}
|
|
|
|
|
free (tem);
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (text, tlen, i);
|
|
|
|
|
start = i;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
}
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
#define ST_BAD 0
|
|
|
|
|
#define ST_INT 1
|
|
|
|
|
#define ST_CHAR 2
|
|
|
|
|
|
|
|
|
|
static char **
|
2006-10-10 14:15:34 +00:00
|
|
|
|
mkseq (start, end, incr, type)
|
|
|
|
|
int start, end, incr, type;
|
2004-07-27 13:29:18 +00:00
|
|
|
|
{
|
2006-10-10 14:15:34 +00:00
|
|
|
|
int n, i;
|
2004-07-27 13:29:18 +00:00
|
|
|
|
char **result, *t;
|
|
|
|
|
|
|
|
|
|
n = abs (end - start) + 1;
|
|
|
|
|
result = strvec_create (n + 1);
|
|
|
|
|
|
2006-10-10 14:15:34 +00:00
|
|
|
|
if (incr == 0)
|
|
|
|
|
incr = 1;
|
|
|
|
|
|
|
|
|
|
if (start > end && incr > 0)
|
|
|
|
|
incr = -incr;
|
|
|
|
|
else if (start < end && incr < 0)
|
|
|
|
|
incr = -incr;
|
2004-07-27 13:29:18 +00:00
|
|
|
|
|
|
|
|
|
/* Make sure we go through the loop at least once, so {3..3} prints `3' */
|
|
|
|
|
i = 0;
|
|
|
|
|
n = start;
|
|
|
|
|
do
|
|
|
|
|
{
|
2006-10-10 14:15:34 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
QUIT; /* XXX - memory leak here */
|
|
|
|
|
#endif
|
2004-07-27 13:29:18 +00:00
|
|
|
|
if (type == ST_INT)
|
|
|
|
|
result[i++] = itos (n);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
t = (char *)xmalloc (2);
|
|
|
|
|
t[0] = n;
|
|
|
|
|
t[1] = '\0';
|
|
|
|
|
result[i++] = t;
|
|
|
|
|
}
|
|
|
|
|
if (n == end)
|
|
|
|
|
break;
|
|
|
|
|
n += incr;
|
|
|
|
|
}
|
|
|
|
|
while (1);
|
|
|
|
|
|
|
|
|
|
result[i] = (char *)0;
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static char **
|
|
|
|
|
expand_seqterm (text, tlen)
|
|
|
|
|
char *text;
|
|
|
|
|
size_t tlen;
|
|
|
|
|
{
|
|
|
|
|
char *t, *lhs, *rhs;
|
|
|
|
|
int i, lhs_t, rhs_t, lhs_v, rhs_v;
|
|
|
|
|
intmax_t tl, tr;
|
|
|
|
|
char **result;
|
|
|
|
|
|
|
|
|
|
t = strstr (text, BRACE_SEQ_SPECIFIER);
|
|
|
|
|
if (t == 0)
|
|
|
|
|
return ((char **)NULL);
|
|
|
|
|
|
|
|
|
|
i = t - text; /* index of start of BRACE_SEQ_SPECIFIER */
|
|
|
|
|
lhs = substring (text, 0, i);
|
|
|
|
|
rhs = substring (text, i + sizeof(BRACE_SEQ_SPECIFIER) - 1, tlen);
|
|
|
|
|
|
|
|
|
|
if (lhs[0] == 0 || rhs[0] == 0)
|
|
|
|
|
{
|
|
|
|
|
free (lhs);
|
|
|
|
|
free (rhs);
|
|
|
|
|
return ((char **)NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Now figure out whether LHS and RHS are integers or letters. Both
|
|
|
|
|
sides have to match. */
|
|
|
|
|
lhs_t = (legal_number (lhs, &tl)) ? ST_INT :
|
|
|
|
|
((ISALPHA (lhs[0]) && lhs[1] == 0) ? ST_CHAR : ST_BAD);
|
|
|
|
|
rhs_t = (legal_number (rhs, &tr)) ? ST_INT :
|
|
|
|
|
((ISALPHA (rhs[0]) && rhs[1] == 0) ? ST_CHAR : ST_BAD);
|
|
|
|
|
|
|
|
|
|
if (lhs_t != rhs_t || lhs_t == ST_BAD || rhs_t == ST_BAD)
|
|
|
|
|
{
|
|
|
|
|
free (lhs);
|
|
|
|
|
free (rhs);
|
|
|
|
|
return ((char **)NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* OK, we have something. It's either a sequence of integers, ascending
|
|
|
|
|
or descending, or a sequence or letters, ditto. Generate the sequence,
|
|
|
|
|
put it into a string vector, and return it. */
|
|
|
|
|
|
|
|
|
|
if (lhs_t == ST_CHAR)
|
|
|
|
|
{
|
2004-11-09 21:37:25 +00:00
|
|
|
|
lhs_v = (unsigned char)lhs[0];
|
|
|
|
|
rhs_v = (unsigned char)rhs[0];
|
2004-07-27 13:29:18 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
lhs_v = tl; /* integer truncation */
|
|
|
|
|
rhs_v = tr;
|
|
|
|
|
}
|
|
|
|
|
|
2006-10-10 14:15:34 +00:00
|
|
|
|
result = mkseq (lhs_v, rhs_v, 1, lhs_t);
|
2004-07-27 13:29:18 +00:00
|
|
|
|
|
|
|
|
|
free (lhs);
|
|
|
|
|
free (rhs);
|
|
|
|
|
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
/* Start at INDEX, and skip characters in TEXT. Set INDEX to the
|
|
|
|
|
index of the character matching SATISFY. This understands about
|
|
|
|
|
quoting. Return the character that caused us to stop searching;
|
|
|
|
|
this is either the same as SATISFY, or 0. */
|
2006-10-10 14:15:34 +00:00
|
|
|
|
/* If SATISFY is `}', we are looking for a brace expression, so we
|
|
|
|
|
should enforce the rules that govern valid brace expansions:
|
|
|
|
|
1) to count as an arg separator, a comma or `..' has to be outside
|
|
|
|
|
an inner set of braces.
|
|
|
|
|
*/
|
1996-08-26 18:22:31 +00:00
|
|
|
|
static int
|
2002-07-17 14:10:11 +00:00
|
|
|
|
brace_gobbler (text, tlen, indx, satisfy)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
char *text;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
size_t tlen;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
int *indx;
|
|
|
|
|
int satisfy;
|
|
|
|
|
{
|
2006-10-10 14:15:34 +00:00
|
|
|
|
register int i, c, quoted, level, commas, pass_next;
|
1997-06-05 14:59:13 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
int si;
|
|
|
|
|
char *t;
|
|
|
|
|
#endif
|
2002-07-17 14:10:11 +00:00
|
|
|
|
DECLARE_MBSTATE;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
level = quoted = pass_next = 0;
|
2006-10-10 14:15:34 +00:00
|
|
|
|
#if defined (CSH_BRACE_COMPAT)
|
|
|
|
|
commas = 1;
|
|
|
|
|
#else
|
|
|
|
|
commas = (satisfy == '}') ? 0 : 1;
|
|
|
|
|
#endif
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
i = *indx;
|
|
|
|
|
while (c = text[i])
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
|
|
|
|
if (pass_next)
|
|
|
|
|
{
|
|
|
|
|
pass_next = 0;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (text, tlen, i);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* A backslash escapes the next character. This allows backslash to
|
|
|
|
|
escape the quote character in a double-quoted string. */
|
|
|
|
|
if (c == '\\' && (quoted == 0 || quoted == '"' || quoted == '`'))
|
2001-04-06 19:14:31 +00:00
|
|
|
|
{
|
|
|
|
|
pass_next = 1;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
i++;
|
2001-04-06 19:14:31 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2004-07-27 13:29:18 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
/* If compiling for the shell, treat ${...} like \{...} */
|
|
|
|
|
if (c == '$' && text[i+1] == '{' && quoted != '\'') /* } */
|
|
|
|
|
{
|
|
|
|
|
pass_next = 1;
|
|
|
|
|
i++;
|
2004-11-09 21:37:25 +00:00
|
|
|
|
if (quoted == 0)
|
|
|
|
|
level++;
|
2004-07-27 13:29:18 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
if (quoted)
|
|
|
|
|
{
|
|
|
|
|
if (c == quoted)
|
|
|
|
|
quoted = 0;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
ADVANCE_CHAR (text, tlen, i);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c == '"' || c == '\'' || c == '`')
|
|
|
|
|
{
|
|
|
|
|
quoted = c;
|
2002-07-17 14:10:11 +00:00
|
|
|
|
i++;
|
1996-08-26 18:22:31 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
1996-12-23 17:02:34 +00:00
|
|
|
|
|
1997-06-05 14:59:13 +00:00
|
|
|
|
#if defined (SHELL)
|
|
|
|
|
/* Pass new-style command substitutions through unchanged. */
|
|
|
|
|
if (c == '$' && text[i+1] == '(') /* ) */
|
|
|
|
|
{
|
|
|
|
|
si = i + 2;
|
|
|
|
|
t = extract_command_subst (text, &si);
|
|
|
|
|
i = si;
|
|
|
|
|
free (t);
|
2002-07-17 14:10:11 +00:00
|
|
|
|
i++;
|
1997-06-05 14:59:13 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2006-10-10 14:15:34 +00:00
|
|
|
|
if (c == satisfy && level == 0 && quoted == 0 && commas > 0)
|
1996-08-26 18:22:31 +00:00
|
|
|
|
{
|
|
|
|
|
/* We ignore an open brace surrounded by whitespace, and also
|
1996-12-23 17:02:34 +00:00
|
|
|
|
an open brace followed immediately by a close brace preceded
|
|
|
|
|
by whitespace. */
|
1996-08-26 18:22:31 +00:00
|
|
|
|
if (c == '{' &&
|
|
|
|
|
((!i || brace_whitespace (text[i - 1])) &&
|
|
|
|
|
(brace_whitespace (text[i + 1]) || text[i + 1] == '}')))
|
2002-07-17 14:10:11 +00:00
|
|
|
|
{
|
|
|
|
|
i++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2004-07-27 13:29:18 +00:00
|
|
|
|
|
1996-08-26 18:22:31 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c == '{')
|
|
|
|
|
level++;
|
|
|
|
|
else if (c == '}' && level)
|
|
|
|
|
level--;
|
2006-10-10 14:15:34 +00:00
|
|
|
|
#if !defined (CSH_BRACE_COMPAT)
|
|
|
|
|
else if (satisfy == '}' && c == brace_arg_separator && level == 0)
|
|
|
|
|
commas++;
|
|
|
|
|
else if (satisfy == '}' && STREQN (text+i, BRACE_SEQ_SPECIFIER, 2) &&
|
|
|
|
|
text[i+2] != satisfy && level == 0)
|
|
|
|
|
commas++;
|
|
|
|
|
#endif
|
2002-07-17 14:10:11 +00:00
|
|
|
|
|
|
|
|
|
ADVANCE_CHAR (text, tlen, i);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*indx = i;
|
|
|
|
|
return (c);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return a new array of strings which is the result of appending each
|
|
|
|
|
string in ARR2 to each string in ARR1. The resultant array is
|
|
|
|
|
len (arr1) * len (arr2) long. For convenience, ARR1 (and its contents)
|
|
|
|
|
are free ()'ed. ARR1 can be NULL, in that case, a new version of ARR2
|
|
|
|
|
is returned. */
|
|
|
|
|
static char **
|
|
|
|
|
array_concat (arr1, arr2)
|
|
|
|
|
char **arr1, **arr2;
|
|
|
|
|
{
|
|
|
|
|
register int i, j, len, len1, len2;
|
|
|
|
|
register char **result;
|
|
|
|
|
|
1996-12-23 17:02:34 +00:00
|
|
|
|
if (arr1 == 0)
|
2002-07-17 14:10:11 +00:00
|
|
|
|
return (strvec_copy (arr2));
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
1996-12-23 17:02:34 +00:00
|
|
|
|
if (arr2 == 0)
|
2002-07-17 14:10:11 +00:00
|
|
|
|
return (strvec_copy (arr1));
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
2002-07-17 14:10:11 +00:00
|
|
|
|
len1 = strvec_len (arr1);
|
|
|
|
|
len2 = strvec_len (arr2);
|
1996-08-26 18:22:31 +00:00
|
|
|
|
|
|
|
|
|
result = (char **)xmalloc ((1 + (len1 * len2)) * sizeof (char *));
|
|
|
|
|
|
|
|
|
|
len = 0;
|
|
|
|
|
for (i = 0; i < len1; i++)
|
|
|
|
|
{
|
|
|
|
|
int strlen_1 = strlen (arr1[i]);
|
|
|
|
|
|
|
|
|
|
for (j = 0; j < len2; j++)
|
|
|
|
|
{
|
2004-07-27 13:29:18 +00:00
|
|
|
|
result[len] = (char *)xmalloc (1 + strlen_1 + strlen (arr2[j]));
|
1996-08-26 18:22:31 +00:00
|
|
|
|
strcpy (result[len], arr1[i]);
|
|
|
|
|
strcpy (result[len] + strlen_1, arr2[j]);
|
|
|
|
|
len++;
|
|
|
|
|
}
|
|
|
|
|
free (arr1[i]);
|
|
|
|
|
}
|
|
|
|
|
free (arr1);
|
|
|
|
|
|
|
|
|
|
result[len] = (char *)NULL;
|
|
|
|
|
return (result);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if defined (TEST)
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
|
|
fatal_error (format, arg1, arg2)
|
|
|
|
|
char *format, *arg1, *arg2;
|
|
|
|
|
{
|
|
|
|
|
report_error (format, arg1, arg2);
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
report_error (format, arg1, arg2)
|
|
|
|
|
char *format, *arg1, *arg2;
|
|
|
|
|
{
|
|
|
|
|
fprintf (stderr, format, arg1, arg2);
|
|
|
|
|
fprintf (stderr, "\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
main ()
|
|
|
|
|
{
|
|
|
|
|
char example[256];
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
char **result;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
fprintf (stderr, "brace_expand> ");
|
|
|
|
|
|
|
|
|
|
if ((!fgets (example, 256, stdin)) ||
|
|
|
|
|
(strncmp (example, "quit", 4) == 0))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (strlen (example))
|
|
|
|
|
example[strlen (example) - 1] = '\0';
|
|
|
|
|
|
|
|
|
|
result = brace_expand (example);
|
|
|
|
|
|
|
|
|
|
for (i = 0; result[i]; i++)
|
|
|
|
|
printf ("%s\n", result[i]);
|
|
|
|
|
|
|
|
|
|
free_array (result);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Local variables:
|
|
|
|
|
* compile-command: "gcc -g -Bstatic -DTEST -o brace_expand braces.c general.o"
|
|
|
|
|
* end:
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#endif /* TEST */
|
1996-12-23 17:02:34 +00:00
|
|
|
|
#endif /* BRACE_EXPANSION */
|