blob: 34abd2f8cf47409bfe9973a1dfba78f96b861f06 [file] [log] [blame]
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +05301/****************************************************************************
Steve Kondikae271bc2015-11-15 02:50:53 +01002 * Copyright (c) 1998-2012,2013 Free Software Foundation, Inc. *
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +05303 * *
4 * Permission is hereby granted, free of charge, to any person obtaining a *
5 * copy of this software and associated documentation files (the *
6 * "Software"), to deal in the Software without restriction, including *
7 * without limitation the rights to use, copy, modify, merge, publish, *
8 * distribute, distribute with modifications, sublicense, and/or sell *
9 * copies of the Software, and to permit persons to whom the Software is *
10 * furnished to do so, subject to the following conditions: *
11 * *
12 * The above copyright notice and this permission notice shall be included *
13 * in all copies or substantial portions of the Software. *
14 * *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
16 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
17 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
18 * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
19 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
20 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
21 * THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
22 * *
23 * Except as contained in this notice, the name(s) of the above copyright *
24 * holders shall not be used in advertising or otherwise to promote the *
25 * sale, use or other dealings in this Software without prior written *
26 * authorization. *
27 ****************************************************************************/
28
29/****************************************************************************
Steve Kondikae271bc2015-11-15 02:50:53 +010030 * Author: Thomas E. Dickey 1997-on *
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +053031 ****************************************************************************/
32
33#include <curses.priv.h>
34#include <ctype.h>
35
Steve Kondikae271bc2015-11-15 02:50:53 +010036MODULE_ID("$Id: safe_sprintf.c,v 1.27 2013/01/20 01:04:32 tom Exp $")
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +053037
38#if USE_SAFE_SPRINTF
39
40typedef enum {
41 Flags, Width, Prec, Type, Format
42} PRINTF;
43
44#define VA_INTGR(type) ival = va_arg(ap, type)
45#define VA_FLOAT(type) fval = va_arg(ap, type)
46#define VA_POINT(type) pval = (void *)va_arg(ap, type)
47
48/*
49 * Scan a variable-argument list for printf to determine the number of
50 * characters that would be emitted.
51 */
52static int
53_nc_printf_length(const char *fmt, va_list ap)
54{
55 size_t length = BUFSIZ;
56 char *buffer;
57 char *format;
58 int len = 0;
59 size_t fmt_len;
60 char fmt_arg[BUFSIZ];
61
62 if (fmt == 0 || *fmt == '\0')
63 return 0;
64 fmt_len = strlen(fmt) + 1;
65 if ((format = typeMalloc(char, fmt_len)) == 0)
66 return -1;
67 if ((buffer = typeMalloc(char, length)) == 0) {
68 free(format);
69 return -1;
70 }
71
72 while (*fmt != '\0') {
73 if (*fmt == '%') {
74 static char dummy[] = "";
75 PRINTF state = Flags;
76 char *pval = dummy; /* avoid const-cast */
77 double fval = 0.0;
78 int done = FALSE;
79 int ival = 0;
80 int prec = -1;
81 int type = 0;
82 int used = 0;
83 int width = -1;
84 size_t f = 0;
85
86 format[f++] = *fmt;
87 while (*++fmt != '\0' && len >= 0 && !done) {
88 format[f++] = *fmt;
89
90 if (isdigit(UChar(*fmt))) {
91 int num = *fmt - '0';
92 if (state == Flags && num != 0)
93 state = Width;
94 if (state == Width) {
95 if (width < 0)
96 width = 0;
97 width = (width * 10) + num;
98 } else if (state == Prec) {
99 if (prec < 0)
100 prec = 0;
101 prec = (prec * 10) + num;
102 }
103 } else if (*fmt == '*') {
104 VA_INTGR(int);
105 if (state == Flags)
106 state = Width;
107 if (state == Width) {
108 width = ival;
109 } else if (state == Prec) {
110 prec = ival;
111 }
Steve Kondikae271bc2015-11-15 02:50:53 +0100112 _nc_SPRINTF(fmt_arg,
113 _nc_SLIMIT(sizeof(fmt_arg))
114 "%d", ival);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530115 fmt_len += strlen(fmt_arg);
Steve Kondikae271bc2015-11-15 02:50:53 +0100116 if ((format = _nc_doalloc(format, fmt_len)) == 0) {
117 free(buffer);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530118 return -1;
119 }
Steve Kondikae271bc2015-11-15 02:50:53 +0100120 --f;
121 _nc_STRCPY(&format[f], fmt_arg, fmt_len - f);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530122 f = strlen(format);
123 } else if (isalpha(UChar(*fmt))) {
124 done = TRUE;
125 switch (*fmt) {
126 case 'Z': /* FALLTHRU */
127 case 'h': /* FALLTHRU */
128 case 'l': /* FALLTHRU */
129 done = FALSE;
130 type = *fmt;
131 break;
132 case 'i': /* FALLTHRU */
133 case 'd': /* FALLTHRU */
134 case 'u': /* FALLTHRU */
135 case 'x': /* FALLTHRU */
136 case 'X': /* FALLTHRU */
137 if (type == 'l')
138 VA_INTGR(long);
139 else if (type == 'Z')
140 VA_INTGR(size_t);
141 else
142 VA_INTGR(int);
143 used = 'i';
144 break;
145 case 'f': /* FALLTHRU */
146 case 'e': /* FALLTHRU */
147 case 'E': /* FALLTHRU */
148 case 'g': /* FALLTHRU */
149 case 'G': /* FALLTHRU */
150 VA_FLOAT(double);
151 used = 'f';
152 break;
153 case 'c':
154 VA_INTGR(int);
155 used = 'i';
156 break;
157 case 's':
158 VA_POINT(char *);
159 if (prec < 0)
160 prec = strlen(pval);
161 if (prec > (int) length) {
162 length = length + prec;
163 buffer = typeRealloc(char, length, buffer);
164 if (buffer == 0) {
165 free(format);
166 return -1;
167 }
168 }
169 used = 'p';
170 break;
171 case 'p':
172 VA_POINT(void *);
173 used = 'p';
174 break;
175 case 'n':
176 VA_POINT(int *);
177 used = 0;
178 break;
179 default:
180 break;
181 }
182 } else if (*fmt == '.') {
183 state = Prec;
184 } else if (*fmt == '%') {
185 done = TRUE;
186 used = 'p';
187 }
188 }
189 format[f] = '\0';
190 switch (used) {
191 case 'i':
Steve Kondikae271bc2015-11-15 02:50:53 +0100192 _nc_SPRINTF(buffer, _nc_SLIMIT(length) format, ival);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530193 break;
194 case 'f':
Steve Kondikae271bc2015-11-15 02:50:53 +0100195 _nc_SPRINTF(buffer, _nc_SLIMIT(length) format, fval);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530196 break;
197 default:
Steve Kondikae271bc2015-11-15 02:50:53 +0100198 _nc_SPRINTF(buffer, _nc_SLIMIT(length) format, pval);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530199 break;
200 }
201 len += (int) strlen(buffer);
202 } else {
203 fmt++;
204 len++;
205 }
206 }
207
208 free(buffer);
209 free(format);
210 return len;
211}
212#endif
213
214#define my_buffer _nc_globals.safeprint_buf
215#define my_length _nc_globals.safeprint_used
216
217/*
218 * Wrapper for vsprintf that allocates a buffer big enough to hold the result.
219 */
220NCURSES_EXPORT(char *)
Steve Kondikae271bc2015-11-15 02:50:53 +0100221NCURSES_SP_NAME(_nc_printf_string) (NCURSES_SP_DCLx
222 const char *fmt,
223 va_list ap)
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530224{
225 char *result = 0;
226
227 if (fmt != 0) {
228#if USE_SAFE_SPRINTF
Steve Kondikae271bc2015-11-15 02:50:53 +0100229 va_list ap2;
230 int len;
231
232 begin_va_copy(ap2, ap);
233 len = _nc_printf_length(fmt, ap2);
234 end_va_copy(ap2);
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530235
236 if ((int) my_length < len + 1) {
237 my_length = 2 * (len + 1);
238 my_buffer = typeRealloc(char, my_length, my_buffer);
239 }
240 if (my_buffer != 0) {
241 *my_buffer = '\0';
242 if (len >= 0) {
243 vsprintf(my_buffer, fmt, ap);
244 }
245 result = my_buffer;
246 }
247#else
248#define MyCols _nc_globals.safeprint_cols
249#define MyRows _nc_globals.safeprint_rows
250
Steve Kondikae271bc2015-11-15 02:50:53 +0100251 if (screen_lines(SP_PARM) > MyRows || screen_columns(SP_PARM) > MyCols) {
252 if (screen_lines(SP_PARM) > MyRows)
253 MyRows = screen_lines(SP_PARM);
254 if (screen_columns(SP_PARM) > MyCols)
255 MyCols = screen_columns(SP_PARM);
256 my_length = (size_t) (MyRows * (MyCols + 1)) + 1;
Amit Daniel Kachhape6a01f52011-07-20 11:45:59 +0530257 my_buffer = typeRealloc(char, my_length, my_buffer);
258 }
259
260 if (my_buffer != 0) {
261# if HAVE_VSNPRINTF
262 vsnprintf(my_buffer, my_length, fmt, ap); /* GNU extension */
263# else
264 vsprintf(my_buffer, fmt, ap); /* ANSI */
265# endif
266 result = my_buffer;
267 }
268#endif
269 } else if (my_buffer != 0) { /* see _nc_freeall() */
270 free(my_buffer);
271 my_buffer = 0;
272 my_length = 0;
273 }
274 return result;
275}
Steve Kondikae271bc2015-11-15 02:50:53 +0100276
277#if NCURSES_SP_FUNCS
278NCURSES_EXPORT(char *)
279_nc_printf_string(const char *fmt, va_list ap)
280{
281 return NCURSES_SP_NAME(_nc_printf_string) (CURRENT_SCREEN, fmt, ap);
282}
283#endif