cstring.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. /* Copyright (c) 2001 Matej Pfajfar.
  2. * Copyright (c) 2001-2004, Roger Dingledine.
  3. * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
  4. * Copyright (c) 2007-2018, The Tor Project, Inc. */
  5. /* See LICENSE for licensing information */
  6. #include "lib/encoding/cstring.h"
  7. #include "lib/log/torlog.h"
  8. #include "lib/log/util_bug.h"
  9. #include "lib/malloc/util_malloc.h"
  10. #include "lib/string/compat_ctype.h"
  11. #include <string.h>
  12. #define TOR_ISODIGIT(c) ('0' <= (c) && (c) <= '7')
  13. /** Given a c-style double-quoted escaped string in <b>s</b>, extract and
  14. * decode its contents into a newly allocated string. On success, assign this
  15. * string to *<b>result</b>, assign its length to <b>size_out</b> (if
  16. * provided), and return a pointer to the position in <b>s</b> immediately
  17. * after the string. On failure, return NULL.
  18. */
  19. const char *
  20. unescape_string(const char *s, char **result, size_t *size_out)
  21. {
  22. const char *cp;
  23. char *out;
  24. if (s[0] != '\"')
  25. return NULL;
  26. cp = s+1;
  27. while (1) {
  28. switch (*cp) {
  29. case '\0':
  30. case '\n':
  31. return NULL;
  32. case '\"':
  33. goto end_of_loop;
  34. case '\\':
  35. if (cp[1] == 'x' || cp[1] == 'X') {
  36. if (!(TOR_ISXDIGIT(cp[2]) && TOR_ISXDIGIT(cp[3])))
  37. return NULL;
  38. cp += 4;
  39. } else if (TOR_ISODIGIT(cp[1])) {
  40. cp += 2;
  41. if (TOR_ISODIGIT(*cp)) ++cp;
  42. if (TOR_ISODIGIT(*cp)) ++cp;
  43. } else if (cp[1] == 'n' || cp[1] == 'r' || cp[1] == 't' || cp[1] == '"'
  44. || cp[1] == '\\' || cp[1] == '\'') {
  45. cp += 2;
  46. } else {
  47. return NULL;
  48. }
  49. break;
  50. default:
  51. ++cp;
  52. break;
  53. }
  54. }
  55. end_of_loop:
  56. out = *result = tor_malloc(cp-s + 1);
  57. cp = s+1;
  58. while (1) {
  59. switch (*cp)
  60. {
  61. case '\"':
  62. *out = '\0';
  63. if (size_out) *size_out = out - *result;
  64. return cp+1;
  65. /* LCOV_EXCL_START -- we caught this in parse_config_from_line. */
  66. case '\0':
  67. tor_fragile_assert();
  68. tor_free(*result);
  69. return NULL;
  70. /* LCOV_EXCL_STOP */
  71. case '\\':
  72. switch (cp[1])
  73. {
  74. case 'n': *out++ = '\n'; cp += 2; break;
  75. case 'r': *out++ = '\r'; cp += 2; break;
  76. case 't': *out++ = '\t'; cp += 2; break;
  77. case 'x': case 'X':
  78. {
  79. int x1, x2;
  80. x1 = hex_decode_digit(cp[2]);
  81. x2 = hex_decode_digit(cp[3]);
  82. if (x1 == -1 || x2 == -1) {
  83. /* LCOV_EXCL_START */
  84. /* we caught this above in the initial loop. */
  85. tor_assert_nonfatal_unreached();
  86. tor_free(*result);
  87. return NULL;
  88. /* LCOV_EXCL_STOP */
  89. }
  90. *out++ = ((x1<<4) + x2);
  91. cp += 4;
  92. }
  93. break;
  94. case '0': case '1': case '2': case '3': case '4': case '5':
  95. case '6': case '7':
  96. {
  97. int n = cp[1]-'0';
  98. cp += 2;
  99. if (TOR_ISODIGIT(*cp)) { n = n*8 + *cp-'0'; cp++; }
  100. if (TOR_ISODIGIT(*cp)) { n = n*8 + *cp-'0'; cp++; }
  101. if (n > 255) { tor_free(*result); return NULL; }
  102. *out++ = (char)n;
  103. }
  104. break;
  105. case '\'':
  106. case '\"':
  107. case '\\':
  108. case '\?':
  109. *out++ = cp[1];
  110. cp += 2;
  111. break;
  112. /* LCOV_EXCL_START */
  113. default:
  114. /* we caught this above in the initial loop. */
  115. tor_assert_nonfatal_unreached();
  116. tor_free(*result); return NULL;
  117. /* LCOV_EXCL_STOP */
  118. }
  119. break;
  120. default:
  121. *out++ = *cp++;
  122. }
  123. }
  124. }