utils.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <assert.h>
  5. #include "utils.h"
  6. // http://c-faq.com/lib/randrange.html
  7. int randint(int N) { return rand() / (RAND_MAX / N + 1); }
  8. // http://c-faq.com/lib/randrange.html
  9. // numbers in the range [M, N] could be generated with something like
  10. int randrange(int M, int N) {
  11. return M + rand() / (RAND_MAX / (N - M + 1) + 1);
  12. }
  13. /**
  14. * Display a repr of the given string.
  15. *
  16. * This converts most \n\r\v\f\t codes,
  17. * defaults to \xHH (hex value).
  18. */
  19. char *repr(const char *data) {
  20. static char buffer[40960];
  21. char *cp;
  22. strcpy(buffer, data);
  23. cp = buffer;
  24. while (*cp != 0) {
  25. char c = *cp;
  26. if (c == ' ') {
  27. cp++;
  28. continue;
  29. };
  30. /* Ok, it's form-feed ('\f'), newline ('\n'), carriage return ('\r'),
  31. * horizontal tab ('\t'), and vertical tab ('\v') */
  32. if (strchr("\f\n\r\t\v\?", c) != NULL) {
  33. memmove(cp + 1, cp, strlen(cp) + 1);
  34. *cp = '\\';
  35. cp++;
  36. switch (c) {
  37. case '\f':
  38. *cp = 'f';
  39. cp++;
  40. break;
  41. case '\n':
  42. *cp = 'n';
  43. cp++;
  44. break;
  45. case '\r':
  46. *cp = 'r';
  47. cp++;
  48. break;
  49. case '\t':
  50. *cp = 't';
  51. cp++;
  52. break;
  53. case '\v':
  54. *cp = 'v';
  55. cp++;
  56. break;
  57. default:
  58. *cp = '?';
  59. cp++;
  60. break;
  61. }
  62. continue;
  63. }
  64. if (c == '\\') {
  65. memmove(cp + 1, cp, strlen(cp) + 1);
  66. *cp = '\\';
  67. cp += 2;
  68. continue;
  69. }
  70. if (c == '"') {
  71. memmove(cp + 1, cp, strlen(cp) + 1);
  72. *cp = '\\';
  73. cp += 2;
  74. continue;
  75. }
  76. if (strchr("[()]{}:;,.<>?!@#$%^&*", c) != NULL) {
  77. cp++;
  78. continue;
  79. }
  80. if (strchr("0123456789", c) != NULL) {
  81. cp++;
  82. continue;
  83. }
  84. if (strchr("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", c) !=
  85. NULL) {
  86. cp++;
  87. continue;
  88. }
  89. // Ok, default to \xHH output.
  90. memmove(cp + 3, cp, strlen(cp) + 1);
  91. char buffer[10];
  92. int slen;
  93. slen = snprintf(buffer, sizeof(buffer), "\\x%02x", (int)c & 0xff);
  94. /*
  95. if (slen >= sizeof(buffer)) {
  96. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  97. }
  98. */
  99. strncpy(cp, buffer, 4);
  100. cp += 4;
  101. continue;
  102. }
  103. return buffer;
  104. }
  105. /*
  106. * strnstr()
  107. *
  108. * buffer safe version that looks for a string.
  109. */
  110. const char *strnstr(const char *source, int len, const char *needle) {
  111. int pos;
  112. for (pos = 0; pos < len; pos++) {
  113. if (source[pos] == needle[0]) {
  114. if (strncmp(source + pos, needle, strlen(needle)) == 0) {
  115. return source + pos;
  116. }
  117. }
  118. }
  119. return NULL;
  120. }
  121. /*
  122. * rstrnstr() Reverse string find in a string
  123. *
  124. * This obeys the len, and handles nulls in buffer.
  125. * find is a c-string (null terminated)
  126. */
  127. int rstrnstr(const char *buffer, int len, const char *find) {
  128. int flen = strlen(find);
  129. if (len < flen) {
  130. // I can't find a string in a buffer smaller then it is!
  131. return -1;
  132. }
  133. int pos = len - flen;
  134. while (pos > 0) {
  135. if (buffer[pos] == find[0]) {
  136. // First chars match, check them all.
  137. if (strncmp(buffer + pos, find, flen) == 0) {
  138. return pos;
  139. }
  140. }
  141. pos--;
  142. }
  143. return -1;
  144. }
  145. /*
  146. * string_insert()
  147. * Inserts a string into a given position.
  148. * This safely checks to make sure the buffer isn't overrun.
  149. */
  150. int string_insert(char *buffer, int max_length, int pos, const char *insert) {
  151. assert(max_length > pos);
  152. assert(buffer != NULL);
  153. assert(insert != NULL);
  154. if (strlen(buffer) + strlen(insert) >= max_length) {
  155. /*
  156. ZF_LOGD("string_insert() failed inserting [%s]", repr(insert));
  157. */
  158. return 0;
  159. }
  160. memmove(buffer + pos + strlen(insert), buffer + pos,
  161. strlen(buffer + pos) + 1);
  162. // cp + strlen(display), cp, strlen(cp) + 1);
  163. strncpy(buffer + pos, insert, strlen(insert));
  164. // (cp, display, strlen(display));
  165. return 1; // success
  166. }
  167. /*
  168. Pascal String Copy. Copy from pascal string, to C String.
  169. First char is pascal string length. (Max 255).
  170. */
  171. void pcopy(char *pstring, char *str) {
  172. int len = (int)*pstring;
  173. strncpy(str, pstring + 1, len);
  174. str[len] = 0;
  175. }