varnish-cache/lib/libvarnish/vsb.c
0
/*-
1
 * Copyright (c) 2000-2008 Poul-Henning Kamp <phk@FreeBSD.org>
2
 * Copyright (c) 2000-2008 Dag-Erling Coïdan Smørgrav
3
 * All rights reserved.
4
 *
5
 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
6
 *
7
 * Redistribution and use in source and binary forms, with or without
8
 * modification, are permitted provided that the following conditions
9
 * are met:
10
 * 1. Redistributions of source code must retain the above copyright
11
 *    notice, this list of conditions and the following disclaimer
12
 *    in this position and unchanged.
13
 * 2. 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
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27
 * SUCH DAMAGE.
28
__FBSDID("$FreeBSD: head/sys/kern/subr_vsb.c 222004 2011-05-17 06:36:32Z phk $")
29
 */
30
31
#include "config.h"
32
33
#include <ctype.h>
34
#include <stdarg.h>
35
#include <stdio.h>
36
#include <stdint.h>
37
#include <stdlib.h>
38
#include <string.h>
39
#include <unistd.h>
40
41
#include "vdef.h"
42
#include "vas.h"        // XXX Flexelint "not used" - but req'ed for assert()
43
#include "vsb.h"
44
45
#define KASSERT(e, m)           assert(e)
46
#define SBMALLOC(size)          malloc(size)
47
#define SBFREE(buf)             free(buf)
48
49
#define rndup2(x, y)    (((x)+((y)-1))&(~((y)-1))) /* if y is powers of two */
50
51
/*
52
 * Predicates
53
 */
54
#define VSB_ISDYNAMIC(s)        ((s)->s_flags & VSB_DYNAMIC)
55
#define VSB_ISDYNSTRUCT(s)      ((s)->s_flags & VSB_DYNSTRUCT)
56
#define VSB_HASROOM(s)          ((s)->s_len < (s)->s_size - 1L)
57
#define VSB_FREESPACE(s)        ((s)->s_size - ((s)->s_len + 1L))
58
#define VSB_CANEXTEND(s)        ((s)->s_flags & VSB_AUTOEXTEND)
59
60
/*
61
 * Set / clear flags
62
 */
63
#define VSB_SETFLAG(s, f)       do { (s)->s_flags |= (f); } while (0)
64
#define VSB_CLEARFLAG(s, f)     do { (s)->s_flags &= ~(f); } while (0)
65
66
#define VSB_MINEXTENDSIZE       16              /* Should be power of 2. */
67
68
#ifdef PAGE_SIZE
69
#define VSB_MAXEXTENDSIZE       PAGE_SIZE
70
#define VSB_MAXEXTENDINCR       PAGE_SIZE
71
#else
72
#define VSB_MAXEXTENDSIZE       4096
73
#define VSB_MAXEXTENDINCR       4096
74
#endif
75
76
/*
77
 * Debugging support
78
 */
