++fci;
if (fci->format_chars == 0)
{
- if (ISGRAPH (format_char))
- format_warning_at_char (format_string_loc, format_string_cst,
- format_chars - orig_format_chars,
- OPT_Wformat_,
- "unknown conversion type character"
- " %qc in format",
- format_char);
- else
- format_warning_at_char (format_string_loc, format_string_cst,
- format_chars - orig_format_chars,
- OPT_Wformat_,
- "unknown conversion type character"
- " 0x%x in format",
- format_char);
+ format_warning_at_char (format_string_loc, format_string_cst,
+ format_chars - orig_format_chars,
+ OPT_Wformat_,
+ "unknown conversion type character"
+ " %qc in format",
+ format_char);
return NULL;
}
#include <iconv.h>
#endif
+static void pp_quoted_string (pretty_printer *, const char *, size_t = -1);
+
/* Overwrite the given location/range within this text_info's rich_location.
For use e.g. when implementing "+" in client format decoders. */
break;
case 'c':
- pp_character (pp, va_arg (*text->args_ptr, int));
- break;
+ {
+ /* When quoting, print alphanumeric, punctuation, and the space
+ character unchanged, and all others in hexadecimal with the
+ "\x" prefix. Otherwise print them all unchanged. */
+ int chr = va_arg (*text->args_ptr, int);
+ if (ISPRINT (chr) || !quote)
+ pp_character (pp, chr);
+ else
+ {
+ const char str [2] = { chr, '\0' };
+ pp_quoted_string (pp, str, 1);
+ }
+ break;
+ }
case 'd':
case 'i':
break;
case 's':
- pp_string (pp, va_arg (*text->args_ptr, const char *));
+ if (quote)
+ pp_quoted_string (pp, va_arg (*text->args_ptr, const char *));
+ else
+ pp_string (pp, va_arg (*text->args_ptr, const char *));
break;
case 'p':
pp_maybe_wrap_text (pp, str, str + strlen (str));
}
+/* Append the leading N characters of STRING to the output area of
+ PRETTY-PRINTER, quoting in hexadecimal non-printable characters.
+ Setting N = -1 is as if N were set to strlen (STRING). The STRING
+ may be line-wrapped if in appropriate mode. */
+static void
+pp_quoted_string (pretty_printer *pp, const char *str, size_t n /* = -1 */)
+{
+ gcc_checking_assert (str);
+
+ const char *last = str;
+ const char *ps;
+
+ /* Compute the length if not specified. */
+ if (n == (size_t) -1)
+ n = strlen (str);
+
+ for (ps = str; n; ++ps, --n)
+ {
+ if (ISPRINT (*ps))
+ continue;
+
+ if (last < ps)
+ pp_maybe_wrap_text (pp, last, ps - 1);
+
+ /* Append the hexadecimal value of the character. Allocate a buffer
+ that's large enough for a 32-bit char plus the hex prefix. */
+ char buf [11];
+ int n = sprintf (buf, "\\x%02x", (unsigned char)*ps);
+ pp_maybe_wrap_text (pp, buf, buf + n);
+ last = ps + 1;
+ }
+
+ pp_maybe_wrap_text (pp, last, ps);
+}
+
/* Maybe print out a whitespace if needed. */
void