|
@@ -0,0 +1,367 @@
|
|
|
+/* quotearg.c - quote arguments for output
|
|
|
+ Copyright (C) 1998, 1999 Free Software Foundation, Inc.
|
|
|
+
|
|
|
+ This program is free software; you can redistribute it and/or modify
|
|
|
+ it under the terms of the GNU General Public License as published by
|
|
|
+ the Free Software Foundation; either version 2, or (at your option)
|
|
|
+ any later version.
|
|
|
+
|
|
|
+ This program 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 this program; if not, write to the Free Software Foundation,
|
|
|
+ Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
|
|
+
|
|
|
+/* Written by Paul Eggert <[email protected]> */
|
|
|
+
|
|
|
+#if HAVE_CONFIG_H
|
|
|
+# include <config.h>
|
|
|
+#endif
|
|
|
+
|
|
|
+#include <sys/types.h>
|
|
|
+#include <quotearg.h>
|
|
|
+#include <xalloc.h>
|
|
|
+
|
|
|
+#include <ctype.h>
|
|
|
+#if defined (STDC_HEADERS) || (!defined (isascii) && !defined (HAVE_ISASCII))
|
|
|
+# define ISASCII(c) 1
|
|
|
+#else
|
|
|
+# define ISASCII(c) isascii (c)
|
|
|
+#endif
|
|
|
+#ifdef isgraph
|
|
|
+# define ISGRAPH(c) (ISASCII (c) && isgraph (c))
|
|
|
+#else
|
|
|
+# define ISGRAPH(c) (ISASCII (c) && isprint (c) && !isspace (c))
|
|
|
+#endif
|
|
|
+
|
|
|
+#if HAVE_LIMITS_H
|
|
|
+# include <limits.h>
|
|
|
+#endif
|
|
|
+#ifndef CHAR_BIT
|
|
|
+# define CHAR_BIT 8
|
|
|
+#endif
|
|
|
+#ifndef UCHAR_MAX
|
|
|
+# define UCHAR_MAX ((unsigned char) -1)
|
|
|
+#endif
|
|
|
+
|
|
|
+#if HAVE_STDLIB_H
|
|
|
+# include <stdlib.h>
|
|
|
+#endif
|
|
|
+
|
|
|
+#if HAVE_STRING_H
|
|
|
+# include <string.h>
|
|
|
+#endif
|
|
|
+
|
|
|
+#define INT_BITS (sizeof (int) * CHAR_BIT)
|
|
|
+
|
|
|
+struct quoting_options
|
|
|
+{
|
|
|
+ /* Basic quoting style. */
|
|
|
+ enum quoting_style style;
|
|
|
+
|
|
|
+ /* Quote the chararacters indicated by this bit vector even if the
|
|
|
+ quoting style would not normally require them to be quoted. */
|
|
|
+ int quote_these_too[((UCHAR_MAX + 1) / INT_BITS
|
|
|
+ + ((UCHAR_MAX + 1) % INT_BITS != 0))];
|
|
|
+};
|
|
|
+
|
|
|
+/* Names of quoting styles. */
|
|
|
+char const *const quoting_style_args[] =
|
|
|
+{
|
|
|
+ "literal",
|
|
|
+ "shell",
|
|
|
+ "shell-always",
|
|
|
+ "c",
|
|
|
+ "escape",
|
|
|
+ 0
|
|
|
+};
|
|
|
+
|
|
|
+/* Correspondances to quoting style names. */
|
|
|
+enum quoting_style const quoting_style_vals[] =
|
|
|
+{
|
|
|
+ literal_quoting_style,
|
|
|
+ shell_quoting_style,
|
|
|
+ shell_always_quoting_style,
|
|
|
+ c_quoting_style,
|
|
|
+ escape_quoting_style
|
|
|
+};
|
|
|
+
|
|
|
+/* The default quoting options. */
|
|
|
+static struct quoting_options default_quoting_options;
|
|
|
+
|
|
|
+/* Allocate a new set of quoting options, with contents initially identical
|
|
|
+ to O if O is not null, or to the default if O is null.
|
|
|
+ It is the caller's responsibility to free the result. */
|
|
|
+struct quoting_options *
|
|
|
+clone_quoting_options (struct quoting_options *o)
|
|
|
+{
|
|
|
+ struct quoting_options *p
|
|
|
+ = (struct quoting_options *) xmalloc (sizeof (struct quoting_options));
|
|
|
+ *p = *(o ? o : &default_quoting_options);
|
|
|
+ return p;
|
|
|
+}
|
|
|
+
|
|
|
+/* Get the value of O's quoting style. If O is null, use the default. */
|
|
|
+enum quoting_style
|
|
|
+get_quoting_style (struct quoting_options *o)
|
|
|
+{
|
|
|
+ return (o ? o : &default_quoting_options)->style;
|
|
|
+}
|
|
|
+
|
|
|
+/* In O (or in the default if O is null),
|
|
|
+ set the value of the quoting style to S. */
|
|
|
+void
|
|
|
+set_quoting_style (struct quoting_options *o, enum quoting_style s)
|
|
|
+{
|
|
|
+ (o ? o : &default_quoting_options)->style = s;
|
|
|
+}
|
|
|
+
|
|
|
+/* In O (or in the default if O is null),
|
|
|
+ set the value of the quoting options for character C to I.
|
|
|
+ Return the old value. Currently, the only values defined for I are
|
|
|
+ 0 (the default) and 1 (which means to quote the character even if
|
|
|
+ it would not otherwise be quoted). */
|
|
|
+int
|
|
|
+set_char_quoting (struct quoting_options *o, char c, int i)
|
|
|
+{
|
|
|
+ unsigned char uc = c;
|
|
|
+ int *p = (o ? o : &default_quoting_options)->quote_these_too + uc / INT_BITS;
|
|
|
+ int shift = uc % INT_BITS;
|
|
|
+ int r = (*p >> shift) & 1;
|
|
|
+ *p ^= ((i & 1) ^ r) << shift;
|
|
|
+ return r;
|
|
|
+}
|
|
|
+
|
|
|
+/* Place into buffer BUFFER (of size BUFFERSIZE) a quoted version of
|
|
|
+ argument ARG (of size ARGSIZE), using O to control quoting.
|
|
|
+ If O is null, use the default.
|
|
|
+ Terminate the output with a null character, and return the written
|
|
|
+ size of the output, not counting the terminating null.
|
|
|
+ If BUFFERSIZE is too small to store the output string, return the
|
|
|
+ value that would have been returned had BUFFERSIZE been large enough.
|
|
|
+ If ARGSIZE is -1, use the string length of the argument for ARGSIZE. */
|
|
|
+size_t
|
|
|
+quotearg_buffer (char *buffer, size_t buffersize,
|
|
|
+ char const *arg, size_t argsize,
|
|
|
+ struct quoting_options const *o)
|
|
|
+{
|
|
|
+ unsigned char c;
|
|
|
+ size_t i;
|
|
|
+ size_t len;
|
|
|
+ int quote_mark;
|
|
|
+ struct quoting_options const *p = o ? o : &default_quoting_options;
|
|
|
+ enum quoting_style quoting_style = p->style;
|
|
|
+#define STORE(c) \
|
|
|
+ do \
|
|
|
+ { \
|
|
|
+ if (len < buffersize) \
|
|
|
+ buffer[len] = (c); \
|
|
|
+ len++; \
|
|
|
+ } \
|
|
|
+ while (0)
|
|
|
+
|
|
|
+ switch (quoting_style)
|
|
|
+ {
|
|
|
+ case shell_quoting_style:
|
|
|
+ if (! (argsize == (size_t) -1 ? arg[0] == '\0' : argsize == 0))
|
|
|
+ {
|
|
|
+ switch (arg[0])
|
|
|
+ {
|
|
|
+ case '#': case '~':
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ len = 0;
|
|
|
+ for (i = 0; ; i++)
|
|
|
+ {
|
|
|
+ if (argsize == (size_t) -1 ? arg[i] == '\0' : i == argsize)
|
|
|
+ goto done;
|
|
|
+
|
|
|
+ c = arg[i];
|
|
|
+
|
|
|
+ switch (c)
|
|
|
+ {
|
|
|
+ case '\t': case '\n': case ' ':
|
|
|
+ case '!': /* special in csh */
|
|
|
+ case '"': case '$': case '&': case '\'':
|
|
|
+ case '(': case ')': case '*': case ';':
|
|
|
+ case '<': case '>': case '?': case '[': case '\\':
|
|
|
+ case '^': /* special in old /bin/sh, e.g. SunOS 4.1.4 */
|
|
|
+ case '`': case '|':
|
|
|
+ goto needs_quoting;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (p->quote_these_too[c / INT_BITS] & (1 << (c % INT_BITS)))
|
|
|
+ goto needs_quoting;
|
|
|
+
|
|
|
+ STORE (c);
|
|
|
+ }
|
|
|
+
|
|
|
+ needs_quoting:;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* Fall through. */
|
|
|
+
|
|
|
+ case shell_always_quoting_style:
|
|
|
+ quote_mark = '\'';
|
|
|
+ break;
|
|
|
+
|
|
|
+ case c_quoting_style:
|
|
|
+ quote_mark = '"';
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ quote_mark = 0;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ len = 0;
|
|
|
+
|
|
|
+ if (quote_mark)
|
|
|
+ STORE (quote_mark);
|
|
|
+
|
|
|
+ for (i = 0; ! (argsize == (size_t) -1 ? arg[i] == '\0' : i == argsize); i++)
|
|
|
+ {
|
|
|
+ c = arg[i];
|
|
|
+
|
|
|
+ switch (quoting_style)
|
|
|
+ {
|
|
|
+ case literal_quoting_style:
|
|
|
+ break;
|
|
|
+
|
|
|
+ case shell_quoting_style:
|
|
|
+ case shell_always_quoting_style:
|
|
|
+ if (c == '\'')
|
|
|
+ {
|
|
|
+ STORE ('\'');
|
|
|
+ STORE ('\\');
|
|
|
+ STORE ('\'');
|
|
|
+ }
|
|
|
+ break;
|
|
|
+
|
|
|
+ case c_quoting_style:
|
|
|
+ case escape_quoting_style:
|
|
|
+ switch (c)
|
|
|
+ {
|
|
|
+ case '?': /* Do not generate trigraphs. */
|
|
|
+ case '\\': goto store_escape;
|
|
|
+ /* Not all C compilers know what \a means. */
|
|
|
+ case 7 : c = 'a'; goto store_escape;
|
|
|
+ case '\b': c = 'b'; goto store_escape;
|
|
|
+ case '\f': c = 'f'; goto store_escape;
|
|
|
+ case '\n': c = 'n'; goto store_escape;
|
|
|
+ case '\r': c = 'r'; goto store_escape;
|
|
|
+ case '\t': c = 't'; goto store_escape;
|
|
|
+ case '\v': c = 'v'; goto store_escape;
|
|
|
+
|
|
|
+ case '"':
|
|
|
+ if (quoting_style == c_quoting_style)
|
|
|
+ goto store_escape;
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ if (!ISGRAPH (c))
|
|
|
+ {
|
|
|
+ STORE ('\\');
|
|
|
+ STORE ('0' + (c >> 6));
|
|
|
+ STORE ('0' + ((c >> 3) & 7));
|
|
|
+ c = '0' + (c & 7);
|
|
|
+ goto store_c;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (! (p->quote_these_too[c / INT_BITS] & (1 << (c % INT_BITS))))
|
|
|
+ goto store_c;
|
|
|
+
|
|
|
+ store_escape:
|
|
|
+ STORE ('\\');
|
|
|
+ }
|
|
|
+
|
|
|
+ store_c:
|
|
|
+ STORE (c);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (quote_mark)
|
|
|
+ STORE (quote_mark);
|
|
|
+
|
|
|
+ done:
|
|
|
+ if (len < buffersize)
|
|
|
+ buffer[len] = '\0';
|
|
|
+ return len;
|
|
|
+}
|
|
|
+
|
|
|
+/* Use storage slot N to return a quoted version of the string ARG.
|
|
|
+ OPTIONS specifies the quoting options.
|
|
|
+ The returned value points to static storage that can be
|
|
|
+ reused by the next call to this function with the same value of N.
|
|
|
+ N must be nonnegative. N is deliberately declared with type `int'
|
|
|
+ to allow for future extensions (using negative values). */
|
|
|
+static char *
|
|
|
+quotearg_n_options (int n, char const *arg,
|
|
|
+ struct quoting_options const *options)
|
|
|
+{
|
|
|
+ static unsigned int nslots;
|
|
|
+ static struct slotvec
|
|
|
+ {
|
|
|
+ size_t size;
|
|
|
+ char *val;
|
|
|
+ } *slotvec;
|
|
|
+
|
|
|
+ if (nslots <= n)
|
|
|
+ {
|
|
|
+ int n1 = n + 1;
|
|
|
+ size_t s = n1 * sizeof (struct slotvec);
|
|
|
+ if (! (0 < n1 && n1 == s / sizeof (struct slotvec)))
|
|
|
+ abort ();
|
|
|
+ slotvec = (struct slotvec *) xrealloc (slotvec, s);
|
|
|
+ memset (slotvec + nslots, 0, (n1 - nslots) * sizeof (struct slotvec));
|
|
|
+ nslots = n;
|
|
|
+ }
|
|
|
+
|
|
|
+ {
|
|
|
+ size_t size = slotvec[n].size;
|
|
|
+ char *val = slotvec[n].val;
|
|
|
+ size_t qsize = quotearg_buffer (val, size, arg, (size_t) -1, options);
|
|
|
+
|
|
|
+ if (size <= qsize)
|
|
|
+ {
|
|
|
+ slotvec[n].size = size = qsize + 1;
|
|
|
+ slotvec[n].val = val = xrealloc (val, size);
|
|
|
+ quotearg_buffer (val, size, arg, (size_t) -1, options);
|
|
|
+ }
|
|
|
+
|
|
|
+ return val;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+char *
|
|
|
+quotearg_n (unsigned int n, char const *arg)
|
|
|
+{
|
|
|
+ return quotearg_n_options (n, arg, &default_quoting_options);
|
|
|
+}
|
|
|
+
|
|
|
+char *
|
|
|
+quotearg (char const *arg)
|
|
|
+{
|
|
|
+ return quotearg_n (0, arg);
|
|
|
+}
|
|
|
+
|
|
|
+char *
|
|
|
+quotearg_char (char const *arg, char ch)
|
|
|
+{
|
|
|
+ struct quoting_options options;
|
|
|
+ options = default_quoting_options;
|
|
|
+ set_char_quoting (&options, ch, 1);
|
|
|
+ return quotearg_n_options (0, arg, &options);
|
|
|
+}
|
|
|
+
|
|
|
+char *
|
|
|
+quotearg_colon (char const *arg)
|
|
|
+{
|
|
|
+ return quotearg_char (arg, ':');
|
|
|
+}
|