-
Notifications
You must be signed in to change notification settings - Fork 173
/
Copy pathcarp_string.h
391 lines (345 loc) · 9.34 KB
/
carp_string.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
#include "carp_utf8.h"
String String_allocate(int len, char byte) {
/* Allocate a string of length 'len + 1'
* setting the first len bytes to byte
* and adding a null terminator
*
* String_alloc(10, "a") == "aaaaaaaaaa"
*/
String ptr = CARP_MALLOC(len + 1);
if (ptr != NULL) {
// calling memset(NULL,...) would exercise undefined behaviour...
memset(ptr, byte, len);
ptr[len] = '\0';
}
return ptr;
}
void String_delete(String s) {
CARP_FREE(s);
}
void String_string_MINUS_set_BANG_(String *s, int i, char ch) {
CHK_INDEX(i, strlen(*s));
(*s)[i] = ch;
}
void String_string_MINUS_set_MINUS_at_BANG_(String *into, int i,
const String *src) {
char *dest = (*into) + i;
size_t lsrc = strlen(*src);
/* given a string and indices
*
* 0 1 2 3 4 5 6 7 8 9
* "a b c d e f g h i j"
* linto = strlen(...) = 10
*
* if we want to insert at '6' a string of length '4'
*
* 0 1 2 3
* "w x y z"
* lsrc = strlen(...) = 4
*
* we need to make sure that the new string will not grow the first
*
* 0 1 2 3 4 5 6 7 8 9
* "a b c d e f g h i j"
* ^
* |
* 0 1 2 3
* "w x y z"
*
* we check this by
* (i + lsrc) < (linto + 1)
* (6 + 4) < (10 + 1)
* 10 < 11
* true
*
* so this write is safe
*/
CHK_INDEX(i + lsrc, strlen(*into) + 1);
memcpy(dest, *src, lsrc);
}
String String_copy(const String *s) {
size_t len = strlen(*s) + 1;
String ptr = CARP_MALLOC(len);
return (String)memcpy(ptr, *s, len);
}
bool String__EQ_(const String *a, const String *b) {
return strcmp(*a, *b) == 0;
}
bool String__GT_(const String *a, const String *b) {
return strcmp(*a, *b) > 0;
}
bool String__LT_(const String *a, const String *b) {
return strcmp(*a, *b) < 0;
}
String String_append(const String *a, const String *b) {
int la = strlen(*a);
int lb = strlen(*b);
int total = la + lb + 1;
String buffer = CARP_MALLOC(total);
memcpy(buffer, *a, la);
memcpy(buffer + la, *b, lb);
buffer[la + lb] = '\0';
return buffer;
}
int String_length(const String *s) {
return strlen(*s);
}
char *String_cstr(const String *s) {
return *s;
}
String String_from_MINUS_cstr(char *s) {
return String_copy(&s);
}
String String_str(const String *s) {
return String_copy(s);
}
int count_occurrences(String s, char c) {
int res = 0;
while (*s != '\0') {
if (*s == c) res++;
s++;
}
return res;
}
String String_prn(const String *s) {
int n = strlen(*s) + 4 + count_occurrences(*s, '"');
String buffer = CARP_MALLOC(n);
buffer[0] = '@';
buffer[1] = '"';
String c = *s;
for (int i = 2; i < n - 2; i++) {
if (*c == '"') buffer[i++] = '\\';
buffer[i] = *c;
c++;
}
buffer[n - 2] = '"';
buffer[n - 1] = '\0';
return buffer;
}
char String_char_MINUS_at(const String *s, int i) {
return (*s)[i];
}
String String_format(const String *str, const String *s) {
int size = snprintf(NULL, 0, *str, *s) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, *s);
return buffer;
}
Array String_chars(const String *ps) {
Char *data;
Array chars;
const char *s = *ps;
const uint8_t *us = (const uint8_t *)s;
uint32_t state = 0;
uint32_t cp = 0;
size_t l = utf8len(s);
chars.len = l;
chars.capacity = l;
data = CARP_MALLOC(chars.capacity * sizeof(*data));
for (size_t si = 0, di = 0; di < l; si++) {
uint32_t r = utf8decode(&state, &cp, us[si]);
switch (r) {
case UTF8_ACCEPT:
data[di++] = cp;
cp = 0;
break;
case UTF8_REJECT:
data[di++] = 0xfffd; // REPLACEMENT CHARACTER
cp = 0;
break;
}
}
chars.data = data;
return chars;
}
String String_from_MINUS_chars(const Array *a) {
Char *data = (Char *)a->data;
size_t cnt = a->len;
size_t sz = wutf8len(data, cnt) + 1;
String s = CARP_MALLOC(sz);
size_t sofar = 0;
for (size_t i = 0; i < cnt; i++) sofar += utf8encode(s + sofar, data[i]);
s[sofar++] = 0;
assert(sofar == sz);
return s;
}
String String_tail(const String *s) {
size_t len = strlen(*s);
String news = CARP_MALLOC(len);
memcpy(news, (*s) + 1, len - 1);
news[len - 1] = '\0';
return news;
}
String String_empty() {
String s = CARP_MALLOC(1);
s[0] = '\0';
return s;
}
Array String_to_MINUS_bytes(const String *s) {
Array chars;
const uint8_t *us = (const uint8_t *)*s;
size_t l = strlen(*s);
chars.len = l;
chars.capacity = l;
chars.data = CARP_MALLOC(chars.capacity);
memcpy(chars.data, us, l);
return chars;
}
String String_from_MINUS_bytes(Array *a) {
String s;
const char *us = (const char *)a->data;
s = CARP_MALLOC(a->len + 1);
memcpy(s, us, a->len);
s[a->len] = '\0';
return s;
}
String Bool_str(bool b) {
const String true_str = "true";
const String false_str = "false";
if (b) {
return String_copy(&true_str);
} else {
return String_copy(&false_str);
}
}
String Bool_format(const String *str, bool b) {
int size = snprintf(NULL, 0, *str, b) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, b);
return buffer;
}
String Char_str(Char c) {
char buf[16];
size_t sz = utf8encode(buf, c);
size_t nsz = sz + 1;
String buffer = CARP_MALLOC(nsz);
memcpy(buffer, buf, nsz);
buffer[nsz - 1] = 0;
return buffer;
}
String Char_prn(Char c) {
char buf[16];
size_t sz = utf8encode(buf, c);
size_t nsz = sz + 1 + 1;
String buffer = CARP_MALLOC(nsz);
buffer[0] = '\\';
memcpy(buffer + 1, buf, sz);
buffer[nsz - 1] = 0;
return buffer;
}
String Char_format(const String *str, char b) {
int size = snprintf(NULL, 0, *str, b) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, b);
return buffer;
}
String Double_str(double x) {
int size = snprintf(NULL, 0, "%g", x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%g", x);
return buffer;
}
String Double_format(const String *s, double x) {
int size = snprintf(NULL, 0, *s, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *s, x);
return buffer;
}
bool Double_from_MINUS_string_MINUS_internal(const String *s, double *target) {
char *err;
*target = strtod(*s, &err);
return *err == 0;
}
String Float_str(float x) {
int size = snprintf(NULL, 0, "%gf", x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%gf", x);
return buffer;
}
String Float_format(const String *str, float x) {
int size = snprintf(NULL, 0, *str, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, x);
return buffer;
}
bool Float_from_MINUS_string_MINUS_internal(const String *s, float *target) {
char *err;
*target = strtof(*s, &err);
return *err == 0;
}
String Int_str(int x) {
int size = snprintf(NULL, 0, "%d", x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%d", x);
return buffer;
}
String Int_format(const String *str, int x) {
int size = snprintf(NULL, 0, *str, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, x);
return buffer;
}
bool Int_from_MINUS_string_MINUS_internal(const String *s, int *target) {
char *err;
*target = (int)strtol(*s, &err, 10);
return *err == 0;
}
String Long_str(Long x) {
int size = snprintf(NULL, 0, "%" PRIi64, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%" PRIi64, x);
return buffer;
}
String Long_format(const String *str, Long x) {
int size = snprintf(NULL, 0, *str, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, x);
return buffer;
}
bool Long_from_MINUS_string_MINUS_internal(const String *s, Long *target) {
char *err;
*target = strtol(*s, &err, 10);
return *err == 0;
}
String Byte_str(uint8_t x) {
int size = snprintf(NULL, 0, "%ub", x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%ub", x);
return buffer;
}
String Byte_format(const String *str, uint8_t x) {
int size = snprintf(NULL, 0, *str, x) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, *str, x);
return buffer;
}
uint8_t Byte_from_MINUS_string_MINUS_internal(const String *s, byte *target) {
char *err;
*target = (uint8_t)strtol(*s, &err, 10);
return *err == 0;
}
int String_index_MINUS_of_MINUS_from(const String *s, char c, int i) {
/* Return index of first occurrence of `c` in `s` AFTER index i
* Returns -1 if not found
*/
++i; // skip first character as we want AFTER i
size_t len = strlen(*s);
for (; i < len; ++i) {
if (c == (*s)[i]) {
return i;
}
}
return -1;
}
int String_index_MINUS_of(const String *s, char c) {
/* Return index of first occurrence of `c` in `s`
* Returns -1 if not found
*/
return String_index_MINUS_of_MINUS_from(s, c, -1);
}
String Pointer_strp(void *in) {
int size = snprintf(NULL, 0, "%p", in) + 1;
String buffer = CARP_MALLOC(size);
sprintf(buffer, "%p", in);
return buffer;
}