xref: /netbsd-src/external/bsd/kyua-testers/dist/text.c (revision 754f425fc237c181450c91977727274098801c74)
1*754f425fSjmmv // Copyright 2012 Google Inc.
2*754f425fSjmmv // All rights reserved.
3*754f425fSjmmv //
4*754f425fSjmmv // Redistribution and use in source and binary forms, with or without
5*754f425fSjmmv // modification, are permitted provided that the following conditions are
6*754f425fSjmmv // met:
7*754f425fSjmmv //
8*754f425fSjmmv // * Redistributions of source code must retain the above copyright
9*754f425fSjmmv //   notice, this list of conditions and the following disclaimer.
10*754f425fSjmmv // * Redistributions in binary form must reproduce the above copyright
11*754f425fSjmmv //   notice, this list of conditions and the following disclaimer in the
12*754f425fSjmmv //   documentation and/or other materials provided with the distribution.
13*754f425fSjmmv // * Neither the name of Google Inc. nor the names of its contributors
14*754f425fSjmmv //   may be used to endorse or promote products derived from this software
15*754f425fSjmmv //   without specific prior written permission.
16*754f425fSjmmv //
17*754f425fSjmmv // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*754f425fSjmmv // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*754f425fSjmmv // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*754f425fSjmmv // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*754f425fSjmmv // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*754f425fSjmmv // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*754f425fSjmmv // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*754f425fSjmmv // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*754f425fSjmmv // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*754f425fSjmmv // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*754f425fSjmmv // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*754f425fSjmmv 
29*754f425fSjmmv #include "text.h"
30*754f425fSjmmv 
31*754f425fSjmmv #include <assert.h>
32*754f425fSjmmv #include <errno.h>
33*754f425fSjmmv #include <stdarg.h>
34*754f425fSjmmv #include <stdlib.h>
35*754f425fSjmmv #include <stdio.h>
36*754f425fSjmmv 
37*754f425fSjmmv #include "error.h"
38*754f425fSjmmv 
39*754f425fSjmmv 
40*754f425fSjmmv /// Calculates the length of a formatting string with its replacements.
41*754f425fSjmmv ///
42*754f425fSjmmv /// \param format The formatting string.
43*754f425fSjmmv /// \param ap List of arguments to apply to the formatting string.
44*754f425fSjmmv ///
45*754f425fSjmmv /// \return -1 if there is an error; or the final length otherwise.
46*754f425fSjmmv static int
calculate_length(const char * format,va_list ap)47*754f425fSjmmv calculate_length(const char* format, va_list ap)
48*754f425fSjmmv {
49*754f425fSjmmv     va_list ap2;
50*754f425fSjmmv 
51*754f425fSjmmv     char buffer[1];
52*754f425fSjmmv     va_copy(ap2, ap);
53*754f425fSjmmv     const int needed = vsnprintf(buffer, sizeof(buffer), format, ap);
54*754f425fSjmmv     va_end(ap2);
55*754f425fSjmmv     return needed;
56*754f425fSjmmv }
57*754f425fSjmmv 
58*754f425fSjmmv 
59*754f425fSjmmv /// Generates a string from a format string and its replacements.
60*754f425fSjmmv ///
61*754f425fSjmmv /// \param [out] output Pointer to the dynamically-allocated string with the
62*754f425fSjmmv ///     result of the operation.  The caller must release this with free().
63*754f425fSjmmv /// \param format The formatting string.
64*754f425fSjmmv /// \param ... Arguments to apply to the formatting string.
65*754f425fSjmmv ///
66*754f425fSjmmv /// \return OK if the string could be formatted; an error otherwise.
67*754f425fSjmmv kyua_error_t
kyua_text_printf(char ** output,const char * format,...)68*754f425fSjmmv kyua_text_printf(char** output, const char* format, ...)
69*754f425fSjmmv {
70*754f425fSjmmv     kyua_error_t error;
71*754f425fSjmmv     va_list ap;
72*754f425fSjmmv 
73*754f425fSjmmv     va_start(ap, format);
74*754f425fSjmmv     error = kyua_text_vprintf(output, format, ap);
75*754f425fSjmmv     va_end(ap);
76*754f425fSjmmv 
77*754f425fSjmmv     return error;
78*754f425fSjmmv }
79*754f425fSjmmv 
80*754f425fSjmmv 
81*754f425fSjmmv /// Generates a string from a format string and its replacements.
82*754f425fSjmmv ///
83*754f425fSjmmv /// \param [out] output Pointer to the dynamically-allocated string with the
84*754f425fSjmmv ///     result of the operation.  The caller must release this with free().
85*754f425fSjmmv /// \param format The formatting string.
86*754f425fSjmmv /// \param ap List of to apply to the formatting string.
87*754f425fSjmmv ///
88*754f425fSjmmv /// \return OK if the string could be formatted; an error otherwise.
89*754f425fSjmmv kyua_error_t
kyua_text_vprintf(char ** output,const char * format,va_list ap)90*754f425fSjmmv kyua_text_vprintf(char** output, const char* format, va_list ap)
91*754f425fSjmmv {
92*754f425fSjmmv     va_list ap2;
93*754f425fSjmmv 
94*754f425fSjmmv     va_copy(ap2, ap);
95*754f425fSjmmv     const int length = calculate_length(format, ap2);
96*754f425fSjmmv     va_end(ap2);
97*754f425fSjmmv     if (length < 0)
98*754f425fSjmmv         return kyua_libc_error_new(errno, "Could not calculate length of "
99*754f425fSjmmv                                    "string with format '%s'", format);
100*754f425fSjmmv 
101*754f425fSjmmv     char* buffer = (char*)malloc(length + 1);
102*754f425fSjmmv     if (buffer == NULL)
103*754f425fSjmmv         return kyua_oom_error_new();
104*754f425fSjmmv 
105*754f425fSjmmv     va_copy(ap2, ap);
106*754f425fSjmmv     const int printed_length = vsnprintf(buffer, length + 1, format, ap2);
107*754f425fSjmmv     va_end(ap2);
108*754f425fSjmmv     assert(printed_length == length);
109*754f425fSjmmv     if (printed_length < 0) {
110*754f425fSjmmv         free(buffer);
111*754f425fSjmmv         return kyua_libc_error_new(errno, "Could generate string with format "
112*754f425fSjmmv                                    "'%s'", format);
113*754f425fSjmmv     }
114*754f425fSjmmv 
115*754f425fSjmmv     *output = buffer;
116*754f425fSjmmv     return kyua_error_ok();
117*754f425fSjmmv }
118