mirror of git://sourceware.org/git/glibc.git
Update.
1998-09-13 14:53 Tim Waugh <tim@cyberelk.demon.co.uk>
* posix/wordexp-test.c: Field-splitting '🔤' with IFS=: should
yield three fields, not two. Test both parameter expansion and
command substitution for correct field-splitting behaviour.
* posix/wordexp.c (w_emptyword): New function.
(parse_param): Use it.
(exec_comm): Likewise, for consistency with the way parse_param
splits fields.
(parse_param): Fix some memory leaks.
This commit is contained in:
parent
2d09b95d5a
commit
1720f4d0a5
12
ChangeLog
12
ChangeLog
|
@ -1,3 +1,15 @@
|
|||
1998-09-13 14:53 Tim Waugh <tim@cyberelk.demon.co.uk>
|
||||
|
||||
* posix/wordexp-test.c: Field-splitting ':abc:' with IFS=: should
|
||||
yield three fields, not two. Test both parameter expansion and
|
||||
command substitution for correct field-splitting behaviour.
|
||||
|
||||
* posix/wordexp.c (w_emptyword): New function.
|
||||
(parse_param): Use it.
|
||||
(exec_comm): Likewise, for consistency with the way parse_param
|
||||
splits fields.
|
||||
(parse_param): Fix some memory leaks.
|
||||
|
||||
1998-09-13 18:04 Geoff Keating <geoffk@ozemail.com.au>
|
||||
|
||||
* sysdeps/powerpc/dl-machine.c (PPC_DCBST,PPC_SYNC,PPC_ISYNC,PPC_ICBI):
|
||||
|
|
|
@ -51,8 +51,19 @@ struct test_case_struct
|
|||
{ 0, "two three", "one $var", 0, 3, { "one", "two", "three", }, IFS },
|
||||
{ 0, "two three", "one \"$var\"", 0, 2, { "one", "two three", }, "" },
|
||||
{ 0, "two three", "one $var", 0, 2, { "one", "two three", }, "" },
|
||||
{ 0, ":abc:", "$var", 0, 2, { "", "abc", }, ":" }, /* cf. bash */
|
||||
{ 0, ":abc:", "$var", 0, 3, { "", "abc", "", }, ":" },
|
||||
{ 0, NULL, "$(echo :abc:)", 0, 3, { "", "abc", "", }, ":" },
|
||||
{ 0, NULL, "$(echo :abc:\\ )", 0, 3, { "", "abc", "", }, ": " },
|
||||
{ 0, NULL, "$(echo :abc\\ )", 0, 2, { "", "abc", }, ": " },
|
||||
{ 0, ":abc:", "$(echo $var)", 0, 3, { "", "abc", "", }, ":" },
|
||||
{ 0, NULL, ":abc:", 0, 1, { " abc ", }, ":" },
|
||||
{ 0, NULL, "$(echo :abc:)def", 0, 3, { "", "abc", "def", }, ":" },
|
||||
{ 0, NULL, "$(echo abc:de)f", 0, 2, { "abc", "def", }, ":" },
|
||||
{ 0, NULL, "$(echo abc:de)f:ghi", 0, 2, { "abc", "def ghi", }, ":" },
|
||||
{ 0, "abc:", "$var$(echo def:ghi)", 0, 3, { "abc", "def", "ghi", }, ":" },
|
||||
{ 0, "abc:d", "$var$(echo ef:ghi)", 0, 3, { "abc", "def", "ghi", }, ":" },
|
||||
{ 0, "def:ghi", "$(echo abc:)$var", 0, 3, { "abc", "def", "ghi", }, ":" },
|
||||
{ 0, "ef:ghi", "$(echo abc:d)$var", 0, 3, { "abc", "def", "ghi", }, ":" },
|
||||
|
||||
/* Simple parameter expansion */
|
||||
{ 0, "foo", "${var}", 0, 1, { "foo", }, IFS },
|
||||
|
|
|
@ -74,6 +74,7 @@ static int eval_expr (char *expr, long int *result) internal_function;
|
|||
|
||||
#define W_CHUNK (100)
|
||||
|
||||
/* Result of w_newword will be ignored if it the last word. */
|
||||
static inline char *
|
||||
w_newword (size_t *actlen, size_t *maxlen)
|
||||
{
|
||||
|
@ -136,6 +137,20 @@ w_addmem (char *buffer, size_t *actlen, size_t *maxlen, const char *str,
|
|||
}
|
||||
|
||||
|
||||
/* Result of w_emptyword will not be ignored even if it is the last. */
|
||||
static inline char *
|
||||
w_emptyword (size_t *actlen, size_t *maxlen)
|
||||
{
|
||||
char *word = malloc (1 + W_CHUNK);
|
||||
*maxlen = W_CHUNK;
|
||||
*actlen = 0;
|
||||
|
||||
if (word)
|
||||
*word = '\0';
|
||||
|
||||
return word;
|
||||
}
|
||||
|
||||
static char *
|
||||
internal_function
|
||||
w_addstr (char *buffer, size_t *actlen, size_t *maxlen, const char *str)
|
||||
|
@ -811,6 +826,7 @@ exec_comm (char *comm, char **word, size_t *word_length, size_t *max_length,
|
|||
int i;
|
||||
char *buffer;
|
||||
pid_t pid;
|
||||
int keep_empty_word = 0;
|
||||
|
||||
/* Don't fork() unless necessary */
|
||||
if (!comm || !*comm)
|
||||
|
@ -911,6 +927,11 @@ exec_comm (char *comm, char **word, size_t *word_length, size_t *max_length,
|
|||
if (strchr (ifs_white, buffer[i]) == NULL)
|
||||
{
|
||||
/* Current character is IFS but not whitespace */
|
||||
|
||||
/* After this delimiter, another field must result.
|
||||
* Make a note. */
|
||||
keep_empty_word = 1;
|
||||
|
||||
if (copying == 2)
|
||||
{
|
||||
/* current character
|
||||
|
@ -935,25 +956,12 @@ exec_comm (char *comm, char **word, size_t *word_length, size_t *max_length,
|
|||
if (copying != 1)
|
||||
continue;
|
||||
|
||||
/* End of field (search for non-IFS afterwards) */
|
||||
/* End of field (search for non-ws IFS afterwards) */
|
||||
copying = 2;
|
||||
}
|
||||
|
||||
/* First IFS white space, or IFS non-whitespace.
|
||||
* Delimit the field. */
|
||||
if (!*word)
|
||||
{
|
||||
/* This field is null, so make it an empty string */
|
||||
*word = w_addchar (*word, word_length, max_length, 0);
|
||||
if (*word == NULL)
|
||||
{
|
||||
__kill (pid, SIGKILL);
|
||||
__waitpid (pid, NULL, 0);
|
||||
__close (fildes[0]);
|
||||
return WRDE_NOSPACE;
|
||||
}
|
||||
}
|
||||
|
||||
* Delimit the field. Nulls are converted by w_addword. */
|
||||
if (w_addword (pwordexp, *word) == WRDE_NOSPACE)
|
||||
{
|
||||
__kill (pid, SIGKILL);
|
||||
|
@ -962,13 +970,20 @@ exec_comm (char *comm, char **word, size_t *word_length, size_t *max_length,
|
|||
return WRDE_NOSPACE;
|
||||
}
|
||||
|
||||
*word = w_newword (word_length, max_length);
|
||||
if (keep_empty_word)
|
||||
*word = w_emptyword (word_length, max_length);
|
||||
else
|
||||
*word = w_newword (word_length, max_length);
|
||||
/* fall back round the loop.. */
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Not IFS character */
|
||||
copying = 1;
|
||||
|
||||
if (buffer[i] != '\n')
|
||||
keep_empty_word = 0;
|
||||
|
||||
*word = w_addchar (*word, word_length, max_length,
|
||||
buffer[i]);
|
||||
if (*word == NULL)
|
||||
|
@ -985,7 +1000,18 @@ exec_comm (char *comm, char **word, size_t *word_length, size_t *max_length,
|
|||
|
||||
/* Bash chops off trailing newlines, which seems sensible. */
|
||||
while (*word_length > 0 && (*word)[*word_length - 1] == '\n')
|
||||
(*word)[--*word_length] = '\0';
|
||||
{
|
||||
(*word)[--*word_length] = '\0';
|
||||
|
||||
/* If the last word was entirely newlines, and the previous word
|
||||
* wasn't delimited with IFS non-whitespace, turn it into a new word
|
||||
* which can be ignored if there's nothing following it. */
|
||||
if (!keep_empty_word && *word_length == 0)
|
||||
{
|
||||
free (*word);
|
||||
*word = w_newword (word_length, max_length);
|
||||
}
|
||||
}
|
||||
|
||||
__close (fildes[0]);
|
||||
return 0;
|
||||
|
@ -1723,7 +1749,7 @@ envsubst:
|
|||
free (value);
|
||||
|
||||
if (value_copy == NULL)
|
||||
return WRDE_NOSPACE;
|
||||
goto no_space;
|
||||
|
||||
do
|
||||
{
|
||||
|
@ -1736,10 +1762,10 @@ envsubst:
|
|||
if (w_addword (pwordexp, *word) == WRDE_NOSPACE)
|
||||
{
|
||||
free (value_copy);
|
||||
return WRDE_NOSPACE;
|
||||
goto no_space;
|
||||
}
|
||||
|
||||
*word = w_newword (word_length, max_length);
|
||||
*word = w_emptyword (word_length, max_length);
|
||||
}
|
||||
|
||||
/* Skip IFS whitespace before the field */
|
||||
|
@ -1773,7 +1799,7 @@ envsubst:
|
|||
if (*word == NULL && *field_begin != '\0')
|
||||
{
|
||||
free (value_copy);
|
||||
return WRDE_NOSPACE;
|
||||
goto no_space;
|
||||
}
|
||||
|
||||
field_begin = next_field;
|
||||
|
@ -2217,7 +2243,7 @@ wordexp (const char *words, wordexp_t *pwordexp, int flags)
|
|||
/* End of string */
|
||||
|
||||
/* There was a word separator at the end */
|
||||
if (word == NULL)
|
||||
if (word == NULL) /* i.e. w_newword */
|
||||
return 0;
|
||||
|
||||
/* There was no field separator at the end */
|
||||
|
|
Loading…
Reference in New Issue