]> git.ipfire.org Git - thirdparty/binutils-gdb.git/commitdiff
Port expandargstr from gcc/libiberty
authorSunil Dora <sunilkumar.dora@windriver.com>
Thu, 11 Jun 2026 12:50:20 +0000 (18:20 +0530)
committerRichard Earnshaw <rearnsha@arm.com>
Mon, 15 Jun 2026 15:58:38 +0000 (16:58 +0100)
Sync recent changes to libiberty.

include/ChangeLog:

* libiberty.h (expandargstr): Declare.

libiberty/ChangeLog:

* argv.c (expandargstr): New function.

Signed-off-by: Sunil Dora <sunilkumar.dora@windriver.com>
include/libiberty.h
libiberty/argv.c

index 1af4fa861017f19e5e82c51bab34b42d92072d91..bdb491b9b373a1a35f5d9a02b99b88d5127bb7cc 100644 (file)
@@ -94,6 +94,11 @@ extern int writeargv (char * const *, FILE *);
 
 extern int countargv (char * const *);
 
+/* Expand VAL as a response file if it begins with '@' and return the
+   result as a shell-quoted string.  */
+
+extern char *expandargstr (const char *, const char *);
+
 /* Return the last component of a path name.  Note that we can't use a
    prototype here because the parameter is declared inconsistently
    across different systems, sometimes as "char *" and sometimes as
index 64127f70c9928b9fa8b72368439c9421fe67d099..35e2750c5ff31c8092683b9e18b463c11d978b91 100644 (file)
@@ -492,6 +492,81 @@ countargv (char * const *argv)
   return argc;
 }
 
+/*
+
+@deftypefn Extension {char *} expandargstr @
+  (const char *@var{progname}, const char *@var{val})
+
+Expand @var{val} as a response file via @code{expandargv} if it begins
+with @samp{@@}, using @var{progname} as @code{argv[0]}, and return the
+expanded option list as a shell-quoted string.  Returns a newly
+allocated string.
+
+@end deftypefn
+
+*/
+
+char *
+expandargstr (const char *progname, const char *val)
+{
+  int argc = 2;
+  char **argv;
+  char **orig;
+  size_t len;
+  char *buf;
+  char *p;
+  int i;
+
+  if (val[0] != '@')
+    return xstrdup (val);
+
+  argv = (char **) xcalloc (3, sizeof (char *));
+  orig = argv;
+  argv[0] = xstrdup (progname);
+  argv[1] = xstrdup (val);
+  argv[2] = NULL;
+  expandargv (&argc, &argv);
+  if (argv != orig)
+    freeargv (orig);
+
+  len = 1;
+  for (i = 1; argv[i] != NULL; i++)
+    {
+      const char *q;
+      if (i > 1)
+       len++;
+      len += 2;
+      for (q = argv[i]; *q; q++)
+       len += (*q == '\'') ? 4 : 1;
+    }
+
+  buf = (char *) xmalloc (len);
+  p = buf;
+  for (i = 1; argv[i] != NULL; i++)
+    {
+      const char *q;
+      if (i > 1)
+       *p++ = ' ';
+      *p++ = '\'';
+      for (q = argv[i]; *q; q++)
+       {
+         if (*q == '\'')
+           {
+             *p++ = '\'';
+             *p++ = '\\';
+             *p++ = '\'';
+             *p++ = '\'';
+           }
+         else
+           *p++ = *q;
+       }
+      *p++ = '\'';
+    }
+  *p = '\0';
+  freeargv (argv);
+  return buf;
+}
+
 #ifdef MAIN
 
 /* Simple little test driver. */