Update.
authorUlrich Drepper <drepper@redhat.com>
Sun, 13 Sep 1998 08:30:54 +0000 (08:30 +0000)
committerUlrich Drepper <drepper@redhat.com>
Sun, 13 Sep 1998 08:30:54 +0000 (08:30 +0000)
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.

ChangeLog
posix/wordexp-test.c
posix/wordexp.c

index e8e7b93..6c8862a 100644 (file)
--- a/ChangeLog
+++ b/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):
index 19048dc..1815241 100644 (file)
@@ -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 },
index 942a9ac..7fc21f5 100644 (file)
@@ -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 */