utils.cpp 4.7 KB

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