ncursesw-morphos/ncurses/base/safe_sprintf.c

280 lines
7.6 KiB
C
Raw Normal View History

1998-03-01 12:21:12 +08:00
/****************************************************************************
* Copyright (c) 1998-2009,2010 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>
MODULE_ID("$Id: safe_sprintf.c,v 1.24 2010/06/05 22:22:27 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);
ncurses 5.7 - patch 20090328 + extend ansi.sys pfkey capability from kf1-kf10 to kf1-kf48, moving function key definitions from emx-base for consistency -TD + correct missing final 'p' in pfkey capability of ansi.sys-old (report by Kalle Olavi Niemitalo). + improve test/ncurses.c 'F' test, show combining characters in color. + quiet a false report by cppcheck in c++/cursesw.cc by eliminating a temporary variable. + use _nc_doalloc() rather than realloc() in a few places in ncurses library to avoid leak in out-of-memory condition (reports by William Egert and Martin Ettl based on cppcheck tool). + add --with-ncurses-wrap-prefix option to test/configure (discussion with Charles Wilson). + use ncurses*-config scripts if available for test/configure. + update test/aclocal.m4 and test/configure > patches by Charles Wilson: + modify CF_WITH_LIBTOOL configure check to allow unreleased libtool version numbers (e.g. which include alphabetic chars, as well as digits, after the final '.'). + improve use of -no-undefined option for libtool by setting an intermediate variable LT_UNDEF in the configure script, and then using that in the libtool link-commands. + fix an missing use of NCURSES_PUBLIC_VAR() in tinfo/MKcodes.awk from 2009031 changes. + improve mk-1st.awk script by writing separate cases for the LIBTOOL_LINK command, depending on which library (ncurses, ticlib, termlib) is to be linked. + modify configure.in to allow broken-linker configurations, not just enable-reentrant, to set public wrap prefix.
2009-03-29 08:06:57 +08:00
if ((format = _nc_doalloc(format, fmt_len)) == 0) {
2004-02-09 10:15:26 +08:00
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
2007-04-22 08:57:08 +08:00
#define my_buffer _nc_globals.safeprint_buf
#define my_length _nc_globals.safeprint_used
1998-03-01 12:21:12 +08:00
/*
* Wrapper for vsprintf that allocates a buffer big enough to hold the result.
*/
2002-10-13 11:35:53 +08:00
NCURSES_EXPORT(char *)
NCURSES_SP_NAME(_nc_printf_string) (NCURSES_SP_DCLx
const char *fmt,
va_list ap)
1998-03-01 12:21:12 +08:00
{
2004-02-09 10:15:26 +08:00
char *result = 0;
if (fmt != 0) {
1998-03-01 12:21:12 +08:00
#if USE_SAFE_SPRINTF
va_list ap2;
int len;
begin_va_copy(ap2, ap);
len = _nc_printf_length(fmt, ap2);
end_va_copy(ap2);
2002-10-13 11:35:53 +08:00
2007-04-22 08:57:08 +08:00
if ((int) my_length < len + 1) {
my_length = 2 * (len + 1);
my_buffer = typeRealloc(char, my_length, my_buffer);
2004-02-09 10:15:26 +08:00
}
2007-04-22 08:57:08 +08:00
if (my_buffer != 0) {
*my_buffer = '\0';
2004-02-09 10:15:26 +08:00
if (len >= 0) {
2007-04-22 08:57:08 +08:00
vsprintf(my_buffer, fmt, ap);
2004-02-09 10:15:26 +08:00
}
2007-04-22 08:57:08 +08:00
result = my_buffer;
2004-02-09 10:15:26 +08:00
}
1998-03-01 12:21:12 +08:00
#else
2007-04-22 08:57:08 +08:00
#define MyCols _nc_globals.safeprint_cols
#define MyRows _nc_globals.safeprint_rows
2004-02-09 10:15:26 +08:00
if (screen_lines(SP_PARM) > MyRows || screen_columns(SP_PARM) > MyCols) {
if (screen_lines(SP_PARM) > MyRows)
MyRows = screen_lines(SP_PARM);
if (screen_columns(SP_PARM) > MyCols)
MyCols = screen_columns(SP_PARM);
my_length = (size_t) (MyRows * (MyCols + 1)) + 1;
2007-04-22 08:57:08 +08:00
my_buffer = typeRealloc(char, my_length, my_buffer);
1998-03-01 12:21:12 +08:00
}
2007-04-22 08:57:08 +08:00
if (my_buffer != 0) {
1998-03-01 12:21:12 +08:00
# if HAVE_VSNPRINTF
2007-04-22 08:57:08 +08:00
vsnprintf(my_buffer, my_length, fmt, ap); /* GNU extension */
1998-03-01 12:21:12 +08:00
# else
2007-04-22 08:57:08 +08:00
vsprintf(my_buffer, fmt, ap); /* ANSI */
1998-03-01 12:21:12 +08:00
# endif
2007-04-22 08:57:08 +08:00
result = my_buffer;
2004-02-09 10:15:26 +08:00
}
1999-10-24 12:32:42 +08:00
#endif
2007-04-22 08:57:08 +08:00
} else if (my_buffer != 0) { /* see _nc_freeall() */
free(my_buffer);
my_buffer = 0;
my_length = 0;
2004-02-09 10:15:26 +08:00
}
return result;
1998-03-01 12:21:12 +08:00
}
#if NCURSES_SP_FUNCS
NCURSES_EXPORT(char *)
_nc_printf_string(const char *fmt, va_list ap)
{
return NCURSES_SP_NAME(_nc_printf_string) (CURRENT_SCREEN, fmt, ap);
}
#endif