1998-03-01 12:21:12 +08:00
|
|
|
/****************************************************************************
|
2004-02-09 10:15:26 +08:00
|
|
|
* Copyright (c) 1998-2001,2003 Free Software Foundation, Inc. *
|
1998-03-01 12:21:12 +08:00
|
|
|
* *
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a *
|
|
|
|
* copy of this software and associated documentation files (the *
|
|
|
|
* "Software"), to deal in the Software without restriction, including *
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish, *
|
|
|
|
* distribute, distribute with modifications, sublicense, and/or sell *
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is *
|
|
|
|
* furnished to do so, subject to the following conditions: *
|
|
|
|
* *
|
|
|
|
* The above copyright notice and this permission notice shall be included *
|
|
|
|
* in all copies or substantial portions of the Software. *
|
|
|
|
* *
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
|
|
|
|
* IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
|
|
|
|
* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
|
|
|
|
* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
|
|
|
|
* THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
|
|
|
|
* *
|
|
|
|
* Except as contained in this notice, the name(s) of the above copyright *
|
|
|
|
* holders shall not be used in advertising or otherwise to promote the *
|
|
|
|
* sale, use or other dealings in this Software without prior written *
|
|
|
|
* authorization. *
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Author: Thomas E. Dickey <dickey@clark.net> 1997 *
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <curses.priv.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
2004-02-09 10:15:26 +08:00
|
|
|
MODULE_ID("$Id: safe_sprintf.c,v 1.18 2003/08/09 21:52:04 tom Exp $")
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
#if USE_SAFE_SPRINTF
|
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
typedef enum {
|
|
|
|
Flags, Width, Prec, Type, Format
|
|
|
|
} PRINTF;
|
1998-03-01 12:21:12 +08:00
|
|
|
|
|
|
|
#define VA_INTGR(type) ival = va_arg(ap, type)
|
|
|
|
#define VA_FLOAT(type) fval = va_arg(ap, type)
|
|
|
|
#define VA_POINT(type) pval = (void *)va_arg(ap, type)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Scan a variable-argument list for printf to determine the number of
|
|
|
|
* characters that would be emitted.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
_nc_printf_length(const char *fmt, va_list ap)
|
|
|
|
{
|
2002-10-13 11:35:53 +08:00
|
|
|
size_t length = BUFSIZ;
|
|
|
|
char *buffer;
|
|
|
|
char *format;
|
|
|
|
int len = 0;
|
2004-02-09 10:15:26 +08:00
|
|
|
size_t fmt_len;
|
|
|
|
char fmt_arg[BUFSIZ];
|
2002-10-13 11:35:53 +08:00
|
|
|
|
|
|
|
if (fmt == 0 || *fmt == '\0')
|
2004-02-09 10:15:26 +08:00
|
|
|
return 0;
|
|
|
|
fmt_len = strlen(fmt) + 1;
|
|
|
|
if ((format = typeMalloc(char, fmt_len)) == 0)
|
2002-10-13 11:35:53 +08:00
|
|
|
return -1;
|
|
|
|
if ((buffer = typeMalloc(char, length)) == 0) {
|
|
|
|
free(format);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (*fmt != '\0') {
|
|
|
|
if (*fmt == '%') {
|
|
|
|
static char dummy[] = "";
|
|
|
|
PRINTF state = Flags;
|
|
|
|
char *pval = dummy; /* avoid const-cast */
|
|
|
|
double fval = 0.0;
|
|
|
|
int done = FALSE;
|
|
|
|
int ival = 0;
|
|
|
|
int prec = -1;
|
|
|
|
int type = 0;
|
|
|
|
int used = 0;
|
|
|
|
int width = -1;
|
|
|
|
size_t f = 0;
|
|
|
|
|
|
|
|
format[f++] = *fmt;
|
|
|
|
while (*++fmt != '\0' && len >= 0 && !done) {
|
|
|
|
format[f++] = *fmt;
|
|
|
|
|
|
|
|
if (isdigit(UChar(*fmt))) {
|
|
|
|
int num = *fmt - '0';
|
|
|
|
if (state == Flags && num != 0)
|
|
|
|
state = Width;
|
|
|
|
if (state == Width) {
|
|
|
|
if (width < 0)
|
|
|
|
width = 0;
|
|
|
|
width = (width * 10) + num;
|
|
|
|
} else if (state == Prec) {
|
|
|
|
if (prec < 0)
|
|
|
|
prec = 0;
|
|
|
|
prec = (prec * 10) + num;
|
|
|
|
}
|
|
|
|
} else if (*fmt == '*') {
|
|
|
|
VA_INTGR(int);
|
|
|
|
if (state == Flags)
|
|
|
|
state = Width;
|
|
|
|
if (state == Width) {
|
|
|
|
width = ival;
|
|
|
|
} else if (state == Prec) {
|
|
|
|
prec = ival;
|
|
|
|
}
|
2004-02-09 10:15:26 +08:00
|
|
|
sprintf(fmt_arg, "%d", ival);
|
|
|
|
fmt_len += strlen(fmt_arg);
|
|
|
|
if ((format = realloc(format, fmt_len)) == 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strcpy(&format[--f], fmt_arg);
|
2002-10-13 11:35:53 +08:00
|
|
|
f = strlen(format);
|
|
|
|
} else if (isalpha(UChar(*fmt))) {
|
|
|
|
done = TRUE;
|
|
|
|
switch (*fmt) {
|
|
|
|
case 'Z': /* FALLTHRU */
|
|
|
|
case 'h': /* FALLTHRU */
|
|
|
|
case 'l': /* FALLTHRU */
|
|
|
|
done = FALSE;
|
|
|
|
type = *fmt;
|
|
|
|
break;
|
|
|
|
case 'i': /* FALLTHRU */
|
|
|
|
case 'd': /* FALLTHRU */
|
|
|
|
case 'u': /* FALLTHRU */
|
|
|
|
case 'x': /* FALLTHRU */
|
|
|
|
case 'X': /* FALLTHRU */
|
|
|
|
if (type == 'l')
|
|
|
|
VA_INTGR(long);
|
|
|
|
else if (type == 'Z')
|
|
|
|
VA_INTGR(size_t);
|
|
|
|
else
|
|
|
|
VA_INTGR(int);
|
|
|
|
used = 'i';
|
|
|
|
break;
|
|
|
|
case 'f': /* FALLTHRU */
|
|
|
|
case 'e': /* FALLTHRU */
|
|
|
|
case 'E': /* FALLTHRU */
|
|
|
|
case 'g': /* FALLTHRU */
|
|
|
|
case 'G': /* FALLTHRU */
|
|
|
|
VA_FLOAT(double);
|
|
|
|
used = 'f';
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
VA_INTGR(int);
|
|
|
|
used = 'i';
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
VA_POINT(char *);
|
|
|
|
if (prec < 0)
|
|
|
|
prec = strlen(pval);
|
|
|
|
if (prec > (int) length) {
|
|
|
|
length = length + prec;
|
|
|
|
buffer = typeRealloc(char, length, buffer);
|
|
|
|
if (buffer == 0) {
|
|
|
|
free(format);
|
|
|
|
return -1;
|
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
2002-10-13 11:35:53 +08:00
|
|
|
used = 'p';
|
|
|
|
break;
|
|
|
|
case 'p':
|
|
|
|
VA_POINT(void *);
|
|
|
|
used = 'p';
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
VA_POINT(int *);
|
|
|
|
used = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (*fmt == '.') {
|
|
|
|
state = Prec;
|
|
|
|
} else if (*fmt == '%') {
|
|
|
|
done = TRUE;
|
|
|
|
used = 'p';
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
2002-10-13 11:35:53 +08:00
|
|
|
}
|
|
|
|
format[f] = '\0';
|
|
|
|
switch (used) {
|
|
|
|
case 'i':
|
|
|
|
sprintf(buffer, format, ival);
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
sprintf(buffer, format, fval);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
sprintf(buffer, format, pval);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
len += (int) strlen(buffer);
|
|
|
|
} else {
|
|
|
|
fmt++;
|
|
|
|
len++;
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
2002-10-13 11:35:53 +08:00
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
|
2002-10-13 11:35:53 +08:00
|
|
|
free(buffer);
|
|
|
|
free(format);
|
|
|
|
return len;
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wrapper for vsprintf that allocates a buffer big enough to hold the result.
|
|
|
|
*/
|
2002-10-13 11:35:53 +08:00
|
|
|
NCURSES_EXPORT(char *)
|
2004-02-09 10:15:26 +08:00
|
|
|
_nc_printf_string(const char *fmt, va_list ap)
|
1998-03-01 12:21:12 +08:00
|
|
|
{
|
2004-02-09 10:15:26 +08:00
|
|
|
static char *buf;
|
|
|
|
static size_t used;
|
|
|
|
char *result = 0;
|
|
|
|
|
|
|
|
if (fmt != 0) {
|
1998-03-01 12:21:12 +08:00
|
|
|
#if USE_SAFE_SPRINTF
|
2004-02-09 10:15:26 +08:00
|
|
|
int len = _nc_printf_length(fmt, ap);
|
2002-10-13 11:35:53 +08:00
|
|
|
|
2004-02-09 10:15:26 +08:00
|
|
|
if ((int) used < len + 1) {
|
|
|
|
used = 2 * (len + 1);
|
|
|
|
buf = typeRealloc(char, used, buf);
|
|
|
|
}
|
|
|
|
if (buf != 0) {
|
|
|
|
*buf = '\0';
|
|
|
|
if (len >= 0) {
|
|
|
|
vsprintf(buf, fmt, ap);
|
|
|
|
}
|
|
|
|
result = buf;
|
|
|
|
}
|
1998-03-01 12:21:12 +08:00
|
|
|
#else
|
2004-02-09 10:15:26 +08:00
|
|
|
static int rows, cols;
|
|
|
|
|
|
|
|
if (screen_lines > rows || screen_columns > cols) {
|
|
|
|
if (screen_lines > rows)
|
|
|
|
rows = screen_lines;
|
|
|
|
if (screen_columns > cols)
|
|
|
|
cols = screen_columns;
|
|
|
|
used = (rows * (cols + 1)) + 1;
|
|
|
|
buf = typeRealloc(char, used, buf);
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|
|
|
|
|
2004-02-09 10:15:26 +08:00
|
|
|
if (buf != 0) {
|
1998-03-01 12:21:12 +08:00
|
|
|
# if HAVE_VSNPRINTF
|
2004-02-09 10:15:26 +08:00
|
|
|
vsnprintf(buf, used, fmt, ap); /* GNU extension */
|
1998-03-01 12:21:12 +08:00
|
|
|
# else
|
2004-02-09 10:15:26 +08:00
|
|
|
vsprintf(buf, fmt, ap); /* ANSI */
|
1998-03-01 12:21:12 +08:00
|
|
|
# endif
|
2004-02-09 10:15:26 +08:00
|
|
|
result = buf;
|
|
|
|
}
|
1999-10-24 12:32:42 +08:00
|
|
|
#endif
|
2004-02-09 10:15:26 +08:00
|
|
|
} else if (buf != 0) { /* see _nc_freeall() */
|
|
|
|
free(buf);
|
|
|
|
buf = 0;
|
|
|
|
used = 0;
|
|
|
|
}
|
|
|
|
return result;
|
1998-03-01 12:21:12 +08:00
|
|
|
}
|