mprintf: fix integer handling in float precision

In the double output function when an extremely large width and
precision is set that reaches the libcurl maximum (325), the handling of
the precision part would do wrong which could lead to bad output.

Also: work-around for single-byte buffer snprintf overflow with mingw.

Extend test 557 to verify.

Coverity CID 1638751.

Closes #15988
This commit is contained in:
Daniel Stenberg 2025-01-13 13:24:31 +01:00
parent 97d278fd76
commit 7e32f65687
No known key found for this signature in database
GPG Key ID: 5CC908FDB71E12C2
2 changed files with 59 additions and 10 deletions

View File

@ -678,12 +678,12 @@ static int formatf(
struct outsegment output[MAX_SEGMENTS];
struct va_input input[MAX_PARAMETERS];
char work[BUFFSIZE];
char work[BUFFSIZE + 2];
/* 'workend' points to the final buffer byte position, but with an extra
byte as margin to avoid the (FALSE?) warning Coverity gives us
otherwise */
char *workend = &work[sizeof(work) - 2];
char *workend = &work[BUFFSIZE - 2];
/* Parse the format string */
if(parsefmt(format, output, input, &ocount, &icount, ap_save))
@ -966,8 +966,8 @@ number:
if(width >= 0) {
size_t dlen;
if(width >= (int)sizeof(work))
width = sizeof(work)-1;
if(width >= BUFFSIZE)
width = BUFFSIZE - 1;
/* RECURSIVE USAGE */
dlen = (size_t)curl_msnprintf(fptr, left, "%d", width);
fptr += dlen;
@ -976,17 +976,19 @@ number:
if(prec >= 0) {
/* for each digit in the integer part, we can have one less
precision */
size_t maxprec = sizeof(work) - 2;
int maxprec = BUFFSIZE - 1;
double val = iptr->val.dnum;
if(prec > maxprec)
prec = maxprec - 1;
if(width > 0 && prec <= width)
maxprec -= (size_t)width;
maxprec -= width;
while(val >= 10.0) {
val /= 10;
maxprec--;
}
if(prec > (int)maxprec)
prec = (int)maxprec-1;
if(prec > maxprec)
prec = maxprec - 1;
if(prec < 0)
prec = 0;
/* RECURSIVE USAGE */
@ -1012,14 +1014,19 @@ number:
/* NOTE NOTE NOTE!! Not all sprintf implementations return number of
output characters */
#ifdef HAVE_SNPRINTF
(snprintf)(work, sizeof(work), formatbuf, iptr->val.dnum); /* NOLINT */
(snprintf)(work, BUFFSIZE, formatbuf, iptr->val.dnum); /* NOLINT */
#else
(sprintf)(work, formatbuf, iptr->val.dnum);
#endif
#ifdef __clang__
#pragma clang diagnostic pop
#endif
DEBUGASSERT(strlen(work) <= sizeof(work));
DEBUGASSERT(strlen(work) < BUFFSIZE);
#ifdef __MINGW32__
/* Work-around for a nasty bug seen with old-mingw and gcc 7.3.0 when it
writes one byte more than permitted. */
work[BUFFSIZE - 1] = 0;
#endif
for(fptr = work; *fptr; fptr++)
OUTCHAR(*fptr);
break;

View File

@ -1207,6 +1207,46 @@ static int test_pos_arguments(void)
return errors;
}
static int test_width_precision(void)
{
/* 325 is max precision (and width) for a double */
char larger[1024];
#define SPACE60 " "
#define SPACE300 SPACE60 SPACE60 SPACE60 SPACE60 SPACE60
#define OK325 SPACE300 " 0"
int rc;
int errors = 0;
rc = curl_msnprintf(larger, sizeof(larger), "%325.325f", 0.1);
if(rc != 325)
errors++;
errors += string_check(larger, OK325);
rc = curl_msnprintf(larger, sizeof(larger), "%326.326f", 0.1);
if(rc != 325)
errors++;
errors += string_check(larger, OK325);
rc = curl_msnprintf(larger, sizeof(larger), "%1000.1000f", 0.1);
if(rc != 325)
errors++;
errors += string_check(larger, OK325);
rc = curl_msnprintf(larger, sizeof(larger), "%324.324f", 0.1);
if(rc != 324)
errors++;
rc = curl_msnprintf(larger, sizeof(larger), "%324.0f", 0.1);
if(rc != 324)
errors++;
rc = curl_msnprintf(larger, sizeof(larger), "%0.324f", 0.1);
if(rc != 325)
errors++;
return errors;
}
static int test_weird_arguments(void)
{
int errors = 0;
@ -1320,6 +1360,8 @@ static int test_weird_arguments(void)
errors += string_check(buf, "");
errors += test_width_precision();
if(errors)
printf("Some curl_mprintf() weird arguments tests failed!\n");