79
#if !defined(NDEBUG)
80
static void
81 1229555087
_assert_VSB_integrity(const char *fun, const struct vsb *s)
82
{
83
84 1229555087
        (void)fun;
85 1229555087
        (void)s;
86 1229555087
        KASSERT(s != NULL,
87
            ("%s called with a NULL vsb pointer", fun));
88 1229555087
        KASSERT(s->magic == VSB_MAGIC,
89
            ("%s called with a bogus vsb pointer", fun));
90 1229555087
        KASSERT(s->s_buf != NULL,
91
            ("%s called with uninitialized or corrupt vsb", fun));
92 1229555087
        KASSERT(s->s_len < s->s_size,
93
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
94 1229555087
}
95
96
static void
97 1153229338
_assert_VSB_state(const char *fun, const struct vsb *s, int state)
98
{
99
100 1153229338
        (void)fun;
101 1153229338
        (void)s;
102 1153229338
        (void)state;
103 1153229338
        KASSERT((s->s_flags & VSB_FINISHED) == state,
104
            ("%s called with %sfinished or corrupt vsb", fun,
105
            (state ? "un" : "")));
106 1153229338
}
107
#define assert_VSB_integrity(s) _assert_VSB_integrity(__func__, (s))
108
#define assert_VSB_state(s, i)   _assert_VSB_state(__func__, (s), (i))
109
#else
110
#define assert_VSB_integrity(s) do { } while (0)
111
#define assert_VSB_state(s, i)   do { } while (0)
112
#endif
113
114
#ifdef CTASSERT
115
CTASSERT(powerof2(VSB_MAXEXTENDSIZE));
116
CTASSERT(powerof2(VSB_MAXEXTENDINCR));
117
#endif
118
119
static ssize_t
120 56634246
VSB_extendsize(ssize_t size)
121
{
122
        ssize_t newsize;
123
124 56634246
        if (size < (int)VSB_MAXEXTENDSIZE) {
125 56008477
                newsize = VSB_MINEXTENDSIZE;
126 133964462
                while (newsize < size)
127 77955985
                        newsize *= 2;
128 56008477
        } else {
129 625769
                newsize = rndup2(size, VSB_MAXEXTENDINCR);
130
        }
131 56634246
        KASSERT(newsize >= size, ("%s: %d < %d\n", __func__, newsize, size));
132 56634246
        return (newsize);
133
}
134
135
/*
136
 * Extend an vsb.
137
 */
138
static ssize_t
139 32452798
VSB_extend(struct vsb *s, ssize_t addlen)
140
{
141
        char *newbuf;
142
        ssize_t newsize;
143
144 32452798
        if (!VSB_CANEXTEND(s))
145 561
                return (-1);
146 32452237
        newsize = VSB_extendsize(s->s_size + addlen);
147 32452237
        if (VSB_ISDYNAMIC(s))
148 32452237
                newbuf = realloc(s->s_buf, newsize);
149
        else
150 0
                newbuf = SBMALLOC(newsize);
151 32452237
        if (newbuf == NULL)
152 0
                return (-1);
153 32452237
        if (!VSB_ISDYNAMIC(s)) {
154 0
                memcpy(newbuf, s->s_buf, s->s_size);
155 0
                VSB_SETFLAG(s, VSB_DYNAMIC);
156 0
        }
157 32452237
        s->s_buf = newbuf;
158 32452237
        s->s_size = newsize;
159 32452237
        return (0);
160 32452798
}
161
162
static void
163 1010491182
_vsb_indent(struct vsb *s)
164
{
165 1012193754
        if (s->s_indent == 0 || s->s_error != 0 ||
166 1702572
            (s->s_len > 0 && s->s_buf[s->s_len - 1] != '\n'))
167 1010371222
                return;
168 119960
        if (VSB_FREESPACE(s) <= s->s_indent &&
169 122
            VSB_extend(s, s->s_indent) < 0) {
170 0
                s->s_error = ENOMEM;
171 0
                return;
172
        }
173 119960
        memset(s->s_buf + s->s_len, ' ', s->s_indent);
174 119960
        s->s_len += s->s_indent;
175 1010491182
}
176
177
/*
178
 * Initialize the internals of an vsb.
179
 * If buf is non-NULL, it points to a static or already-allocated string
180
 * big enough to hold at least length characters.
181
 */
182
static struct vsb *
183 24714032
VSB_newbuf(struct vsb *s, char *buf, int length, int flags)
184
{
185
186 24714032
        memset(s, 0, sizeof(*s));
187 24714032
        s->magic = VSB_MAGIC;
188 24714032
        s->s_flags = flags;
189 24714032
        s->s_size = length;
190 24714032
        s->s_buf = buf;
191
192 24714032
        if ((s->s_flags & VSB_AUTOEXTEND) == 0) {
193 531994
                KASSERT(s->s_size > 1,
194
                    ("attempt to create a too small vsb"));
195 531994
        }
196
197 24714032
        if (s->s_buf != NULL)
198 531995
                return (s);
199
200 24182037
        if ((flags & VSB_AUTOEXTEND) != 0)
201 24182027
                s->s_size = VSB_extendsize(s->s_size);
202
203 24182037
        s->s_buf = SBMALLOC(s->s_size);
204 24182037
        if (s->s_buf == NULL)
205 0
                return (NULL);
206 24182023
        VSB_SETFLAG(s, VSB_DYNAMIC);
207 24182023
        return (s);
208 24714018
}
209
210
struct vsb *
211 531992
VSB_init(struct vsb *s, void *buf, ssize_t length)
212
{
213 531992
        AN(s);
214 531992
        AN(buf);
215
216 531992
        KASSERT(length >= 0,
217
            ("attempt to create an vsb of negative length (%zd)", length));
218 531992
        return (VSB_newbuf(s, buf, length, VSB_FIXEDLEN));
219
}
220
221
/*
222
 * Allocate a dynamic vsb
223
 */
224
struct vsb *
225 24182025
VSB_new_auto(void)
226
{
227
        struct vsb *s;
228
229 24182025
        s = SBMALLOC(sizeof(*s));
230 24182025
        if (s == NULL)
231 0
                return (NULL);
232 24182025
        if (VSB_newbuf(s, NULL, 0, VSB_AUTOEXTEND) == NULL) {
233 0
                SBFREE(s);
234 0
                return (NULL);
235
        }
236 24182025
        VSB_SETFLAG(s, VSB_DYNSTRUCT);
237 24182025
        return (s);
238 24182025
}
239
240
/*
241
 * Clear an vsb and reset its position.
242
 */
243
void
244 37238605
VSB_clear(struct vsb *s)
245
{
246
247 37238605
        assert_VSB_integrity(s);
248
        /* don't care if it's finished or not */
249
250 37238605
        VSB_CLEARFLAG(s, VSB_FINISHED);
251 37238605
        s->s_error = 0;
252 37238605
        s->s_len = 0;
253 37238605
        s->s_indent = 0;
254 37238605
}
255
256
/*
257
 * Append a byte to an vsb.  This is the core function for appending
258
 * to an vsb and is the main place that deals with extending the
259
 * buffer and marking overflow.
260
 */
261
static void
262 772581056
VSB_put_byte(struct vsb *s, int c)
263
{
264
265 772581056
        assert_VSB_integrity(s);
266 772581056
        assert_VSB_state(s, 0);
267
268 772581056
        if (s->s_error != 0)
269 0
                return;
270 772581056
        _vsb_indent(s);
271 772581056
        if (VSB_FREESPACE(s) <= 0) {
272 7468744
                if (VSB_extend(s, 1) < 0)
273 0
                        s->s_error = ENOMEM;
274 7468744
                if (s->s_error != 0)
275 0
                        return;
276 7468744
        }
277 772581056
        s->s_buf[s->s_len++] = (char)c;
278 772581056
}
279
280
/*
281
 * Append a byte string to an vsb.
282
 */
283
int
284 47229064
VSB_bcat(struct vsb *s, const void *buf, ssize_t len)
285
{
286 47229064
        assert_VSB_integrity(s);
287 47229064
        assert_VSB_state(s, 0);
288
289 47229064
        assert(len >= 0);
290 47229064
        if (s->s_error != 0)
291 0
                return (-1);
292 47229064
        if (len == 0)
293 2762968
                return (0);
294 44466096
        _vsb_indent(s);
295 44466096
        if (len > VSB_FREESPACE(s)) {
296 8702391
                if (VSB_extend(s, len - VSB_FREESPACE(s)) < 0)
297 136
                        s->s_error = ENOMEM;
298 8702391
                if (s->s_error != 0)
299 136
                        return (-1);
300 8702255
        }
301 44465960
        memcpy(s->s_buf + s->s_len, buf, len);
302 44465960
        s->s_len += len;
303 44465960
        return (0);
304 47229064
}
305
306
/*
307
 * Append a string to an vsb.
308
 */
309
int
310 43335696
VSB_cat(struct vsb *s, const char *str)
311
{
312
        const char *nl;
313
        size_t l;
314
315 43335696
        assert_VSB_integrity(s);
316 43335696
        assert_VSB_state(s, 0);
317 43335696
        KASSERT(str != NULL,
318
            ("%s called with a NULL str pointer", __func__));
319
320 43335696
        if (s->s_error != 0)
321 85
                return (-1);
322
323 43352797
        while (s->s_indent > 0 && (nl = strchr(str, '\n')) != NULL) {
324 17186
                l = (nl - str) + 1;
325 17186
                if (VSB_bcat(s, str, l) < 0)
326 0
                        return (-1);
327 17186
                str += l;
328
        }
329
330 43335611
        l = strlen(str);
331 43335611
        return (VSB_bcat(s, str, l));
332 43335696
}
333
334
/*
335
 * Format the given argument list and append the resulting string to an vsb.
336
 */
337
int
338 193545672
VSB_vprintf(struct vsb *s, const char *fmt, va_list ap)
339
{
340
        va_list ap_copy;
341
        int len;
342
343 193545672
        assert_VSB_integrity(s);
344 193545672
        assert_VSB_state(s, 0);
345
346 193545672
        KASSERT(fmt != NULL,
347
            ("%s called with a NULL format string", __func__));
348
349 193545672
        if (s->s_error != 0)
350 0
                return (-1);
351 193545672
        _vsb_indent(s);
352
353
        /*
354
         * For the moment, there is no way to get vsnprintf(3) to hand
355
         * back a character at a time, to push everything into
356
         * VSB_putc_func() as was done for the kernel.
357
         *
358
         * In userspace, while drains are useful, there's generally
359
         * not a problem attempting to malloc(3) on out of space.  So
360
         * expand a userland vsb if there is not enough room for the
361
         * data produced by VSB_[v]printf(3).
362
         */
363
364 193545672
        do {
365 209826485
                va_copy(ap_copy, ap);
366 419652970
                len = vsnprintf(&s->s_buf[s->s_len], VSB_FREESPACE(s) + 1,
367 209826485
                    fmt, ap_copy);
368 209826485
                va_end(ap_copy);
369 209826485
                if (len < 0) {
370 0
                        s->s_error = errno;
371 0
                        return (-1);
372
                }
373 226108063
        } while (len > VSB_FREESPACE(s) &&
374 16281578
            VSB_extend(s, len - VSB_FREESPACE(s)) == 0);
375
376
        /*
377
         * s->s_len is the length of the string, without the terminating nul.
378
         * When updating s->s_len, we must subtract 1 from the length that
379
         * we passed into vsnprintf() because that length includes the
380
         * terminating nul.
381
         *
382
         * vsnprintf() returns the amount that would have been copied,
383
         * given sufficient space, so don't over-increment s_len.
384
         */
385 193545672
        s->s_len += vmin_t(ssize_t, len, VSB_FREESPACE(s));
386 193545672
        if (!VSB_HASROOM(s) && !VSB_CANEXTEND(s))
387 425
                s->s_error = ENOMEM;
388
389 193545672
        KASSERT(s->s_len < s->s_size,
390
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
391
392 193545672
        if (s->s_error != 0)
393 439
                return (-1);
394 193545233
        return (0);
395 193545672
}
396
397
/*
398
 * Format the given arguments and append the resulting string to an vsb.
399
 */
400
int
401 104555703
VSB_printf(struct vsb *s, const char *fmt, ...)
402
{
403
        va_list ap;
404
        int result;
405
406 104555703
        va_start(ap, fmt);
407 104555703
        result = VSB_vprintf(s, fmt, ap);
408 104555703
        va_end(ap);
409 104555703
        return (result);
410
}
411
412
/*
413
 * Append a character to an vsb.
414
 */
415
int
416 772586127
VSB_putc(struct vsb *s, int c)
417
{
418
419 772586127
        VSB_put_byte(s, c);
420 772586127
        if (s->s_error != 0)
421 0
                return (-1);
422 772586127
        return (0);
423 772586127
}
424
425
/*
426
 * Check if an vsb has an error.
427
 */
428
int
429 25664
VSB_error(const struct vsb *s)
430
{
431
432 25664
        return (s->s_error);
433
}
434
435
/*
436
 * Finish off an vsb.
437
 */
438
int
439 47343751
VSB_finish(struct vsb *s)
440
{
441
442 47343751
        assert_VSB_integrity(s);
443 47343751
        assert_VSB_state(s, 0);
444
445 47343751
        s->s_buf[s->s_len] = '\0';
446 47343751
        VSB_SETFLAG(s, VSB_FINISHED);
447 47343751
        errno = s->s_error;
448 47343751
        if (s->s_error)
449 561
                return (-1);
450 47343190
        return (0);
451 47343751
}
452
453
/*
454
 * Return a pointer to the vsb data.
455
 */
456
char *
457 48286220
VSB_data(const struct vsb *s)
458
{
459
460 48286220
        assert_VSB_integrity(s);
461 48286220
        assert_VSB_state(s, VSB_FINISHED);
462
463 48286220
        return (s->s_buf);
464
}
465
466
/*
467
 * Return the length of the vsb data.
468
 */
469
ssize_t
470 18553375
VSB_len(const struct vsb *s)
471
{
472
473 18553375
        assert_VSB_integrity(s);
474
        /* don't care if it's finished or not */
475
476 18553375
        if (s->s_error != 0)
477 0
                return (-1);
478 18553375
        return (s->s_len);
479 18553375
}
480
481
void
482 516098
VSB_fini(struct vsb *s)
483
{
484
485 516098
        assert_VSB_integrity(s);
486 516098
        assert(!VSB_ISDYNAMIC(s));
487 516098
        assert(!VSB_ISDYNSTRUCT(s));
488 516098
        memset(s, 0, sizeof(*s));
489 516098
}
490
491
void
492 20109973
VSB_destroy(struct vsb **s)
493
{
494
495 20109973
        AN(s);
496 20109973
        assert_VSB_integrity(*s);
497 20109973
        assert(VSB_ISDYNAMIC(*s));
498 20109973
        assert(VSB_ISDYNSTRUCT(*s));
499 20109973
        SBFREE((*s)->s_buf);
500 20109973
        memset(*s, 0, sizeof(**s));
501 20109973
        SBFREE(*s);
502 20109973
        *s = NULL;
503 20109973
}
504
505
/*
506
 * Quote a string
507
 */
508
509
static void
510 68773
vsb_quote_hex(struct vsb *s, const uint8_t *u, size_t len)
511
{
512
        const uint8_t *w;
513
514 68773
        VSB_cat(s, "0x");
515 166119
        for (w = u; w < u + len; w++)
516 156123
                if (*w != 0x00)
517 58777
                        break;
518 68773
        if (w == u + len && len > 4) {
519 306
                VSB_cat(s, "0...0");
520 306
        } else {
521 11379017
                for (w = u; w < u + len; w++)
522 11310550
                        VSB_printf(s, "%02x", *w);
523
        }
524 68773
}
525
526
void
527 4275837
VSB_quote_pfx(struct vsb *s, const char *pfx, const void *v, int len, int how)
528
{
529 4275837
        const uint8_t *p = v;
530
        const uint8_t *q;
531 4275837
        int quote = 0;
532
        int nl;
533
534 4275837
        nl = how &
535
            (VSB_QUOTE_JSON|VSB_QUOTE_HEX|VSB_QUOTE_CSTR|VSB_QUOTE_UNSAFE);
536 4275837
        AZ(nl & (nl - 1)); // Only one bit can be set
537
538 4275837
        if (how & VSB_QUOTE_ESCHEX)
539 2025158
                AZ(how & (VSB_QUOTE_JSON|VSB_QUOTE_HEX));
540
541 4275837
        if (how & VSB_QUOTE_UNSAFE)
542 2019821
                how |= VSB_QUOTE_NONL;
543
544 4275837
        assert(p != NULL);
545 4275837
        if (len == -1)
546 2048647
                len = strlen(v);
547
548 4275837
        if (len == 0 && (how & VSB_QUOTE_CSTR)) {
549 3468
                VSB_printf(s, "%s\"\"", pfx);
550 3468
                if ((how & VSB_QUOTE_NONL))
551 0
                        VSB_putc(s, '\n');
552 3468
        }
553
554 4275837
        if (len == 0)
555 138869
                return;
556
557 4136968
        VSB_cat(s, pfx);
558
559 4136968
        if (how & VSB_QUOTE_HEX) {
560 68773
                vsb_quote_hex(s, v, len);
561 68773
                if (how & VSB_QUOTE_NONL)
562 17
                        VSB_putc(s, '\n');
563 68773
                return;
564
        }
565
566 4068195
        if (how & VSB_QUOTE_CSTR)
567 1909831
                VSB_putc(s, '"');
568
569 85725416
        for (q = p; q < p + len; q++) {
570
                if (
571 82879057
                    *q < 0x20 ||
572 81772372
                    *q == '"' ||
573 81757895
                    *q == '\\' ||
574 81754993
                    (*q == '?' && (how & VSB_QUOTE_CSTR)) ||
575 81657221
                    (*q > 0x7e && !(how & VSB_QUOTE_JSON))
576
                ) {
577 1219594
                        quote++;
578 1219594
                        break;
579
                }
580 81657221
        }
581
582 4068195
        if (!quote) {
583 2848603
                VSB_bcat(s, p, len);
584 2848603
                if ((how & VSB_QUOTE_NONL) &&
585 1309407
                    p[len-1] != '\n')
586 1309410
                        (void)VSB_putc(s, '\n');
587 2848609
                if (how & VSB_QUOTE_CSTR)
588 1297593
                        VSB_putc(s, '"');
589 2848609
                return;
590
        }
591
592 1219592
        nl = 0;
593 269959874
        for (q = p; q < p + len; q++) {
594 268740282
                if (nl)
595 2793313
                        VSB_cat(s, pfx);
596 268740282
                nl = 0;
597 268740282
                switch (*q) {
598
                case '?':
599
                        /* Avoid C Trigraph insanity */
600 406254
                        if (how & VSB_QUOTE_CSTR && !(how & VSB_QUOTE_JSON))
601 136986
                                (void)VSB_putc(s, '\\');
602 406254
                        (void)VSB_putc(s, *q);
603 406254
                        break;
604
                case '\\':
605
                case '"':
606 2715145
                        if (!(how & VSB_QUOTE_UNSAFE))
607 1788373
                                (void)VSB_putc(s, '\\');
608 2715145
                        (void)VSB_putc(s, *q);
609 2715145
                        break;
610
                case '\n':
611 11187959
                        if (how & VSB_QUOTE_CSTR) {
612 7813455
                                VSB_printf(s, "\\n\"\n%s\"", pfx);
613 11187959
                        } else if (how & VSB_QUOTE_JSON) {
614 25228
                                VSB_cat(s, "\\n");
615 3374504
                        } else if (how & VSB_QUOTE_NONL) {
616 3349055
                                VSB_putc(s, *q);
617 3349055
                                nl = 1;
618 3349055
                        } else {
619 221
                                VSB_cat(s, "\\n");
620
                        }
621 11187959
                        break;
622
                case '\r':
623 1349144
                        VSB_cat(s, "\\r");
624 1349144
                        break;
625
                case '\t':
626 3710946
                        VSB_cat(s, "\\t");
627 3710946
                        break;
628
                default:
629 249370834
                        if (0x20 <= *q && *q <= 0x7e)
630 249218029
                                VSB_putc(s, *q);
631 152805
                        else if (*q > 0x7e && (how & VSB_QUOTE_JSON))
632 68
                                VSB_putc(s, *q);
633 152737
                        else if (how & VSB_QUOTE_JSON)
634 119
                                VSB_printf(s, "\\u%04x", *q);
635 152618
                        else if (how & VSB_QUOTE_ESCHEX)
636 151530
                                VSB_printf(s, "\\x%02x", *q);
637
                        else
638 1088
                                VSB_printf(s, "\\%03o", *q);
639 249370834
                        break;
640
                }
641 268740282
        }
642 1219592
        if (how & VSB_QUOTE_CSTR)
643 612238
                VSB_putc(s, '"');
644 1219592
        if ((how & VSB_QUOTE_NONL) && !nl)
645 44666
                VSB_putc(s, '\n');
646 4275843
}
647
648
void
649 2187177
VSB_quote(struct vsb *s, const void *v, int len, int how)
650
{
651 2187177
        VSB_quote_pfx(s, "", v, len, how);
652 2187177
}
653
654
/*
655
 * Indentation
656
 */
657
658
void
659 41442
VSB_indent(struct vsb *s, int i)
660
{
661
662 41442
        assert_VSB_integrity(s);
663 41442
        if (s->s_indent + i < 0)
664 0
                s->s_error = EINVAL;
665
        else
666 41442
                s->s_indent += i;
667 41442
}
668
669
int
670 1175017
VSB_tofile(const struct vsb *s, int fd)
671
{
672
        const char *p;
673
        ssize_t r;
674
        size_t sz;
675
676 1175017
        assert_VSB_integrity(s);
677 1175017
        assert_VSB_state(s, VSB_FINISHED);
678 1175017
        assert(s->s_len >= 0);
679 1175017
        r = 0;
680 1175017
        p = s->s_buf;
681 1175017
        sz = (typeof(sz))s->s_len;
682 2349699
        while (sz > 0) {
683 1174794
                r = write(fd, p, sz);
684 1174794
                if (r < 0)
685 112
                        return (-1);
686 1174682
                assert((typeof(sz))r <= sz);
687 1174682
                p += r;
688 1174682
                sz -= r;
689
        }
690 1174905
        return (0);
691 1175017
}