123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 |
- /* 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, ':');
- }
|