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 2855414725
_assert_VSB_integrity(const char *fun, const struct vsb *s)
82
{
83
84 2855414725
        (void)fun;
85 2855414725
        (void)s;
86 2855414725
        KASSERT(s != NULL,
87
            ("%s called with a NULL vsb pointer", fun));
88 2855414725
        KASSERT(s->magic == VSB_MAGIC,
89
            ("%s called with a bogus vsb pointer", fun));
90 2855414725
        KASSERT(s->s_buf != NULL,
91
            ("%s called with uninitialized or corrupt vsb", fun));
92 2855414725
        KASSERT(s->s_len < s->s_size,
93
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
94 2855414725
}
95
96
static void
97 2681531425
_assert_VSB_state(const char *fun, const struct vsb *s, int state)
98
{
99
100 2681531425
        (void)fun;
101 2681531425
        (void)s;
102 2681531425
        (void)state;
103 2681531425
        KASSERT((s->s_flags & VSB_FINISHED) == state,
104
            ("%s called with %sfinished or corrupt vsb", fun,
105
            (state ? "un" : "")));
106 2681531425
}
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 129858442
VSB_extendsize(ssize_t size)
121
{
122
        ssize_t newsize;
123
124 129858442
        if (size < (int)VSB_MAXEXTENDSIZE) {
125 128400761
                newsize = VSB_MINEXTENDSIZE;
126 306999264
                while (newsize < size)
127 178598503
                        newsize *= 2;
128 128400761
        } else {
129 1457681
                newsize = rndup2(size, VSB_MAXEXTENDINCR);
130
        }
131 129858442
        KASSERT(newsize >= size, ("%s: %d < %d\n", __func__, newsize, size));
132 129858442
        return (newsize);
133
}
134
135
/*
136
 * Extend an vsb.
137
 */
138
static ssize_t
139 74585422
VSB_extend(struct vsb *s, ssize_t addlen)
140
{
141
        char *newbuf;
142
        ssize_t newsize;
143
144 74585422
        if (!VSB_CANEXTEND(s))
145 1320
                return (-1);
146 74584102
        newsize = VSB_extendsize(s->s_size + addlen);
147 74584102
        if (VSB_ISDYNAMIC(s))
148 74584102
                newbuf = realloc(s->s_buf, newsize);
149
        else
150 0
                newbuf = SBMALLOC(newsize);
151 74584102
        if (newbuf == NULL)
152 0
                return (-1);
153 74584102
        if (!VSB_ISDYNAMIC(s)) {
154 0
                memcpy(newbuf, s->s_buf, s->s_size);
155 0
                VSB_SETFLAG(s, VSB_DYNAMIC);
156 0
        }
157 74584102
        s->s_buf = newbuf;
158 74584102
        s->s_size = newsize;
159 74584102
        return (0);
160 74585422
}
161
162
static void
163 2351544312
_vsb_indent(struct vsb *s)
164
{
165 2355407588
        if (s->s_indent == 0 || s->s_error != 0 ||
166 3863276
            (s->s_len > 0 && s->s_buf[s->s_len - 1] != '\n'))
167 2351265007
                return;
168 279305
        if (VSB_FREESPACE(s) <= s->s_indent &&
169 373
            VSB_extend(s, s->s_indent) < 0) {
170 0
                s->s_error = ENOMEM;
171 0
                return;
172
        }
173 279305
        memset(s->s_buf + s->s_len, ' ', s->s_indent);
174 279305
        s->s_len += s->s_indent;
175 2351544312
}
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 56504395
VSB_newbuf(struct vsb *s, char *buf, int length, int flags)
184
{
185
186 56504395
        memset(s, 0, sizeof(*s));
187 56504395
        s->magic = VSB_MAGIC;
188 56504395
        s->s_flags = flags;
189 56504395
        s->s_size = length;
190 56504395
        s->s_buf = buf;
191
192 56504395
        if ((s->s_flags & VSB_AUTOEXTEND) == 0) {
193 1229999
                KASSERT(s->s_size > 1,
194
                    ("attempt to create a too small vsb"));
195 1229999
        }
196
197 56504395
        if (s->s_buf != NULL)
198 1230002
                return (s);
199
200 55274393
        if ((flags & VSB_AUTOEXTEND) != 0)
201 55274370
                s->s_size = VSB_extendsize(s->s_size);
202
203 55274393
        s->s_buf = SBMALLOC(s->s_size);
204 55274393
        if (s->s_buf == NULL)
205 0
                return (NULL);
206 55274367
        VSB_SETFLAG(s, VSB_DYNAMIC);
207 55274367
        return (s);
208 56504369
}
209
210
struct vsb *
211 1230014
VSB_init(struct vsb *s, void *buf, ssize_t length)
212
{
213 1230014
        AN(s);
214 1230014
        AN(buf);
215
216 1230014
        KASSERT(length >= 0,
217
            ("attempt to create an vsb of negative length (%zd)", length));
218 1230014
        return (VSB_newbuf(s, buf, length, VSB_FIXEDLEN));
219
}
220
221
/*
222
 * Allocate a dynamic vsb
223
 */
224
struct vsb *
225 55274368
VSB_new_auto(void)
226
{
227
        struct vsb *s;
228
229 55274368
        s = SBMALLOC(sizeof(*s));
230 55274368
        if (s == NULL)
231 0
                return (NULL);
232 55274368
        if (VSB_newbuf(s, NULL, 0, VSB_AUTOEXTEND) == NULL) {
233 0
                SBFREE(s);
234 0
                return (NULL);
235
        }
236 55274368
        VSB_SETFLAG(s, VSB_DYNSTRUCT);
237 55274368
        return (s);
238 55274368
}
239
240
/*
241
 * Clear an vsb and reset its position.
242
 */
243
void
244 85762028
VSB_clear(struct vsb *s)
245
{
246
247 85762028
        assert_VSB_integrity(s);
248
        /* don't care if it's finished or not */
249
250 85762028
        VSB_CLEARFLAG(s, VSB_FINISHED);
251 85762028
        s->s_error = 0;
252 85762028
        s->s_len = 0;
253 85762028
        s->s_indent = 0;
254 85762028
}
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 1799280460
VSB_put_byte(struct vsb *s, int c)
263
{
264
265 1799280460
        assert_VSB_integrity(s);
266 1799280460
        assert_VSB_state(s, 0);
267
268 1799280460
        if (s->s_error != 0)
269 0
                return;
270 1799280460
        _vsb_indent(s);
271 1799280460
        if (VSB_FREESPACE(s) <= 0) {
272 17410786
                if (VSB_extend(s, 1) < 0)
273 0
                        s->s_error = ENOMEM;
274 17410786
                if (s->s_error != 0)
275 0
                        return;
276 17410786
        }
277 1799280460
        s->s_buf[s->s_len++] = (char)c;
278 1799280460
}
279
280
/*
281
 * Append a byte string to an vsb.
282
 */
283
int
284 109855817
VSB_bcat(struct vsb *s, const void *buf, ssize_t len)
285
{
286 109855817
        assert_VSB_integrity(s);
287 109855817
        assert_VSB_state(s, 0);
288
289 109855817
        assert(len >= 0);
290 109855817
        if (s->s_error != 0)
291 0
                return (-1);
292 109855817
        if (len == 0)
293 6427880
                return (0);
294 103427937
        _vsb_indent(s);
295 103427937
        if (len > VSB_FREESPACE(s)) {
296 20278312
                if (VSB_extend(s, len - VSB_FREESPACE(s)) < 0)
297 320
                        s->s_error = ENOMEM;
298 20278312
                if (s->s_error != 0)
299 320
                        return (-1);
300 20277992
        }
301 103427617
        memcpy(s->s_buf + s->s_len, buf, len);
302 103427617
        s->s_len += len;
303 103427617
        return (0);
304 109855817
}
305
306
/*
307
 * Append a string to an vsb.
308
 */
309
int
310 100808244
VSB_cat(struct vsb *s, const char *str)
311
{
312
        const char *nl;
313
        size_t l;
314
315 100808244
        assert_VSB_integrity(s);
316 100808244
        assert_VSB_state(s, 0);
317 100808244
        KASSERT(str != NULL,
318
            ("%s called with a NULL str pointer", __func__));
319
320 100808244
        if (s->s_error != 0)
321 200
                return (-1);
322
323 100848261
        while (s->s_indent > 0 && (nl = strchr(str, '\n')) != NULL) {
324 40217
                l = (nl - str) + 1;
325 40217
                if (VSB_bcat(s, str, l) < 0)
326 0
                        return (-1);
327 40217
                str += l;
328
        }
329
330 100808044
        l = strlen(str);
331 100808044
        return (VSB_bcat(s, str, l));
332 100808244
}
333
334
/*
335
 * Format the given argument list and append the resulting string to an vsb.
336
 */
337
int
338 449060534
VSB_vprintf(struct vsb *s, const char *fmt, va_list ap)
339
{
340
        va_list ap_copy;
341
        int len;
342
343 449060534
        assert_VSB_integrity(s);
344 449060534
        assert_VSB_state(s, 0);
345
346 449060534
        KASSERT(fmt != NULL,
347
            ("%s called with a NULL format string", __func__));
348
349 449060534
        if (s->s_error != 0)
350 0
                return (-1);
351 449060534
        _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 449060534
        do {
365 485954775
                va_copy(ap_copy, ap);
366 971909550
                len = vsnprintf(&s->s_buf[s->s_len], VSB_FREESPACE(s) + 1,
367 485954775
                    fmt, ap_copy);
368 485954775
                va_end(ap_copy);
369 485954775
                if (len < 0) {
370 0
                        s->s_error = errno;
371 0
                        return (-1);
372
                }
373 522850818
        } while (len > VSB_FREESPACE(s) &&
374 36896043
            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 449060534
        s->s_len += vmin_t(ssize_t, len, VSB_FREESPACE(s));
386 449060534
        if (!VSB_HASROOM(s) && !VSB_CANEXTEND(s))
387 1000
                s->s_error = ENOMEM;
388
389 449060534
        KASSERT(s->s_len < s->s_size,
390
            ("wrote past end of vsb (%d >= %d)", s->s_len, s->s_size));
391
392 449060534
        if (s->s_error != 0)
393 1066
                return (-1);
394 449059484
        return (0);
395 449060550
}
396
397
/*
398
 * Format the given arguments and append the resulting string to an vsb.
399
 */
400
int
401 243001270
VSB_printf(struct vsb *s, const char *fmt, ...)
402
{
403
        va_list ap;
404
        int result;
405
406 243001270
        va_start(ap, fmt);
407 243001270
        result = VSB_vprintf(s, fmt, ap);
408 243001270
        va_end(ap);
409 243001270
        return (result);
410
}
411
412
/*
413
 * Append a character to an vsb.
414
 */
415
int
416 1799294834
VSB_putc(struct vsb *s, int c)
417
{
418
419 1799294834
        VSB_put_byte(s, c);
420 1799294834
        if (s->s_error != 0)
421 0
                return (-1);
422 1799294834
        return (0);
423 1799294834
}
424
425
/*
426
 * Check if an vsb has an error.
427
 */
428
int
429 48548
VSB_error(const struct vsb *s)
430
{
431
432 48548
        return (s->s_error);
433
}
434
435
/*
436
 * Finish off an vsb.
437
 */
438
int
439 108445270
VSB_finish(struct vsb *s)
440
{
441
442 108445270
        assert_VSB_integrity(s);
443 108445270
        assert_VSB_state(s, 0);
444
445 108445270
        s->s_buf[s->s_len] = '\0';
446 108445270
        VSB_SETFLAG(s, VSB_FINISHED);
447 108445270
        errno = s->s_error;
448 108445270
        if (s->s_error)
449 1320
                return (-1);
450 108443950
        return (0);
451 108445270
}
452
453
/*
454
 * Return a pointer to the vsb data.
455
 */
456
char *
457 111972042
VSB_data(const struct vsb *s)
458
{
459
460 111972042
        assert_VSB_integrity(s);
461 111972042
        assert_VSB_state(s, VSB_FINISHED);
462
463 111972042
        return (s->s_buf);
464
}
465
466
/*
467
 * Return the length of the vsb data.
468
 */
469
ssize_t
470 41347074
VSB_len(const struct vsb *s)
471
{
472
473 41347074
        assert_VSB_integrity(s);
474
        /* don't care if it's finished or not */
475
476 41347074
        if (s->s_error != 0)
477 0
                return (-1);
478 41347074
        return (s->s_len);
479 41347074
}
480
481
void
482 1193058
VSB_fini(struct vsb *s)
483
{
484
485 1193058
        assert_VSB_integrity(s);
486 1193058
        assert(!VSB_ISDYNAMIC(s));
487 1193058
        assert(!VSB_ISDYNSTRUCT(s));
488 1193058
        memset(s, 0, sizeof(*s));
489 1193058
}
490
491
void
492 45785961
VSB_destroy(struct vsb **s)
493
{
494
495 45785961
        AN(s);
496 45785961
        assert_VSB_integrity(*s);
497 45785961
        assert(VSB_ISDYNAMIC(*s));
498 45785961
        assert(VSB_ISDYNSTRUCT(*s));
499 45785961
        SBFREE((*s)->s_buf);
500 45785961
        memset(*s, 0, sizeof(**s));
501 45785961
        SBFREE(*s);
502 45785961
        *s = NULL;
503 45785961
}
504
505
/*
506
 * Quote a string
507
 */
508
509
static void
510 154883
vsb_quote_hex(struct vsb *s, const uint8_t *u, size_t len)
511
{
512
        const uint8_t *w;
513
514 154883
        VSB_cat(s, "0x");
515 371670
        for (w = u; w < u + len; w++)
516 349394
                if (*w != 0x00)
517 132607
                        break;
518 154883
        if (w == u + len && len > 4) {
519 680
                VSB_cat(s, "0...0");
520 680
        } else {
521 26663637
                for (w = u; w < u + len; w++)
522 26509434
                        VSB_printf(s, "%02x", *w);
523
        }
524 154883
}
525
526
void
527 9943178
VSB_quote_pfx(struct vsb *s, const char *pfx, const void *v, int len, int how)
528
{
529 9943178
        const uint8_t *p = v;
530
        const uint8_t *q;
531 9943178
        int quote = 0;
532
        int nl;
533
534 9943178
        nl = how &
535
            (VSB_QUOTE_JSON|VSB_QUOTE_HEX|VSB_QUOTE_CSTR|VSB_QUOTE_UNSAFE);
536 9943178
        AZ(nl & (nl - 1)); // Only one bit can be set
537
538 9943178
        if (how & VSB_QUOTE_ESCHEX)
539 4706933
                AZ(how & (VSB_QUOTE_JSON|VSB_QUOTE_HEX));
540
541 9943178
        if (how & VSB_QUOTE_UNSAFE)
542 4695694
                how |= VSB_QUOTE_NONL;
543
544 9943178
        assert(p != NULL);
545 9943178
        if (len == -1)
546 4771456
                len = strlen(v);
547
548 9943178
        if (len == 0 && (how & VSB_QUOTE_CSTR)) {
549 8160
                VSB_printf(s, "%s\"\"", pfx);
550 8160
                if ((how & VSB_QUOTE_NONL))
551 0
                        VSB_putc(s, '\n');
552 8160
        }
553
554 9943178
        if (len == 0)
555 322986
                return;
556
557 9620192
        VSB_cat(s, pfx);
558
559 9620192
        if (how & VSB_QUOTE_HEX) {
560 154883
                vsb_quote_hex(s, v, len);
561 154883
                if (how & VSB_QUOTE_NONL)
562 40
                        VSB_putc(s, '\n');
563 154883
                return;
564
        }
565
566 9465309
        if (how & VSB_QUOTE_CSTR)
567 4448525
                VSB_putc(s, '"');
568
569 199659581
        for (q = p; q < p + len; q++) {
570
                if (
571 193039285
                    *q < 0x20 ||
572 190462691
                    *q == '"' ||
573 190428936
                    *q == '\\' ||
574 190420618
                    (*q == '?' && (how & VSB_QUOTE_CSTR)) ||
575 190194272
                    (*q > 0x7e && !(how & VSB_QUOTE_JSON))
576
                ) {
577 2839738
                        quote++;
578 2839738
                        break;
579
                }
580 190194272
        }
581
582 9465309
        if (!quote) {
583 6625573
                VSB_bcat(s, p, len);
584 6625573
                if ((how & VSB_QUOTE_NONL) &&
585 3043024
                    p[len-1] != '\n')
586 3043021
                        (void)VSB_putc(s, '\n');
587 6625573
                if (how & VSB_QUOTE_CSTR)
588 3022085
                        VSB_putc(s, '"');
589 6625573
                return;
590
        }
591
592 2839736
        nl = 0;
593 628764264
        for (q = p; q < p + len; q++) {
594 625924528
                if (nl)
595 6509960
                        VSB_cat(s, pfx);
596 625924528
                nl = 0;
597 625924528
                switch (*q) {
598
                case '?':
599
                        /* Avoid C Trigraph insanity */
600 950587
                        if (how & VSB_QUOTE_CSTR && !(how & VSB_QUOTE_JSON))
601 318880
                                (void)VSB_putc(s, '\\');
602 950587
                        (void)VSB_putc(s, *q);
603 950587
                        break;
604
                case '\\':
605
                case '"':
606 6314980
                        if (!(how & VSB_QUOTE_UNSAFE))
607 4160918
                                (void)VSB_putc(s, '\\');
608 6314980
                        (void)VSB_putc(s, *q);
609 6314980
                        break;
610
                case '\n':
611 26064811
                        if (how & VSB_QUOTE_CSTR) {
612 18202120
                                VSB_printf(s, "\\n\"\n%s\"", pfx);
613 26064811
                        } else if (how & VSB_QUOTE_JSON) {
614 58120
                                VSB_cat(s, "\\n");
615 7862691
                        } else if (how & VSB_QUOTE_NONL) {
616 7804051
                                VSB_putc(s, *q);
617 7804051
                                nl = 1;
618 7804051
                        } else {
619 520
                                VSB_cat(s, "\\n");
620
                        }
621 26064811
                        break;
622
                case '\r':
623 3132618
                        VSB_cat(s, "\\r");
624 3132618
                        break;
625
                case '\t':
626 8638043
                        VSB_cat(s, "\\t");
627 8638043
                        break;
628
                default:
629 580823489
                        if (0x20 <= *q && *q <= 0x7e)
630 580470772
                                VSB_putc(s, *q);
631 352717
                        else if (*q > 0x7e && (how & VSB_QUOTE_JSON))
632 160
                                VSB_putc(s, *q);
633 352557
                        else if (how & VSB_QUOTE_JSON)
634 280
                                VSB_printf(s, "\\u%04x", *q);
635 352277
                        else if (how & VSB_QUOTE_ESCHEX)
636 349717
                                VSB_printf(s, "\\x%02x", *q);
637
                        else
638 2560
                                VSB_printf(s, "\\%03o", *q);
639 580823489
                        break;
640
                }
641 625924528
        }
642 2839736
        if (how & VSB_QUOTE_CSTR)
643 1426440
                VSB_putc(s, '"');
644 2839736
        if ((how & VSB_QUOTE_NONL) && !nl)
645 103216
                VSB_putc(s, '\n');
646 9943178
}
647
648
void
649 5086647
VSB_quote(struct vsb *s, const void *v, int len, int how)
650
{
651 5086647
        VSB_quote_pfx(s, "", v, len, how);
652 5086647
}
653
654
/*
655
 * Indentation
656
 */
657
658
void
659 95552
VSB_indent(struct vsb *s, int i)
660
{
661
662 95552
        assert_VSB_integrity(s);
663 95552
        if (s->s_indent + i < 0)
664 0
                s->s_error = EINVAL;
665
        else
666 95552
                s->s_indent += i;
667 95552
}
668
669
int
670 2693252
VSB_tofile(const struct vsb *s, int fd)
671
{
672
        const char *p;
673
        ssize_t r;
674
        size_t sz;
675
676 2693252
        assert_VSB_integrity(s);
677 2693252
        assert_VSB_state(s, VSB_FINISHED);
678 2693252
        assert(s->s_len >= 0);
679 2693252
        r = 0;
680 2693252
        p = s->s_buf;
681 2693252
        sz = (typeof(sz))s->s_len;
682 5386115
        while (sz > 0) {
683 2693133
                r = write(fd, p, sz);
684 2693133
                if (r < 0)
685 270
                        return (-1);
686 2692863
                assert((typeof(sz))r <= sz);
687 2692863
                p += r;
688 2692863
                sz -= r;
689
        }
690 2692982
        return (0);
691 2693252
}