Coverage Report

Created: 2025-11-16 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/suricata7/libhtp/htp/bstr.c
Line
Count
Source
1
/***************************************************************************
2
 * Copyright (c) 2009-2010 Open Information Security Foundation
3
 * Copyright (c) 2010-2013 Qualys, Inc.
4
 * All rights reserved.
5
 * 
6
 * Redistribution and use in source and binary forms, with or without
7
 * modification, are permitted provided that the following conditions are
8
 * met:
9
 * 
10
 * - Redistributions of source code must retain the above copyright
11
 *   notice, this list of conditions and the following disclaimer.
12
13
 * - Redistributions in binary form must reproduce the above copyright
14
 *   notice, this list of conditions and the following disclaimer in the
15
 *   documentation and/or other materials provided with the distribution.
16
17
 * - Neither the name of the Qualys, Inc. nor the names of its
18
 *   contributors may be used to endorse or promote products derived from
19
 *   this software without specific prior written permission.
20
 * 
21
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25
 * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32
 ***************************************************************************/
33
34
/**
35
 * @file
36
 * @author Ivan Ristic <ivanr@webkreator.com>
37
 */
38
39
#include <ctype.h>
40
41
#include "bstr.h"
42
43
4.86M
bstr *bstr_alloc(size_t len) {
44
4.86M
    bstr *b = malloc(sizeof (bstr) + len);
45
4.86M
    if (b == NULL) return NULL;
46
47
4.86M
    b->len = 0;
48
4.86M
    b->size = len;
49
4.86M
    b->realptr = NULL;
50
51
4.86M
    return b;
52
4.86M
}
53
54
0
bstr *bstr_add(bstr *destination, const bstr *source) {
55
0
    return bstr_add_mem(destination, bstr_ptr(source), bstr_len(source));
56
0
}
57
58
0
bstr *bstr_add_c(bstr *bdestination, const char *csource) {
59
0
    return bstr_add_mem(bdestination, csource, strlen(csource));
60
0
}
61
62
13.3k
bstr *bstr_add_c_noex(bstr *destination, const char *source) {
63
13.3k
    return bstr_add_mem_noex(destination, source, strlen(source));
64
13.3k
}
65
66
380k
bstr *bstr_add_mem(bstr *destination, const void *data, size_t len) {
67
    // Expand the destination if necessary
68
380k
    if (bstr_size(destination) < bstr_len(destination) + len) {
69
349k
        destination = bstr_expand(destination, bstr_len(destination) + len);
70
349k
        if (destination == NULL) return NULL;
71
349k
    }
72
73
    // Add source to destination
74
380k
    bstr *b = (bstr *) destination;
75
380k
    memcpy(bstr_ptr(destination) + bstr_len(b), data, len);
76
380k
    bstr_adjust_len(b, bstr_len(b) + len);
77
78
380k
    return destination;
79
380k
}
80
81
770k
bstr *bstr_add_mem_noex(bstr *destination, const void *data, size_t len) {
82
770k
    size_t copylen = len;
83
84
    // Is there enough room in the destination?
85
770k
    if (bstr_size(destination) < bstr_len(destination) + copylen) {
86
0
        copylen = bstr_size(destination) - bstr_len(destination);
87
0
        if (copylen <= 0) return destination;
88
0
    }
89
90
    // Copy over the bytes
91
770k
    bstr *b = (bstr *) destination;
92
770k
    memcpy(bstr_ptr(destination) + bstr_len(b), data, copylen);
93
770k
    bstr_adjust_len(b, bstr_len(b) + copylen);
94
95
770k
    return destination;
96
770k
}
97
98
444k
bstr *bstr_add_noex(bstr *destination, const bstr *source) {
99
444k
    return bstr_add_mem_noex(destination, bstr_ptr(source), bstr_len(source));
100
444k
}
101
102
6.12M
void bstr_adjust_len(bstr *b, size_t newlen) {
103
6.12M
    b->len = newlen;
104
6.12M
}
105
106
0
void bstr_adjust_realptr(bstr *b, void *newrealptr) {
107
0
    b->realptr = newrealptr;
108
0
}
109
110
661k
void bstr_adjust_size(bstr *b, size_t newsize) {
111
661k
    b->size = newsize;
112
661k
}
113
114
0
int bstr_begins_with(const bstr *haystack, const bstr *needle) {
115
0
    return bstr_begins_with_mem(haystack, bstr_ptr(needle), bstr_len(needle));
116
0
}
117
118
0
int bstr_begins_with_c(const bstr *haystack, const char *needle) {
119
0
    return bstr_begins_with_mem(haystack, needle, strlen(needle));
120
0
}
121
122
1.24k
int bstr_begins_with_c_nocase(const bstr *haystack, const char *needle) {
123
1.24k
    return bstr_begins_with_mem_nocase(haystack, needle, strlen(needle));
124
1.24k
}
125
126
0
int bstr_begins_with_nocase(const bstr *haystack, const bstr *needle) {
127
0
    return bstr_begins_with_mem_nocase(haystack, bstr_ptr(needle), bstr_len(needle));
128
0
}
129
130
0
int bstr_begins_with_mem(const bstr *haystack, const void *_data, size_t len) {
131
0
    const unsigned char *data = (unsigned char *) _data;
132
0
    const unsigned char *hdata = bstr_ptr(haystack);
133
0
    size_t hlen = bstr_len(haystack);
134
0
    size_t pos = 0;
135
136
0
    while ((pos < len) && (pos < hlen)) {
137
0
        if (hdata[pos] != data[pos]) {
138
0
            return 0;
139
0
        }
140
141
0
        pos++;
142
0
    }
143
144
0
    if (pos == len) {
145
0
        return 1;
146
0
    } else {
147
0
        return 0;
148
0
    }
149
0
}
150
151
1.24k
int bstr_begins_with_mem_nocase(const bstr *haystack, const void *_data, size_t len) {
152
1.24k
    const unsigned char *data = (const unsigned char *) _data;
153
1.24k
    const unsigned char *hdata = bstr_ptr(haystack);
154
1.24k
    size_t hlen = bstr_len(haystack);
155
1.24k
    size_t pos = 0;
156
157
6.98k
    while ((pos < len) && (pos < hlen)) {
158
5.85k
        if (tolower((int) hdata[pos]) != tolower((int) data[pos])) {
159
122
            return 0;
160
122
        }
161
162
5.73k
        pos++;
163
5.73k
    }
164
165
1.12k
    if (pos == len) {
166
1.12k
        return 1;
167
1.12k
    } else {
168
0
        return 0;
169
0
    }
170
1.12k
}
171
172
0
int bstr_char_at(const bstr *b, size_t pos) {
173
0
    unsigned char *data = bstr_ptr(b);
174
0
    size_t len = bstr_len(b);
175
176
0
    if (pos >= len) return -1;
177
0
    return data[pos];
178
0
}
179
180
30.2k
int bstr_char_at_end(const bstr *b, size_t pos) {
181
30.2k
    unsigned char *data = bstr_ptr(b);
182
30.2k
    size_t len = bstr_len(b);
183
184
30.2k
    if (pos >= len) return -1;
185
27.0k
    return data[len - 1 - pos];
186
30.2k
}
187
188
1.79k
void bstr_chop(bstr *b) {
189
1.79k
    if (bstr_len(b) > 0) {
190
1.79k
        bstr_adjust_len(b, bstr_len(b) - 1);
191
1.79k
    }
192
1.79k
}
193
194
35.5k
int bstr_chr(const bstr *b, int c) {
195
35.5k
    unsigned char *data = bstr_ptr(b);
196
35.5k
    size_t len = bstr_len(b);
197
198
35.5k
    size_t i = 0;
199
2.16M
    while (i < len) {
200
2.15M
        if (data[i] == c) {
201
30.7k
            return (int) i;
202
30.7k
        }
203
204
2.12M
        i++;
205
2.12M
    }
206
207
4.77k
    return -1;
208
35.5k
}
209
210
0
int bstr_cmp(const bstr *b1, const bstr *b2) {
211
0
    return bstr_util_cmp_mem(bstr_ptr(b1), bstr_len(b1), bstr_ptr(b2), bstr_len(b2));
212
0
}
213
214
21.5M
int bstr_cmp_c(const bstr *b, const char *c) {
215
21.5M
    return bstr_util_cmp_mem(bstr_ptr(b), bstr_len(b), c, strlen(c));
216
21.5M
}
217
218
363k
int bstr_cmp_c_nocase(const bstr *b, const char *c) {
219
363k
    return bstr_util_cmp_mem_nocase(bstr_ptr(b), bstr_len(b), c, strlen(c));
220
363k
}
221
222
4.96M
int bstr_cmp_c_nocasenorzero(const bstr *b, const char *c) {
223
4.96M
    return bstr_util_cmp_mem_nocasenorzero(bstr_ptr(b), bstr_len(b), c, strlen(c));
224
4.96M
}
225
226
0
int bstr_cmp_mem(const bstr *b, const void *data, size_t len) {
227
0
    return bstr_util_cmp_mem(bstr_ptr(b), bstr_len(b), data, len);
228
0
}
229
230
0
int bstr_cmp_mem_nocase(const bstr *b, const void *data, size_t len) {
231
0
    return bstr_util_cmp_mem_nocase(bstr_ptr(b), bstr_len(b), data, len);
232
0
}
233
234
2.15M
int bstr_cmp_nocase(const bstr *b1, const bstr *b2) {
235
2.15M
    return bstr_util_cmp_mem_nocase(bstr_ptr(b1), bstr_len(b1), bstr_ptr(b2), bstr_len(b2));
236
2.15M
}
237
238
649k
bstr *bstr_dup(const bstr *b) {
239
649k
    return bstr_dup_ex(b, 0, bstr_len(b));
240
649k
}
241
242
414k
bstr *bstr_dup_c(const char *cstr) {
243
414k
    return bstr_dup_mem(cstr, strlen(cstr));
244
414k
}
245
246
659k
bstr *bstr_dup_ex(const bstr *b, size_t offset, size_t len) {
247
659k
    bstr *bnew = bstr_alloc(len);
248
659k
    if (bnew == NULL) return NULL;
249
659k
    memcpy(bstr_ptr(bnew), bstr_ptr(b) + offset, len);
250
659k
    bstr_adjust_len(bnew, len);
251
659k
    return bnew;
252
659k
}
253
254
47.0k
bstr *bstr_dup_lower(const bstr *b) {
255
47.0k
    return bstr_to_lowercase(bstr_dup(b));
256
47.0k
}
257
258
4.01M
bstr *bstr_dup_mem(const void *data, size_t len) {
259
4.01M
    bstr *bnew = bstr_alloc(len);
260
4.01M
    if (bnew == NULL) return NULL;
261
4.01M
    memcpy(bstr_ptr(bnew), data, len);
262
4.01M
    bstr_adjust_len(bnew, len);
263
4.01M
    return bnew;
264
4.01M
}
265
266
661k
bstr *bstr_expand(bstr *b, size_t newsize) {
267
661k
    if (bstr_realptr(b) != NULL) {
268
        // Refuse to expand a wrapped bstring. In the future,
269
        // we can change this to make a copy of the data, thus
270
        // leaving the original memory area intact.
271
0
        return NULL;
272
0
    }
273
274
    // Catch attempts to "expand" to a smaller size
275
661k
    if (bstr_size(b) > newsize) return NULL;
276
277
661k
    bstr *bnew = realloc(b, sizeof (bstr) + newsize);
278
661k
    if (bnew == NULL) return NULL;
279
280
661k
    bstr_adjust_size(bnew, newsize);
281
282
661k
    return bnew;
283
661k
}
284
285
10.7M
void bstr_free(bstr *b) {
286
10.7M
    if (b == NULL) return;
287
4.86M
    free(b);
288
4.86M
}
289
290
0
int bstr_index_of(const bstr *haystack, const bstr *needle) {
291
0
    return bstr_index_of_mem(haystack, bstr_ptr(needle), bstr_len(needle));
292
0
}
293
294
1.10k
int bstr_index_of_c(const bstr *haystack, const char *needle) {
295
1.10k
    return bstr_index_of_mem(haystack, needle, strlen(needle));
296
1.10k
}
297
298
303
int bstr_index_of_c_nocase(const bstr *haystack, const char *needle) {
299
303
    return bstr_index_of_mem_nocase(haystack, needle, strlen(needle));
300
303
}
301
302
10.2k
int bstr_index_of_c_nocasenorzero(const bstr *haystack, const char *needle) {
303
10.2k
    return bstr_util_mem_index_of_mem_nocasenorzero(bstr_ptr(haystack), bstr_len(haystack), needle, strlen(needle));
304
10.2k
}
305
306
1.10k
int bstr_index_of_mem(const bstr *haystack, const void *_data2, size_t len2) {
307
1.10k
    return bstr_util_mem_index_of_mem(bstr_ptr(haystack), bstr_len(haystack), _data2, len2);
308
1.10k
}
309
310
303
int bstr_index_of_mem_nocase(const bstr *haystack, const void *_data2, size_t len2) {
311
303
    return bstr_util_mem_index_of_mem_nocase(bstr_ptr(haystack), bstr_len(haystack), _data2, len2);
312
303
}
313
314
0
int bstr_index_of_nocase(const bstr *haystack, const bstr *needle) {
315
0
    return bstr_index_of_mem_nocase(haystack, bstr_ptr(needle), bstr_len(needle));
316
0
}
317
318
0
int bstr_rchr(const bstr *b, int c) {
319
0
    const unsigned char *data = bstr_ptr(b);
320
0
    size_t len = bstr_len(b);
321
322
0
    size_t i = len;
323
0
    while (i > 0) {
324
0
        if (data[i - 1] == c) {
325
0
            return (int) (i - 1);
326
0
        }
327
328
0
        i--;
329
0
    }
330
331
0
    return -1;
332
0
}
333
334
95.1k
bstr *bstr_to_lowercase(bstr *b) {
335
95.1k
    if (b == NULL) return NULL;
336
337
95.1k
    unsigned char *data = bstr_ptr(b);
338
95.1k
    size_t len = bstr_len(b);
339
340
95.1k
    size_t i = 0;
341
4.63M
    while (i < len) {
342
4.53M
        data[i] = (uint8_t)tolower(data[i]);
343
4.53M
        i++;
344
4.53M
    }
345
346
95.1k
    return b;
347
95.1k
}
348
349
21.5M
int bstr_util_cmp_mem(const void *_data1, size_t len1, const void *_data2, size_t len2) {
350
21.5M
    const unsigned char *data1 = (const unsigned char *) _data1;
351
21.5M
    const unsigned char *data2 = (const unsigned char *) _data2;
352
21.5M
    size_t p1 = 0, p2 = 0;
353
354
22.0M
    while ((p1 < len1) && (p2 < len2)) {
355
21.6M
        if (data1[p1] != data2[p2]) {
356
            // Difference.
357
21.1M
            return (data1[p1] < data2[p2]) ? -1 : 1;
358
21.1M
        }
359
360
522k
        p1++;
361
522k
        p2++;
362
522k
    }
363
364
393k
    if ((p1 == len2) && (p2 == len1)) {
365
        // They're identical.
366
58.6k
        return 0;
367
335k
    } else {
368
        // One string is shorter.
369
335k
        if (p1 == len1) return -1;
370
2.46k
        else return 1;
371
335k
    }
372
393k
}
373
374
2.52M
int bstr_util_cmp_mem_nocase(const void *_data1, size_t len1, const void *_data2, size_t len2) {
375
2.52M
    const unsigned char *data1 = (const unsigned char *) _data1;
376
2.52M
    const unsigned char *data2 = (const unsigned char *) _data2;
377
2.52M
    size_t p1 = 0, p2 = 0;
378
379
7.40M
    while ((p1 < len1) && (p2 < len2)) {
380
5.96M
        if (tolower(data1[p1]) != tolower(data2[p2])) {
381
            // Difference.
382
1.07M
            return (tolower(data1[p1]) < tolower(data2[p2])) ? -1 : 1;
383
1.07M
        }
384
385
4.88M
        p1++;
386
4.88M
        p2++;
387
4.88M
    }
388
389
1.44M
    if ((p1 == len2) && (p2 == len1)) {
390
        // They're identical.
391
670k
        return 0;
392
776k
    } else {
393
        // One string is shorter.
394
776k
        if (p1 == len1) return -1;
395
272k
        else return 1;
396
776k
    }
397
1.44M
}
398
399
4.96M
int bstr_util_cmp_mem_nocasenorzero(const void *_data1, size_t len1, const void *_data2, size_t len2) {
400
4.96M
    const unsigned char *data1 = (const unsigned char *) _data1;
401
4.96M
    const unsigned char *data2 = (const unsigned char *) _data2;
402
4.96M
    size_t p1 = 0, p2 = 0;
403
404
10.6M
    while ((p1 < len1) && (p2 < len2)) {
405
8.91M
        if (data1[p1] == 0) {
406
385k
            p1++;
407
385k
            continue;
408
385k
        }
409
8.52M
        if (tolower(data1[p1]) != tolower(data2[p2])) {
410
            // Difference.
411
3.27M
            return (tolower(data1[p1]) < tolower(data2[p2])) ? -1 : 1;
412
3.27M
        }
413
414
5.25M
        p1++;
415
5.25M
        p2++;
416
5.25M
    }
417
418
1.72M
    while((p1 < len1) && (data1[p1] == 0)) {
419
26.2k
        p1++;
420
26.2k
    }
421
1.69M
    if ((p1 == len1) && (p2 == len2)) {
422
        // They're identical.
423
158k
        return 0;
424
1.53M
    } else {
425
        // One string is shorter.
426
1.53M
        if (p1 == len1) return -1;
427
4.58k
        else return 1;
428
1.53M
    }
429
1.69M
}
430
431
195k
int64_t bstr_util_mem_to_pint(const void *_data, size_t len, int base, size_t *lastlen) {
432
195k
    const unsigned char *data = (unsigned char *) _data;
433
195k
    int64_t rval = 0, tflag = 0;
434
195k
    size_t i = 0;
435
436
195k
    *lastlen = i;
437
438
529k
    for (i = 0; i < len; i++) {
439
368k
        int d = data[i];
440
441
368k
        *lastlen = i;
442
443
        // Convert character to digit.
444
368k
        if ((d >= '0') && (d <= '9')) {
445
320k
            d -= '0';
446
320k
        } else if ((d >= 'a') && (d <= 'z')) {
447
13.7k
            d -= 'a' - 10;
448
34.3k
        } else if ((d >= 'A') && (d <= 'Z')) {
449
10.6k
            d -= 'A' - 10;
450
23.6k
        } else {
451
23.6k
            d = -1;
452
23.6k
        }
453
454
        // Check that the digit makes sense with the base we are using.
455
368k
        if ((d == -1) || (d >= base)) {
456
34.2k
            if (tflag) {
457
                // Return what we have so far; lastlen points
458
                // to the first non-digit position.
459
18.2k
                return rval;
460
18.2k
            } else {
461
                // We didn't see a single digit.
462
15.9k
                return -1;
463
15.9k
            }
464
34.2k
        }
465
466
333k
        if (tflag) {
467
154k
            if (((INT64_MAX - d) / base) < rval) {
468
                // Overflow
469
488
                return -2;
470
488
            }
471
472
153k
            rval *= base;
473
153k
            rval += d;
474
179k
        } else {
475
179k
            rval = d;
476
179k
            tflag = 1;
477
179k
        }
478
333k
    }
479
480
160k
    *lastlen = i + 1;
481
482
160k
    return rval;
483
195k
}
484
485
0
int bstr_util_mem_index_of_c(const void *_data1, size_t len1, const char *cstr) {
486
0
    return bstr_util_mem_index_of_mem(_data1, len1, cstr, strlen(cstr));
487
0
}
488
489
40.6k
int bstr_util_mem_index_of_c_nocase(const void *_data1, size_t len1, const char *cstr) {
490
40.6k
    return bstr_util_mem_index_of_mem_nocase(_data1, len1, cstr, strlen(cstr));
491
40.6k
}
492
493
1.10k
int bstr_util_mem_index_of_mem(const void *_data1, size_t len1, const void *_data2, size_t len2) {
494
1.10k
    const unsigned char *data1 = (unsigned char *) _data1;
495
1.10k
    const unsigned char *data2 = (unsigned char *) _data2;
496
1.10k
    size_t i, j;
497
498
    // If we ever want to optimize this function, the following link
499
    // might be useful: http://en.wikipedia.org/wiki/Knuth-Morris-Pratt_algorithm
500
501
147k
    for (i = 0; i < len1; i++) {
502
146k
        size_t k = i;
503
504
147k
        for (j = 0; ((j < len2) && (k < len1)); j++, k++) {
505
147k
            if (data1[k] != data2[j]) break;
506
147k
        }
507
508
146k
        if (j == len2) {
509
99
            return (int) i;
510
99
        }
511
146k
    }
512
513
1.00k
    return -1;
514
1.10k
}
515
516
40.9k
int bstr_util_mem_index_of_mem_nocase(const void *_data1, size_t len1, const void *_data2, size_t len2) {
517
40.9k
    const unsigned char *data1 = (unsigned char *) _data1;
518
40.9k
    const unsigned char *data2 = (unsigned char *) _data2;
519
40.9k
    size_t i, j;
520
521
    // If we ever want to optimize this function, the following link
522
    // might be useful: http://en.wikipedia.org/wiki/Knuth-Morris-Pratt_algorithm
523
524
836k
    for (i = 0; i < len1; i++) {
525
809k
        size_t k = i;
526
527
907k
        for (j = 0; ((j < len2) && (k < len1)); j++, k++) {
528
892k
            if (toupper(data1[k]) != toupper(data2[j])) break;
529
892k
        }
530
531
809k
        if (j == len2) {
532
14.2k
            return (int) i;
533
14.2k
        }
534
809k
    }
535
536
26.7k
    return -1;
537
40.9k
}
538
539
10.2k
int bstr_util_mem_index_of_mem_nocasenorzero(const void *_data1, size_t len1, const void *_data2, size_t len2) {
540
10.2k
    const unsigned char *data1 = (unsigned char *) _data1;
541
10.2k
    const unsigned char *data2 = (unsigned char *) _data2;
542
10.2k
    size_t i, j;
543
544
    // If we ever want to optimize this function, the following link
545
    // might be useful: http://en.wikipedia.org/wiki/Knuth-Morris-Pratt_algorithm
546
547
169k
    for (i = 0; i < len1; i++) {
548
169k
        size_t k = i;
549
169k
        if (data1[i] == 0) {
550
            // skip leading zeroes to avoid quadratic complexity
551
15.1k
            continue;
552
15.1k
        }
553
554
239k
        for (j = 0; ((j < len2) && (k < len1)); j++, k++) {
555
229k
            if (data1[k] == 0) {
556
12.7k
                j--;
557
12.7k
                continue;
558
12.7k
            }
559
216k
            if (toupper(data1[k]) != toupper(data2[j])) break;
560
216k
        }
561
562
154k
        if (j == len2) {
563
9.94k
            return (int) i;
564
9.94k
        }
565
154k
    }
566
567
275
    return -1;
568
10.2k
}
569
570
25.1k
void bstr_util_mem_trim(unsigned char **data, size_t *len) {
571
25.1k
    if ((data == NULL)||(len == NULL)) return;
572
573
25.1k
    unsigned char *d = *data;
574
25.1k
    size_t l = *len;
575
576
    // Ignore whitespace at the beginning.
577
25.1k
    size_t pos = 0;
578
27.2k
    while ((pos < l) && isspace(d[pos])) pos++;
579
25.1k
    d += pos;
580
25.1k
    l -= pos;
581
582
    // Ignore whitespace at the end.
583
25.8k
    while ((l > 0)&&(isspace(d[l - 1]))) l--;
584
585
25.1k
    *data = d;
586
25.1k
    *len = l;
587
25.1k
}
588
589
0
char *bstr_util_memdup_to_c(const void *_data, size_t len) {
590
0
    const unsigned char *data = (unsigned char *) _data;
591
592
    // Count how many NUL bytes we have in the string.
593
0
    size_t i, nulls = 0;
594
0
    for (i = 0; i < len; i++) {
595
0
        if (data[i] == '\0') {
596
0
            nulls++;
597
0
        }
598
0
    }
599
600
    // Now copy the string into a NUL-terminated buffer.
601
602
0
    char *r, *d;
603
0
    r = d = malloc(len + nulls + 1);
604
0
    if (d == NULL) return NULL;
605
606
0
    while (len--) {
607
0
        if (*data == '\0') {
608
0
            data++;
609
0
            *d++ = '\\';
610
0
            *d++ = '0';
611
0
        } else {
612
0
            *d++ = *data++;
613
0
        }
614
0
    }
615
616
0
    *d = '\0';
617
618
0
    return r;
619
0
}
620
621
0
char *bstr_util_strdup_to_c(const bstr *b) {
622
0
    if (b == NULL) return NULL;
623
0
    return bstr_util_memdup_to_c(bstr_ptr(b), bstr_len(b));
624
0
}
625
626
0
bstr *bstr_wrap_c(const char *cstr) {
627
0
    return bstr_wrap_mem((unsigned char *) cstr, strlen(cstr));
628
0
}
629
630
0
bstr *bstr_wrap_mem(const void *data, size_t len) {
631
0
    bstr *b = (bstr *) malloc(sizeof (bstr));
632
0
    if (b == NULL) return NULL;
633
634
0
    b->size = b->len = len;
635
0
    b->realptr = (unsigned char *) data;
636
637
0
    return b;
638
0
}