iron_string.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. #include "iron_string.h"
  2. #include <stdlib.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. #include <ctype.h>
  6. void *gc_alloc(size_t size);
  7. void *gc_leaf(void *ptr);
  8. char *string_alloc(int size) {
  9. char *r = gc_alloc(size);
  10. gc_leaf(r);
  11. return r;
  12. }
  13. char *string_join(char *a, char *b) {
  14. char *r = string_alloc(strlen(a) + strlen(b) + 1);
  15. strcpy(r, a);
  16. strcat(r, b);
  17. return r;
  18. }
  19. char *string_copy(char *a) {
  20. if (a == NULL) {
  21. return NULL;
  22. }
  23. char *r = string_alloc(strlen(a) + 1);
  24. strcpy(r, a);
  25. return r;
  26. }
  27. int string_length(char *str) {
  28. return strlen(str);
  29. }
  30. bool string_equals(char *a, char *b) {
  31. if (a == NULL || b == NULL) {
  32. return false;
  33. }
  34. return strcmp(a, b) == 0;
  35. }
  36. char *i32_to_string(int32_t i) {
  37. int l = snprintf(NULL, 0, "%d", i);
  38. char *r = string_alloc(l + 1);
  39. sprintf(r, "%d", i);
  40. return r;
  41. }
  42. char *i32_to_string_hex(int32_t i) {
  43. int l = snprintf(NULL, 0, "%X", i);
  44. char *r = string_alloc(l + 1);
  45. sprintf(r, "%X", i);
  46. return r;
  47. }
  48. char *i64_to_string(int64_t i) {
  49. int l = snprintf(NULL, 0, "%ld", i);
  50. char *r = string_alloc(l + 1);
  51. sprintf(r, "%ld", i);
  52. return r;
  53. }
  54. char *u64_to_string(uint64_t i) {
  55. int l = snprintf(NULL, 0, "%lu", i);
  56. char *r = string_alloc(l + 1);
  57. sprintf(r, "%lu", i);
  58. return r;
  59. }
  60. char *f32_to_string_with_zeros(float f) {
  61. int l = snprintf(NULL, 0, "%f", f);
  62. char *r = string_alloc(l + 1);
  63. sprintf(r, "%f", f);
  64. return r;
  65. }
  66. char *f32_to_string(float f) {
  67. char *r = f32_to_string_with_zeros(f);
  68. string_strip_trailing_zeros(r);
  69. return r;
  70. }
  71. void string_strip_trailing_zeros(char *str) {
  72. int len = strlen(str);
  73. while (str[--len] == '0') {
  74. str[len] = '\0';
  75. }
  76. if (str[len] == '.') {
  77. str[len] = '\0';
  78. }
  79. }
  80. int32_t string_index_of_pos(char *s, char *search, int pos) {
  81. char *found = strstr(s + pos, search);
  82. if (found != NULL) {
  83. return found - s;
  84. }
  85. return -1;
  86. }
  87. int32_t string_index_of(char *s, char *search) {
  88. return string_index_of_pos(s, search, 0);
  89. }
  90. int32_t string_last_index_of(char *str, char *search) {
  91. char *s = str;
  92. char *found = NULL;
  93. while (1) {
  94. char *p = strstr(s, search);
  95. if (p == NULL) {
  96. break;
  97. }
  98. found = p;
  99. s = p + 1;
  100. }
  101. if (found != NULL) {
  102. return found - str;
  103. }
  104. return -1;
  105. }
  106. any_array_t *string_split(char *s, char *sep) {
  107. char *r = string_alloc(strlen(s) + 1); // Modified by strtok
  108. any_array_t *a = gc_alloc(sizeof(any_array_t));
  109. strcpy(r, s);
  110. int count = 0;
  111. char *token = strtok(r, sep);
  112. while (token != NULL) {
  113. count++;
  114. token = strtok(NULL, sep);
  115. }
  116. any_array_resize(a, count);
  117. strcpy(r, s);
  118. token = strtok(r, sep);
  119. while (token != NULL) {
  120. any_array_push(a, string_copy(token)); // Make a copy to keep gc ref
  121. // TODO: this works only for single char sep
  122. token = strtok(NULL, sep);
  123. }
  124. if (a->length == 0) { // Separator not found
  125. any_array_push(a, r);
  126. }
  127. return a;
  128. }
  129. char *string_array_join(any_array_t *a, char *separator) {
  130. int len = 0;
  131. int len_sep = strlen(separator);
  132. for (int i = 0; i < a->length; ++i) {
  133. len += strlen(a->buffer[i]);
  134. if (i < a->length - 1) {
  135. len += len_sep;
  136. }
  137. }
  138. char *r = string_alloc(len + 1);
  139. for (int i = 0; i < a->length; ++i) {
  140. strcat(r, a->buffer[i]);
  141. if (i < a->length - 1) {
  142. strcat(r, separator);
  143. }
  144. }
  145. return r;
  146. }
  147. char *string_replace_all(char *s, char *search, char *replace) {
  148. char *buffer = string_alloc(1024);
  149. char *buffer_pos = buffer;
  150. size_t search_len = strlen(search);
  151. size_t replace_len = strlen(replace);
  152. while (1) {
  153. char *p = strstr(s, search);
  154. if (p == NULL) {
  155. strcpy(buffer_pos, s);
  156. break;
  157. }
  158. memcpy(buffer_pos, s, p - s);
  159. buffer_pos += p - s;
  160. memcpy(buffer_pos, replace, replace_len);
  161. buffer_pos += replace_len;
  162. s = p + search_len;
  163. }
  164. return buffer;
  165. }
  166. char *substring(char *s, int32_t start, int32_t end) {
  167. char *buffer = string_alloc(end - start + 1);
  168. for (int i = 0; i < end - start; ++i) {
  169. buffer[i] = s[start + i];
  170. }
  171. return buffer;
  172. }
  173. char *string_from_char_code(int32_t c) {
  174. char *r = string_alloc(2);
  175. r[0] = c;
  176. r[1] = '\0';
  177. return r;
  178. }
  179. int32_t char_code_at(char *s, int32_t i) {
  180. return s[i];
  181. }
  182. char *char_at(char *s, int32_t i) {
  183. char *r = string_alloc(2);
  184. r[0] = s[i];
  185. r[1] = '\0';
  186. return r;
  187. }
  188. bool starts_with(char *s, char *start) {
  189. return strncmp(start, s, strlen(start)) == 0;
  190. }
  191. bool ends_with(char *s, char *end) {
  192. size_t len_s = strlen(s);
  193. size_t len_end = strlen(end);
  194. return strncmp(s + len_s - len_end, end, len_end) == 0;
  195. }
  196. char *to_lower_case(char *s) {
  197. char *r = string_alloc(strlen(s) + 1);
  198. strcpy(r, s);
  199. int len = string_length(r);
  200. for (int i = 0; i < len; ++i) {
  201. r[i] = tolower(r[i]);
  202. }
  203. return r;
  204. }
  205. char *to_upper_case(char *s) {
  206. char *r = string_alloc(strlen(s) + 1);
  207. strcpy(r, s);
  208. int len = string_length(r);
  209. for (int i = 0; i < len; ++i) {
  210. r[i] = toupper(r[i]);
  211. }
  212. return r;
  213. }
  214. char *trim_end(char *str) {
  215. int pos = string_length(str) - 1;
  216. while (pos >= 0 && (str[pos] == ' ' || str[pos] == '\n' || str[pos] == '\r')) {
  217. pos--;
  218. }
  219. return substring(str, 0, pos + 1);
  220. }
  221. // Per Lowgren, CC BY-SA 3.0
  222. // https://stackoverflow.com/a/35332046
  223. #define is_unicode(c) (((c) & 0xc0) == 0xc0)
  224. int string_utf8_decode(const char *str, int *i) {
  225. const unsigned char *s = (const unsigned char *)str;
  226. int u = *s, l = 1;
  227. if (is_unicode(u)) {
  228. int a = (u & 0x20) ? ((u & 0x10) ? ((u & 0x08) ? ((u & 0x04) ? 6 : 5) : 4) : 3) : 2;
  229. if (a < 6 || !(u & 0x02)) {
  230. int b, p = 0;
  231. u = ((u << (a + 1)) & 0xff) >> (a + 1);
  232. for (b = 1; b < a; ++b)
  233. u = (u << 6) | (s[l++] & 0x3f);
  234. }
  235. }
  236. if (i) *i += l;
  237. return u;
  238. }