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 1667379132
_assert_VSB_integrity(const char *fun, const struct vsb *s)
82
{
83
84 1667379132
        (void)fun;
85 1667379132
        (void)s;
86 1667379132
        KASSERT(s != NULL,
87
            ("%s called with a NULL vsb pointer", fun));
88 1667379132
        KASSERT(s->magic == VSB_MAGIC,
89
            ("%s called wih an bogus vsb pointer", fun));
90 1667379132
        KASSERT(s->s_buf != NULL,
91
            ("%s called with uninitialized or corrupt vsb", fun));
92 1667379132
        KASSERT(s->s_len < s->s_size,
93
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
94 1667379132
}
95
96
static void
97 1566513847
_assert_VSB_state(const char *fun, const struct vsb *s, int state)
98
{
99
100 1566513847
        (void)fun;
101 1566513847
        (void)s;
102 1566513847
        (void)state;
103 1566513847
        KASSERT((s->s_flags & VSB_FINISHED) == state,
104
            ("%s called with %sfinished or corrupt vsb", fun,
105
            (state ? "un" : "")));
106 1566513847
}
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 76465524
VSB_extendsize(ssize_t size)
121
{
122
        ssize_t newsize;
123
124 76465524
        if (size < (int)VSB_MAXEXTENDSIZE) {
125 75578223
                newsize = VSB_MINEXTENDSIZE;
126 180701552
                while (newsize < size)
127 105123329
                        newsize *= 2;
128 75578223
        } else {
129 887301
                newsize = rndup2(size, VSB_MAXEXTENDINCR);
130
        }
131 76465524
        KASSERT(newsize >= size, ("%s: %d < %d\n", __func__, newsize, size));
132 76465524
        return (newsize);
133
}
134
135
/*
136
 * Extend an vsb.
137
 */
138
static ssize_t
139 43870803
VSB_extend(struct vsb *s, ssize_t addlen)
140
{
141
        char *newbuf;
142
        ssize_t newsize;
143
144 43870803
        if (!VSB_CANEXTEND(s))
145 725
                return (-1);
146 43870078
        newsize = VSB_extendsize(s->s_size + addlen);
147 43870078
        if (VSB_ISDYNAMIC(s))
148 43870078
                newbuf = realloc(s->s_buf, newsize);
149
        else
150 0
                newbuf = SBMALLOC(newsize);
151 43870078
        if (newbuf == NULL)
152 0
                return (-1);
153 43870078
        if (!VSB_ISDYNAMIC(s)) {
154 0
                memcpy(newbuf, s->s_buf, s->s_size);
155 0
                VSB_SETFLAG(s, VSB_DYNAMIC);
156 0
        }
157 43870078
        s->s_buf = newbuf;
158 43870078
        s->s_size = newsize;
159 43870078
        return (0);
160 43870803
}
161
162
static void
163 1373826156
_vsb_indent(struct vsb *s)
164
{
165 1376072803
        if (s->s_indent == 0 || s->s_error != 0 ||
166 2246647
            (s->s_len > 0 && s->s_buf[s->s_len - 1] != '\n'))
167 1373660458
                return;
168 165698
        if (VSB_FREESPACE(s) <= s->s_indent &&
169 208
            VSB_extend(s, s->s_indent) < 0) {
170 0
                s->s_error = ENOMEM;
171 0
                return;
172
        }
173 165698
        memset(s->s_buf + s->s_len, ' ', s->s_indent);
174 165698
        s->s_len += s->s_indent;
175 1373826156
}
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 33334655
VSB_newbuf(struct vsb *s, char *buf, int length, int flags)
184
{
185
186 33334655
        memset(s, 0, sizeof(*s));
187 33334655
        s->magic = VSB_MAGIC;
188 33334655
        s->s_flags = flags;
189 33334655
        s->s_size = length;
190 33334655
        s->s_buf = buf;
191
192 33334655
        if ((s->s_flags & VSB_AUTOEXTEND) == 0) {
193 739177
                KASSERT(s->s_size > 1,
194
                    ("attempt to create a too small vsb"));
195 739177
        }
196
197 33334655
        if (s->s_buf != NULL)
198 739176
                return (s);
199
200 32595479
        if ((flags & VSB_AUTOEXTEND) != 0)
201 32595471
                s->s_size = VSB_extendsize(s->s_size);
202
203 32595479
        s->s_buf = SBMALLOC(s->s_size);
204 32595479
        if (s->s_buf == NULL)
205 0
                return (NULL);
206 32595475
        VSB_SETFLAG(s, VSB_DYNAMIC);
207 32595475
        return (s);
208 33334651
}
209
210
struct vsb *
211 739180
VSB_init(struct vsb *s, void *buf, ssize_t length)
212
{
213 739180
        AN(s);
214 739180
        AN(buf);
215
216 739180
        KASSERT(length >= 0,
217
            ("attempt to create an vsb of negative length (%zd)", length));
218 739180
        return (VSB_newbuf(s, buf, length, VSB_FIXEDLEN));
219
}
220
221
/*
222
 * Allocate a dynamic vsb
223
 */
224
struct vsb *
225 32595476
VSB_new_auto(void)
226
{
227
        struct vsb *s;
228
229 32595476
        s = SBMALLOC(sizeof(*s));
230 32595476
        if (s == NULL)
231 0
                return (NULL);
232 32595476
        if (VSB_newbuf(s, NULL, 0, VSB_AUTOEXTEND) == NULL) {
233 0
                SBFREE(s);
234 0
                return (NULL);
235
        }
236 32595476
        VSB_SETFLAG(s, VSB_DYNSTRUCT);
237 32595476
        return (s);
238 32595476
}
239
240
/*
241
 * Clear an vsb and reset its position.
242
 */
243
void
244 50578412
VSB_clear(struct vsb *s)
245
{
246
247 50578412
        assert_VSB_integrity(s);
248
        /* don't care if it's finished or not */
249
250 50578412
        VSB_CLEARFLAG(s, VSB_FINISHED);
251 50578412
        s->s_error = 0;
252 50578412
        s->s_len = 0;
253 50578412
        s->s_indent = 0;
254 50578412
}
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 1049537826
VSB_put_byte(struct vsb *s, int c)
263
{
264
265 1049537826
        assert_VSB_integrity(s);
266 1049537826
        assert_VSB_state(s, 0);
267
268 1049537826
        if (s->s_error != 0)
269 0
                return;
270 1049537826
        _vsb_indent(s);
271 1049537826
        if (VSB_FREESPACE(s) <= 0) {
272 10244431
                if (VSB_extend(s, 1) < 0)
273 0
                        s->s_error = ENOMEM;
274 10244431
                if (s->s_error != 0)
275 0
                        return;
276 10244431
        }
277 1049537826
        s->s_buf[s->s_len++] = (char)c;
278 1049537826
}
279
280
/*
281
 * Append a byte string to an vsb.
282
 */
283
int
284 63197522
VSB_bcat(struct vsb *s, const void *buf, ssize_t len)
285
{
286 63197522
        assert_VSB_integrity(s);
287 63197522
        assert_VSB_state(s, 0);
288
289 63197522
        assert(len >= 0);
290 63197522
        if (s->s_error != 0)
291 0
                return (-1);
292 63197522
        if (len == 0)
293 3899228
                return (0);
294 59298294
        _vsb_indent(s);
295 59298294
        if (len > VSB_FREESPACE(s)) {
296 11749981
                if (VSB_extend(s, len - VSB_FREESPACE(s)) < 0)
297 200
                        s->s_error = ENOMEM;
298 11749981
                if (s->s_error != 0)
299 200
                        return (-1);
300 11749781
        }
301 59298094
        memcpy(s->s_buf + s->s_len, buf, len);
302 59298094
        s->s_len += len;
303 59298094
        return (0);
304 63197522
}
305
306
/*
307
 * Append a string to an vsb.
308
 */
309
int
310 57711491
VSB_cat(struct vsb *s, const char *str)
311
{
312
        const char *nl;
313
        size_t l;
314
315 57711491
        assert_VSB_integrity(s);
316 57711491
        assert_VSB_state(s, 0);
317 57711491
        KASSERT(str != NULL,
318
            ("%s called with a NULL str pointer", __func__));
319
320 57711491
        if (s->s_error != 0)
321 125
                return (-1);
322
323 57733563
        while (s->s_indent > 0 && (nl = strchr(str, '\n')) != NULL) {
324 22197
                l = (nl - str) + 1;
325 22197
                if (VSB_bcat(s, str, l) < 0)
326 0
                        return (-1);
327 22197
                str += l;
328
        }
329
330 57711366
        l = strlen(str);
331 57711366
        return (VSB_bcat(s, str, l));
332 57711491
}
333
334
/*
335
 * Format the given argument list and append the resulting string to an vsb.
336
 */
337
int
338 265071650
VSB_vprintf(struct vsb *s, const char *fmt, va_list ap)
339
{
340
        va_list ap_copy;
341
        int len;
342
343 265071650
        assert_VSB_integrity(s);
344 265071650
        assert_VSB_state(s, 0);
345
346 265071650
        KASSERT(fmt != NULL,
347
            ("%s called with a NULL format string", __func__));
348
349 265071650
        if (s->s_error != 0)
350 0
                return (-1);
351 265071650
        _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 265071650
        do {
365 286946945
                va_copy(ap_copy, ap);
366 573893890
                len = vsnprintf(&s->s_buf[s->s_len], VSB_FREESPACE(s) + 1,
367 286946945
                    fmt, ap_copy);
368 286946945
                va_end(ap_copy);
369 286946945
                if (len < 0) {
370 0
                        s->s_error = errno;
371 0
                        return (-1);
372
                }
373 308823167
        } while (len > VSB_FREESPACE(s) &&
374 21876222
            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 265071650
        s->s_len += vmin_t(ssize_t, len, VSB_FREESPACE(s));
386 265071650
        if (!VSB_HASROOM(s) && !VSB_CANEXTEND(s))
387 575
                s->s_error = ENOMEM;
388
389 265071650
        KASSERT(s->s_len < s->s_size,
390
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
391
392 265071650
        if (s->s_error != 0)
393 633
                return (-1);
394 265071017
        return (0);
395 265071650
}
396
397
/*
398
 * Format the given arguments and append the resulting string to an vsb.
399
 */
400
int
401 143766416
VSB_printf(struct vsb *s, const char *fmt, ...)
402
{
403
        va_list ap;
404
        int result;
405
406 143766416
        va_start(ap, fmt);
407 143766416
        result = VSB_vprintf(s, fmt, ap);
408 143766416
        va_end(ap);
409 143766416
        return (result);
410
}
411
412
/*
413
 * Append a character to an vsb.
414
 */
415
int
416 1049539308
VSB_putc(struct vsb *s, int c)
417
{
418
419 1049539308
        VSB_put_byte(s, c);
420 1049539308
        if (s->s_error != 0)
421 0
                return (-1);
422 1049539308
        return (0);
423 1049539308
}
424
425
/*
426
 * Check if an vsb has an error.
427
 */
428
int
429 29541
VSB_error(const struct vsb *s)
430
{
431
432 29541
        return (s->s_error);
433
}
434
435
/*
436
 * Finish off an vsb.
437
 */
438
int
439 63736105
VSB_finish(struct vsb *s)
440
{
441
442 63736105
        assert_VSB_integrity(s);
443 63736105
        assert_VSB_state(s, 0);
444
445 63736105
        s->s_buf[s->s_len] = '\0';
446 63736105
        VSB_SETFLAG(s, VSB_FINISHED);
447 63736105
        errno = s->s_error;
448 63736105
        if (s->s_error)
449 775
                return (-1);
450 63735330
        return (0);
451 63736105
}
452
453
/*
454
 * Return a pointer to the vsb data.
455
 */
456
char *
457 65918286
VSB_data(const struct vsb *s)
458
{
459
460 65918286
        assert_VSB_integrity(s);
461 65918286
        assert_VSB_state(s, VSB_FINISHED);
462
463 65918286
        return (s->s_buf);
464
}
465
466
/*
467
 * Return the length of the vsb data.
468
 */
469
ssize_t
470 22740728
VSB_len(const struct vsb *s)
471
{
472
473 22740728
        assert_VSB_integrity(s);
474
        /* don't care if it's finished or not */
475
476 22740728
        if (s->s_error != 0)
477 0
                return (-1);
478 22740728
        return (s->s_len);
479 22740728
}
480
481
void
482 716809
VSB_fini(struct vsb *s)
483
{
484
485 716809
        assert_VSB_integrity(s);
486 716809
        assert(!VSB_ISDYNAMIC(s));
487 716809
        assert(!VSB_ISDYNSTRUCT(s));
488 716809
        memset(s, 0, sizeof(*s));
489 716809
}
490
491
void
492 26917942
VSB_destroy(struct vsb **s)
493
{
494
495 26917942
        AN(s);
496 26917942
        assert_VSB_integrity(*s);
497 26917942
        assert(VSB_ISDYNAMIC(*s));
498 26917942
        assert(VSB_ISDYNSTRUCT(*s));
499 26917942
        SBFREE((*s)->s_buf);
500 26917942
        memset(*s, 0, sizeof(**s));
501 26917942
        SBFREE(*s);
502 26917942
        *s = NULL;
503 26917942
}
504
505
/*
506
 * Quote a string
507
 */
508
509
static void
510 94557
vsb_quote_hex(struct vsb *s, const uint8_t *u, size_t len)
511
{
512
        const uint8_t *w;
513
514 94557
        VSB_cat(s, "0x");
515 224400
        for (w = u; w < u + len; w++)
516 210903
                if (*w != 0x00)
517 81060
                        break;
518 94557
        if (w == u + len && len > 4) {
519 375
                VSB_cat(s, "0...0");
520 375
        } else {
521 16631364
                for (w = u; w < u + len; w++)
522 16537182
                        VSB_printf(s, "%02x", *w);
523
        }
524 94557
}
525
526
void
527 5997896
VSB_quote_pfx(struct vsb *s, const char *pfx, const void *v, int len, int how)
528
{
529 5997896
        const uint8_t *p = v;
530
        const uint8_t *q;
531 5997896
        int quote = 0;
532
        int nl;
533
534 5997896
        nl = how &
535
            (VSB_QUOTE_JSON|VSB_QUOTE_HEX|VSB_QUOTE_CSTR|VSB_QUOTE_UNSAFE);
536 5997896
        AZ(nl & (nl - 1)); // Only one bit can be set
537
538 5997896
        if (how & VSB_QUOTE_ESCHEX)
539 2814331
                AZ(how & (VSB_QUOTE_JSON|VSB_QUOTE_HEX));
540
541 5997896
        if (how & VSB_QUOTE_UNSAFE)
542 2807407
                how |= VSB_QUOTE_NONL;
543
544 5997896
        assert(p != NULL);
545 5997896
        if (len == -1)
546 2901437
                len = strlen(v);
547
548 5997896
        if (len == 0 && (how & VSB_QUOTE_CSTR)) {
549 5050
                VSB_printf(s, "%s\"\"", pfx);
550 5050
                if ((how & VSB_QUOTE_NONL))
551 0
                        VSB_putc(s, '\n');
552 5050
        }
553
554 5997896
        if (len == 0)
555 196912
                return;
556
557 5800984
        VSB_cat(s, pfx);
558
559 5800984
        if (how & VSB_QUOTE_HEX) {
560 94557
                vsb_quote_hex(s, v, len);
561 94557
                if (how & VSB_QUOTE_NONL)
562 25
                        VSB_putc(s, '\n');
563 94557
                return;
564
        }
565
566 5706427
        if (how & VSB_QUOTE_CSTR)
567 2703395
                VSB_putc(s, '"');
568
569 123080581
        for (q = p; q < p + len; q++) {
570
                if (
571 119073253
                    *q < 0x20 ||
572 117536474
                    *q == '"' ||
573 117516795
                    *q == '\\' ||
574 117511686
                    (*q == '?' && (how & VSB_QUOTE_CSTR)) ||
575 117374154
                    (*q > 0x7e && !(how & VSB_QUOTE_JSON))
576
                ) {
577 1695803
                        quote++;
578 1695803
                        break;
579
                }
580 117374154
        }
581
582 5706427
        if (!quote) {
583 4010641
                VSB_bcat(s, p, len);
584 4010641
                if ((how & VSB_QUOTE_NONL) &&
585 1832694
                    p[len-1] != '\n')
586 1832697
                        (void)VSB_putc(s, '\n');
587 4010649
                if (how & VSB_QUOTE_CSTR)
588 1836570
                        VSB_putc(s, '"');
589 4010649
                return;
590
        }
591
592 1695786
        nl = 0;
593 361745499
        for (q = p; q < p + len; q++) {
594 360049713
                if (nl)
595 3719967
                        VSB_cat(s, pfx);
596 360049713
                nl = 0;
597 360049713
                switch (*q) {
598
                case '?':
599
                        /* Avoid C Trigraph insanity */
600 473345
                        if (how & VSB_QUOTE_CSTR && !(how & VSB_QUOTE_JSON))
601 193600
                                (void)VSB_putc(s, '\\');
602 473345
                        (void)VSB_putc(s, *q);
603 473345
                        break;
604
                case '\\':
605
                case '"':
606 3644594
                        if (!(how & VSB_QUOTE_UNSAFE))
607 2520980
                                (void)VSB_putc(s, '\\');
608 3644594
                        (void)VSB_putc(s, *q);
609 3644594
                        break;
610
                case '\n':
611 15275782
                        if (how & VSB_QUOTE_CSTR) {
612 10764400
                                VSB_printf(s, "\\n\"\n%s\"", pfx);
613 15275782
                        } else if (how & VSB_QUOTE_JSON) {
614 35175
                                VSB_cat(s, "\\n");
615 4511382
                        } else if (how & VSB_QUOTE_NONL) {
616 4475882
                                VSB_putc(s, *q);
617 4475882
                                nl = 1;
618 4475882
                        } else {
619 325
                                VSB_cat(s, "\\n");
620
                        }
621 15275782
                        break;
622
                case '\r':
623 1883122
                        VSB_cat(s, "\\r");
624 1883122
                        break;
625
                case '\t':
626 4972798
                        VSB_cat(s, "\\t");
627 4972798
                        break;
628
                default:
629 333800072
                        if (0x20 <= *q && *q <= 0x7e)
630 333580672
                                VSB_putc(s, *q);
631 219400
                        else if (*q > 0x7e && (how & VSB_QUOTE_JSON))
632 100
                                VSB_putc(s, *q);
633 219300
                        else if (how & VSB_QUOTE_JSON)
634 175
                                VSB_printf(s, "\\u%04x", *q);
635 219125
                        else if (how & VSB_QUOTE_ESCHEX)
636 217525
                                VSB_printf(s, "\\x%02x", *q);
637
                        else
638 1600
                                VSB_printf(s, "\\%03o", *q);
639 333800072
                        break;
640
                }
641 360049713
        }
642 1695786
        if (how & VSB_QUOTE_CSTR)
643 866825
                VSB_putc(s, '"');
644 1695786
        if ((how & VSB_QUOTE_NONL) && !nl)
645 63467
                VSB_putc(s, '\n');
646 5997904
}
647
648
void
649 3092600
VSB_quote(struct vsb *s, const void *v, int len, int how)
650
{
651 3092600
        VSB_quote_pfx(s, "", v, len, how);
652 3092600
}
653
654
/*
655
 * Indentation
656
 */
657
658
void
659 57064
VSB_indent(struct vsb *s, int i)
660
{
661
662 57064
        assert_VSB_integrity(s);
663 57064
        if (s->s_indent + i < 0)
664 0
                s->s_error = EINVAL;
665
        else
666 57064
                s->s_indent += i;
667 57064
}
668
669
int
670 1564744
VSB_tofile(const struct vsb *s, int fd)
671
{
672
        const char *p;
673
        ssize_t r;
674
        size_t sz;
675
676 1564744
        assert_VSB_integrity(s);
677 1564744
        assert_VSB_state(s, VSB_FINISHED);
678 1564744
        assert(s->s_len >= 0);
679 1564744
        r = 0;
680 1564744
        p = s->s_buf;
681 1564744
        sz = (typeof(sz))s->s_len;
682 3129272
        while (sz > 0) {
683 1564692
                r = write(fd, p, sz);
684 1564692
                if (r < 0)
685 164
                        return (-1);
686 1564528
                assert((typeof(sz))r <= sz);
687 1564528
                p += r;
688 1564528
                sz -= r;
689
        }
690 1564580
        return (0);
691 1564744
}