vasprintf.c revision 1.14 1 1.14 christos /* $NetBSD: vasprintf.c,v 1.14 2012/03/15 18:22:30 christos Exp $ */
2 1.1 perry
3 1.1 perry /*
4 1.1 perry * Copyright (c) 1997 Todd C. Miller <Todd.Miller (at) courtesan.com>
5 1.1 perry * All rights reserved.
6 1.1 perry *
7 1.1 perry * Redistribution and use in source and binary forms, with or without
8 1.1 perry * modification, are permitted provided that the following conditions
9 1.1 perry * are met:
10 1.1 perry * 1. Redistributions of source code must retain the above copyright
11 1.1 perry * notice, this list of conditions and the following disclaimer.
12 1.1 perry * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 perry * notice, this list of conditions and the following disclaimer in the
14 1.1 perry * documentation and/or other materials provided with the distribution.
15 1.1 perry * 3. The name of the author may not be used to endorse or promote products
16 1.1 perry * derived from this software without specific prior written permission.
17 1.1 perry *
18 1.1 perry * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
19 1.1 perry * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
20 1.1 perry * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
21 1.1 perry * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22 1.1 perry * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23 1.1 perry * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
24 1.1 perry * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 1.1 perry * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26 1.1 perry * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
27 1.1 perry * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 1.1 perry */
29 1.1 perry
30 1.1 perry #include <sys/cdefs.h>
31 1.1 perry #if defined(LIBC_SCCS) && !defined(lint)
32 1.14 christos __RCSID("$NetBSD: vasprintf.c,v 1.14 2012/03/15 18:22:30 christos Exp $");
33 1.1 perry #endif /* LIBC_SCCS and not lint */
34 1.1 perry
35 1.5 lukem #include <assert.h>
36 1.5 lukem #include <errno.h>
37 1.1 perry #include <stdio.h>
38 1.1 perry #include <stdlib.h>
39 1.9 thorpej #include "reentrant.h"
40 1.8 yamt #include "local.h"
41 1.1 perry
42 1.1 perry int
43 1.13 joerg vasprintf(char **str, const char *fmt, va_list ap)
44 1.1 perry {
45 1.1 perry int ret;
46 1.1 perry FILE f;
47 1.8 yamt struct __sfileext fext;
48 1.1 perry unsigned char *_base;
49 1.5 lukem
50 1.5 lukem _DIAGASSERT(str != NULL);
51 1.5 lukem _DIAGASSERT(fmt != NULL);
52 1.1 perry
53 1.8 yamt _FILEEXT_SETUP(&f, &fext);
54 1.7 mycroft f._file = -1;
55 1.1 perry f._flags = __SWR | __SSTR | __SALC;
56 1.1 perry f._bf._base = f._p = (unsigned char *)malloc(128);
57 1.3 mycroft if (f._bf._base == NULL)
58 1.3 mycroft goto err;
59 1.2 kleink f._bf._size = f._w = 127; /* Leave room for the NUL */
60 1.10 kleink ret = __vfprintf_unlocked(&f, fmt, ap);
61 1.3 mycroft if (ret == -1)
62 1.3 mycroft goto err;
63 1.1 perry *f._p = '\0';
64 1.12 christos _base = realloc(f._bf._base, (size_t)(ret + 1));
65 1.3 mycroft if (_base == NULL)
66 1.3 mycroft goto err;
67 1.3 mycroft *str = (char *)_base;
68 1.14 christos return ret;
69 1.3 mycroft
70 1.3 mycroft err:
71 1.3 mycroft if (f._bf._base)
72 1.3 mycroft free(f._bf._base);
73 1.3 mycroft *str = NULL;
74 1.3 mycroft errno = ENOMEM;
75 1.14 christos return -1;
76 1.1 perry }
77