utils.cpp 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. #include <fstream>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <string>
  6. #include "utils.h"
  7. // http://c-faq.com/lib/randrange.html
  8. int randint(int N) { return rand() / (RAND_MAX / N + 1); }
  9. // http://c-faq.com/lib/randrange.html
  10. // numbers in the range [M, N] could be generated with something like
  11. int randrange(int M, int N) {
  12. return M + rand() / (RAND_MAX / (N - M + 1) + 1);
  13. }
  14. /**
  15. * random_activate()
  16. *
  17. * Is a weight (1-10),
  18. * tests if random number is < weight * 10.
  19. *
  20. * So random_activate(9) happens more frequently
  21. * then random_activate(8) or lower.
  22. *
  23. * This probably needs to be fixed.
  24. * We need a better randint(RANGE) code.
  25. */
  26. int random_activate(int w) {
  27. int r = randint(100);
  28. if (r <= (w * 10)) {
  29. return 1;
  30. };
  31. return 0;
  32. }
  33. /**
  34. * Display a repr of the given string.
  35. *
  36. * This converts most \n\r\v\f\t codes,
  37. * defaults to \xHH (hex value).
  38. */
  39. char *repr(const char *data) {
  40. static char buffer[40960];
  41. char *cp;
  42. strcpy(buffer, data);
  43. cp = buffer;
  44. while (*cp != 0) {
  45. char c = *cp;
  46. if (c == ' ') {
  47. cp++;
  48. continue;
  49. };
  50. /* Ok, it's form-feed ('\f'), newline ('\n'), carriage return ('\r'),
  51. * horizontal tab ('\t'), and vertical tab ('\v') */
  52. if (strchr("\f\n\r\t\v\?", c) != NULL) {
  53. memmove(cp + 1, cp, strlen(cp) + 1);
  54. *cp = '\\';
  55. cp++;
  56. switch (c) {
  57. case '\f':
  58. *cp = 'f';
  59. cp++;
  60. break;
  61. case '\n':
  62. *cp = 'n';
  63. cp++;
  64. break;
  65. case '\r':
  66. *cp = 'r';
  67. cp++;
  68. break;
  69. case '\t':
  70. *cp = 't';
  71. cp++;
  72. break;
  73. case '\v':
  74. *cp = 'v';
  75. cp++;
  76. break;
  77. default:
  78. *cp = '?';
  79. cp++;
  80. break;
  81. }
  82. continue;
  83. }
  84. if (c == '\\') {
  85. memmove(cp + 1, cp, strlen(cp) + 1);
  86. *cp = '\\';
  87. cp += 2;
  88. continue;
  89. }
  90. if (c == '"') {
  91. memmove(cp + 1, cp, strlen(cp) + 1);
  92. *cp = '\\';
  93. cp += 2;
  94. continue;
  95. }
  96. if (strchr("[()]{}:;,.<>?!@#$%^&*", c) != NULL) {
  97. cp++;
  98. continue;
  99. }
  100. if (strchr("0123456789", c) != NULL) {
  101. cp++;
  102. continue;
  103. }
  104. if (strchr("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", c) !=
  105. NULL) {
  106. cp++;
  107. continue;
  108. }
  109. // Ok, default to \xHH output.
  110. memmove(cp + 3, cp, strlen(cp) + 1);
  111. char buffer[10];
  112. int slen;
  113. slen = snprintf(buffer, sizeof(buffer), "\\x%02x", (int)c & 0xff);
  114. /*
  115. if (slen >= sizeof(buffer)) {
  116. ZF_LOGE("snprintf %d > size %d", slen, (int)sizeof(buffer));
  117. }
  118. */
  119. strncpy(cp, buffer, 4);
  120. cp += 4;
  121. continue;
  122. }
  123. return buffer;
  124. }
  125. /*
  126. * strnstr()
  127. *
  128. * buffer safe version that looks for a string.
  129. */
  130. const char *strnstr(const char *source, int len, const char *needle) {
  131. int pos;
  132. for (pos = 0; pos < len; pos++) {
  133. if (source[pos] == needle[0]) {
  134. if (strncmp(source + pos, needle, strlen(needle)) == 0) {
  135. return source + pos;
  136. }
  137. }
  138. }
  139. return NULL;
  140. }
  141. /*
  142. * rstrnstr() Reverse string find in a string
  143. *
  144. * This obeys the len, and handles nulls in buffer.
  145. * find is a c-string (null terminated)
  146. */
  147. int rstrnstr(const char *buffer, int len, const char *find) {
  148. int flen = strlen(find);
  149. if (len < flen) {
  150. // I can't find a string in a buffer smaller then it is!
  151. return -1;
  152. }
  153. int pos = len - flen;
  154. while (pos > 0) {
  155. if (buffer[pos] == find[0]) {
  156. // First chars match, check them all.
  157. if (strncmp(buffer + pos, find, flen) == 0) {
  158. return pos;
  159. }
  160. }
  161. pos--;
  162. }
  163. return -1;
  164. }
  165. /*
  166. * string_insert()
  167. * Inserts a string into a given position.
  168. * This safely checks to make sure the buffer isn't overrun.
  169. *
  170. * buffer is a c null terminated string.
  171. */
  172. int string_insert(char *buffer, int max_length, int pos, const char *insert) {
  173. /*
  174. assert(max_length > pos);
  175. assert(buffer != NULL);
  176. assert(insert != NULL);
  177. */
  178. if (pos >= max_length)
  179. return 0;
  180. if (buffer == NULL)
  181. return 0;
  182. if (insert == NULL)
  183. return 0;
  184. if (strlen(insert) == 0)
  185. return 0;
  186. if (pos > strlen(buffer))
  187. return 0;
  188. if (strlen(buffer) + strlen(insert) >= max_length) {
  189. /*
  190. ZF_LOGD("string_insert() failed inserting [%s]", repr(insert));
  191. */
  192. return 0;
  193. }
  194. memmove(buffer + pos + strlen(insert), buffer + pos,
  195. strlen(buffer + pos) + 1);
  196. // cp + strlen(display), cp, strlen(cp) + 1);
  197. strncpy(buffer + pos, insert, strlen(insert));
  198. // (cp, display, strlen(display));
  199. return 1; // success
  200. }
  201. /*
  202. Pascal String Copy. Copy from pascal string, to C String.
  203. First char is pascal string length. (Max 255).
  204. */
  205. void pcopy(char *pstring, char *str) {
  206. int len = (int)*pstring;
  207. strncpy(str, pstring + 1, len);
  208. str[len] = 0;
  209. }
  210. /*
  211. * tail file, return new lines of text.
  212. *
  213. * Only handles lines < 256 chars.
  214. * Does not handle if the file is closed/unlinked/...
  215. *
  216. */
  217. std::string &find_new_text(std::ifstream &infile,
  218. std::streampos &last_position) {
  219. static std::string line;
  220. line.clear();
  221. infile.seekg(0, std::ios::end);
  222. std::streampos filesize = infile.tellg();
  223. if (filesize == -1) {
  224. // Ok, we've failed somehow. Now what?
  225. // cout << "SNAP!";
  226. return line;
  227. }
  228. // check if the new file started
  229. // we don't detect if the file has been unlinked/reset
  230. if (filesize < last_position) {
  231. // cout << "reset! " << filesize << endl;
  232. last_position = 0;
  233. }
  234. while (last_position < filesize) {
  235. // this loop -- seems broken.
  236. // for (long n = last_position; n < filesize; n++) {
  237. infile.seekg(last_position, std::ios::beg);
  238. char test[256];
  239. infile.getline(test, sizeof(test));
  240. if (infile.eof()) {
  241. // We got EOF instead of something.
  242. // Seek back to our last, good, known position
  243. // and exit (wait for the rest of the line)
  244. infile.seekg(last_position, std::ios::beg);
  245. return line;
  246. }
  247. std::streampos new_pos = infile.tellg();
  248. if (new_pos == -1)
  249. return line;
  250. last_position = new_pos;
  251. line.assign(test);
  252. return line;
  253. }
  254. return line;
  255. }