Coverage Report

Created: 2026-06-13 07:01

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/ext/standard/formatted_print.c
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Copyright © The PHP Group and Contributors.                          |
4
   +----------------------------------------------------------------------+
5
   | This source file is subject to the Modified BSD License that is      |
6
   | bundled with this package in the file LICENSE, and is available      |
7
   | through the World Wide Web at <https://www.php.net/license/>.        |
8
   |                                                                      |
9
   | SPDX-License-Identifier: BSD-3-Clause                                |
10
   +----------------------------------------------------------------------+
11
   | Author: Stig Sæther Bakken <ssb@php.net>                             |
12
   +----------------------------------------------------------------------+
13
 */
14
15
#include <math.h>       /* modf() */
16
#include "php.h"
17
#include "zend_execute.h"
18
19
#include <locale.h>
20
#ifdef ZTS
21
#include "ext/standard/php_string.h" /* for localeconv_r() */
22
#define LCONV_DECIMAL_POINT (*lconv.decimal_point)
23
#else
24
787
#define LCONV_DECIMAL_POINT (*lconv->decimal_point)
25
#endif
26
27
951
#define ALIGN_LEFT 0
28
2.07k
#define ALIGN_RIGHT 1
29
95
#define ADJ_WIDTH 1
30
1.58k
#define ADJ_PRECISION 2
31
46
#define NUM_BUF_SIZE 500
32
38
#define FLOAT_PRECISION 6
33
768
#define MAX_FLOAT_PRECISION 53
34
35
#if 0
36
/* trick to control varargs functions through cpp */
37
# define PRINTF_DEBUG(arg) php_printf arg
38
#else
39
# define PRINTF_DEBUG(arg)
40
#endif
41
42
static const char hexchars[] = "0123456789abcdef";
43
static const char HEXCHARS[] = "0123456789ABCDEF";
44
45
/* php_spintf_appendchar() {{{ */
46
inline static void
47
php_sprintf_appendchar(zend_string **buffer, size_t *pos, char add)
48
259
{
49
259
  if ((*pos + 1) >= ZSTR_LEN(*buffer)) {
50
0
    PRINTF_DEBUG(("%s(): ereallocing buffer to %zu bytes\n", get_active_function_name(), ZSTR_LEN(*buffer)));
51
0
    *buffer = zend_string_extend(*buffer, ZSTR_LEN(*buffer) << 1, 0);
52
0
  }
53
259
  PRINTF_DEBUG(("sprintf: appending '%c', pos=%zu\n", add, *pos));
54
259
  ZSTR_VAL(*buffer)[(*pos)++] = add;
55
259
}
56
/* }}} */
57
58
/* php_spintf_appendchar() {{{ */
59
inline static void
60
php_sprintf_appendchars(zend_string **buffer, size_t *pos, char *add, size_t len)
61
1.43k
{
62
1.43k
  if ((*pos + len) >= ZSTR_LEN(*buffer)) {
63
78
    size_t nlen = ZSTR_LEN(*buffer);
64
65
78
    PRINTF_DEBUG(("%s(): ereallocing buffer to %zu bytes\n", get_active_function_name(), ZSTR_LEN(*buffer)));
66
105
    do {
67
105
      nlen = nlen << 1;
68
105
    } while ((*pos + len) >= nlen);
69
78
    *buffer = zend_string_extend(*buffer, nlen, 0);
70
78
  }
71
1.43k
  PRINTF_DEBUG(("sprintf: appending \"%s\", pos=%zu\n", add, *pos));
72
1.43k
  memcpy(ZSTR_VAL(*buffer) + (*pos), add, len);
73
1.43k
  *pos += len;
74
1.43k
}
75
/* }}} */
76
77
/* php_spintf_appendstring() {{{ */
78
inline static void
79
php_sprintf_appendstring(zend_string **buffer, size_t *pos, char *add,
80
               size_t min_width, size_t max_width, char padding,
81
               size_t alignment, size_t len, bool neg, int expprec, int always_sign)
82
921
{
83
921
  size_t npad;
84
921
  size_t req_size;
85
921
  size_t copy_len;
86
921
  size_t m_width;
87
88
921
  copy_len = (expprec ? MIN(max_width, len) : len);
89
921
  npad = (min_width < copy_len) ? 0 : min_width - copy_len;
90
91
921
  PRINTF_DEBUG(("sprintf: appendstring(%p, %zu, %zu, \"%s\", %zu, '%c', %zu)\n",
92
921
          *buffer, *pos, ZSTR_LEN(*buffer), add, min_width, padding, alignment));
93
921
  m_width = MAX(min_width, copy_len);
94
95
921
  if(m_width > INT_MAX - *pos - 1) {
96
0
    zend_error_noreturn(E_ERROR, "Field width %zd is too long", m_width);
97
0
  }
98
99
921
  req_size = *pos + m_width + 1;
100
101
921
  if (req_size > ZSTR_LEN(*buffer)) {
102
12
    size_t size = ZSTR_LEN(*buffer);
103
96
    while (req_size > size) {
104
84
      if (size > SIZE_MAX/2) {
105
0
        zend_error_noreturn(E_ERROR, "Field width %zd is too long", req_size);
106
0
      }
107
84
      size <<= 1;
108
84
    }
109
12
    PRINTF_DEBUG(("sprintf ereallocing buffer to %zu bytes\n", size));
110
12
    *buffer = zend_string_extend(*buffer, size, 0);
111
12
  }
112
921
  if (alignment == ALIGN_RIGHT) {
113
915
    if ((neg || always_sign) && padding=='0') {
114
0
      ZSTR_VAL(*buffer)[(*pos)++] = (neg) ? '-' : '+';
115
0
      add++;
116
0
      len--;
117
0
      copy_len--;
118
0
    }
119
50.0M
    while (npad-- > 0) {
120
50.0M
      ZSTR_VAL(*buffer)[(*pos)++] = padding;
121
50.0M
    }
122
915
  }
123
921
  PRINTF_DEBUG(("sprintf: appending \"%s\"\n", add));
124
921
  memcpy(&ZSTR_VAL(*buffer)[*pos], add, copy_len + 1);
125
921
  *pos += copy_len;
126
921
  if (alignment == ALIGN_LEFT) {
127
6
    while (npad--) {
128
0
      ZSTR_VAL(*buffer)[(*pos)++] = padding;
129
0
    }
130
6
  }
131
921
}
132
/* }}} */
133
134
/* php_spintf_appendint() {{{ */
135
inline static void
136
php_sprintf_appendint(zend_string **buffer, size_t *pos, zend_long number,
137
            size_t width, char padding, size_t alignment,
138
            int always_sign)
139
12
{
140
12
  char numbuf[NUM_BUF_SIZE];
141
12
  zend_ulong magn, nmagn;
142
12
  unsigned int i = NUM_BUF_SIZE - 1, neg = 0;
143
144
12
  PRINTF_DEBUG(("sprintf: appendint(%p, %zu, %zu, " ZEND_LONG_FMT ", %zu, '%c', %zu)\n",
145
12
          *buffer, *pos, ZSTR_LEN(*buffer), number, width, padding, alignment));
146
12
  if (number < 0) {
147
0
    neg = 1;
148
0
    magn = ((zend_ulong) -(number + 1)) + 1;
149
12
  } else {
150
12
    magn = (zend_ulong) number;
151
12
  }
152
153
  /* Can't right-pad 0's on integers */
154
12
  if(alignment==0 && padding=='0') padding=' ';
155
156
12
  numbuf[i] = '\0';
157
158
33
  do {
159
33
    nmagn = magn / 10;
160
161
33
    numbuf[--i] = (unsigned char)(magn - (nmagn * 10)) + '0';
162
33
    magn = nmagn;
163
33
  }
164
33
  while (magn > 0 && i > 1);
165
12
  if (neg) {
166
0
    numbuf[--i] = '-';
167
12
  } else if (always_sign) {
168
0
    numbuf[--i] = '+';
169
0
  }
170
12
  PRINTF_DEBUG(("sprintf: appending " ZEND_LONG_FMT " as \"%s\", i=%u\n",
171
12
          number, &numbuf[i], i));
172
12
  php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
173
12
               padding, alignment, (NUM_BUF_SIZE - 1) - i,
174
12
               neg, 0, always_sign);
175
12
}
176
/* }}} */
177
178
/* php_spintf_appenduint() {{{ */
179
inline static void
180
php_sprintf_appenduint(zend_string **buffer, size_t *pos,
181
             zend_ulong number,
182
             size_t width, char padding, size_t alignment)
183
0
{
184
0
  char numbuf[NUM_BUF_SIZE];
185
0
  zend_ulong magn, nmagn;
186
0
  unsigned int i = NUM_BUF_SIZE - 1;
187
188
0
  PRINTF_DEBUG(("sprintf: appenduint(%p, %zu, %zu, " ZEND_LONG_FMT ", %zu, '%c', %zu)\n",
189
0
          *buffer, *pos, ZSTR_LEN(*buffer), number, width, padding, alignment));
190
0
  magn = (zend_ulong) number;
191
192
  /* Can't right-pad 0's on integers */
193
0
  if (alignment == 0 && padding == '0') padding = ' ';
194
195
0
  numbuf[i] = '\0';
196
197
0
  do {
198
0
    nmagn = magn / 10;
199
200
0
    numbuf[--i] = (unsigned char)(magn - (nmagn * 10)) + '0';
201
0
    magn = nmagn;
202
0
  } while (magn > 0 && i > 0);
203
204
0
  PRINTF_DEBUG(("sprintf: appending " ZEND_LONG_FMT " as \"%s\", i=%d\n", number, &numbuf[i], i));
205
0
  php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
206
0
               padding, alignment, (NUM_BUF_SIZE - 1) - i, /* neg */ false, 0, 0);
207
0
}
208
/* }}} */
209
210
/* php_spintf_appenddouble() {{{ */
211
inline static void
212
php_sprintf_appenddouble(zend_string **buffer, size_t *pos,
213
             double number,
214
             size_t width, char padding,
215
             size_t alignment, int precision,
216
             int adjust, char fmt,
217
             int always_sign
218
            )
219
806
{
220
806
  char num_buf[NUM_BUF_SIZE];
221
806
  char *s = NULL;
222
806
  size_t s_len = 0;
223
806
  bool is_negative = false;
224
#ifdef ZTS
225
  struct lconv lconv;
226
#else
227
806
  struct lconv *lconv;
228
806
#endif
229
230
806
  PRINTF_DEBUG(("sprintf: appenddouble(%p, %zu, %zu, %f, %zu, '%c', %zu, %c)\n",
231
806
          *buffer, *pos, ZSTR_LEN(*buffer), number, width, padding, alignment, fmt));
232
806
  if ((adjust & ADJ_PRECISION) == 0) {
233
38
    precision = FLOAT_PRECISION;
234
768
  } else if (precision > MAX_FLOAT_PRECISION) {
235
0
    php_error_docref(NULL, E_NOTICE, "Requested precision of %d digits was truncated to PHP maximum of %d digits", precision, MAX_FLOAT_PRECISION);
236
0
    precision = MAX_FLOAT_PRECISION;
237
0
  }
238
239
806
  if (zend_isnan(number)) {
240
0
    is_negative = (number<0);
241
0
    php_sprintf_appendstring(buffer, pos, "NaN", 3, 0, padding,
242
0
                 alignment, 3, is_negative, 0, always_sign);
243
0
    return;
244
0
  }
245
246
806
  if (zend_isinf(number)) {
247
0
    is_negative = (number<0);   
248
0
    char *str = is_negative ? "-INF" : "INF";
249
0
    php_sprintf_appendstring(buffer, pos, str, strlen(str), 0, padding,
250
0
                alignment, strlen(str), is_negative, 0, always_sign);
251
0
    return;
252
0
  }
253
254
806
  switch (fmt) {
255
16
    case 'e':
256
19
    case 'E':
257
799
    case 'f':
258
799
    case 'F':
259
#ifdef ZTS
260
      localeconv_r(&lconv);
261
#else
262
799
      lconv = localeconv();
263
799
#endif
264
799
      s = php_conv_fp((fmt == 'f')?'F':fmt, number, 0, precision,
265
799
            (fmt == 'f')?LCONV_DECIMAL_POINT:'.',
266
799
            &is_negative, &num_buf[1], &s_len);
267
799
      if (is_negative) {
268
153
        num_buf[0] = '-';
269
153
        s = num_buf;
270
153
        s_len++;
271
646
      } else if (always_sign) {
272
0
        num_buf[0] = '+';
273
0
        s = num_buf;
274
0
        s_len++;
275
0
      }
276
799
      break;
277
278
7
    case 'g':
279
7
    case 'G':
280
7
    case 'h':
281
7
    case 'H':
282
7
    {
283
7
      if (precision == 0)
284
0
        precision = 1;
285
286
7
      char decimal_point = '.';
287
7
      if (fmt == 'g' || fmt == 'G') {
288
#ifdef ZTS
289
        localeconv_r(&lconv);
290
#else
291
7
        lconv = localeconv();
292
7
#endif
293
7
        decimal_point = LCONV_DECIMAL_POINT;
294
7
      }
295
296
7
      char exp_char = fmt == 'G' || fmt == 'H' ? 'E' : 'e';
297
      /* We use &num_buf[ 1 ], so that we have room for the sign. */
298
7
      s = zend_gcvt(number, precision, decimal_point, exp_char, &num_buf[1]);
299
7
      is_negative = false;
300
7
      if (*s == '-') {
301
0
        is_negative = true;
302
0
        s = &num_buf[1];
303
7
      } else if (always_sign) {
304
0
        num_buf[0] = '+';
305
0
        s = num_buf;
306
0
      }
307
308
7
      s_len = strlen(s);
309
7
      break;
310
7
    }
311
806
  }
312
313
806
  php_sprintf_appendstring(buffer, pos, s, width, 0, padding,
314
806
               alignment, s_len, is_negative, 0, always_sign);
315
806
}
316
/* }}} */
317
318
/* php_spintf_appendd2n() {{{ */
319
inline static void
320
php_sprintf_append2n(zend_string **buffer, size_t *pos, zend_long number,
321
           size_t width, char padding, size_t alignment, int n,
322
           const char *chartable, int expprec)
323
11
{
324
11
  char numbuf[NUM_BUF_SIZE];
325
11
  zend_ulong num;
326
11
  zend_ulong  i = NUM_BUF_SIZE - 1;
327
11
  int andbits = (1 << n) - 1;
328
329
11
  PRINTF_DEBUG(("sprintf: append2n(%p, %zu, %zu, " ZEND_LONG_FMT ", %zu, '%c', %zu, %d, %p)\n",
330
11
          *buffer, *pos, ZSTR_LEN(*buffer), number, width, padding, alignment, n,
331
11
          chartable));
332
11
  PRINTF_DEBUG(("sprintf: append2n 2^%d andbits=%x\n", n, andbits));
333
334
11
  num = (zend_ulong) number;
335
11
  numbuf[i] = '\0';
336
337
33
  do {
338
33
    numbuf[--i] = chartable[(num & andbits)];
339
33
    num >>= n;
340
33
  }
341
33
  while (num > 0);
342
343
11
  php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
344
11
               padding, alignment, (NUM_BUF_SIZE - 1) - i,
345
11
               /* neg */ false, expprec, 0);
346
11
}
347
/* }}} */
348
349
/* php_spintf_getnumber() {{{ */
350
inline static int
351
php_sprintf_getnumber(char **buffer, size_t *len)
352
863
{
353
863
  char *endptr;
354
863
  zend_long num = ZEND_STRTOL(*buffer, &endptr, 10);
355
863
  size_t i;
356
357
863
  if (endptr != NULL) {
358
863
    i = (endptr - *buffer);
359
863
    *len -= i;
360
863
    *buffer = endptr;
361
863
  }
362
863
  PRINTF_DEBUG(("sprintf_getnumber: number was %zu bytes long\n", i));
363
364
863
  if (num >= INT_MAX || num < 0) {
365
3
    return -1;
366
860
  } else {
367
860
    return (int) num;
368
860
  }
369
863
}
370
/* }}} */
371
372
2.31k
#define ARG_NUM_NEXT -1
373
1.00k
#define ARG_NUM_INVALID -2
374
375
1.00k
int php_sprintf_get_argnum(char **format, size_t *format_len) {
376
1.00k
  char *temppos = *format;
377
1.00k
  while (isdigit((unsigned char)*temppos)) temppos++;
378
1.00k
  if (*temppos != '$') {
379
1.00k
    return ARG_NUM_NEXT;
380
1.00k
  }
381
382
3
  int argnum = php_sprintf_getnumber(format, format_len);
383
3
  if (argnum <= 0) {
384
3
    zend_value_error("Argument number specifier must be greater than zero and less than %d", INT_MAX);
385
3
    return ARG_NUM_INVALID;
386
3
  }
387
388
0
  (*format)++;  /* skip the '$' */
389
0
  (*format_len)--;
390
0
  return argnum - 1;
391
3
}
392
393
/* php_formatted_print() {{{
394
 * New sprintf implementation for PHP.
395
 *
396
 * Modifiers:
397
 *
398
 *  " "   pad integers with spaces
399
 *  "-"   left adjusted field
400
 *   n    field size
401
 *  "."n  precision (floats only)
402
 *  "+"   Always place a sign (+ or -) in front of a number
403
 *
404
 * Type specifiers:
405
 *
406
 *  "%"   literal "%", modifiers are ignored.
407
 *  "b"   integer argument is printed as binary
408
 *  "c"   integer argument is printed as a single character
409
 *  "d"   argument is an integer
410
 *  "f"   the argument is a float
411
 *  "o"   integer argument is printed as octal
412
 *  "s"   argument is a string
413
 *  "x"   integer argument is printed as lowercase hexadecimal
414
 *  "X"   integer argument is printed as uppercase hexadecimal
415
 *
416
 * nb_additional_parameters is used for throwing errors:
417
 *  - -1: ValueError is thrown (for vsprintf where args originates from an array)
418
 *  - 0 or more: ArgumentCountError is thrown
419
 */
420
static zend_string *
421
php_formatted_print(char *format, size_t format_len, zval *args, int argc, int nb_additional_parameters)
422
470
{
423
470
  size_t size = 240, outpos = 0;
424
470
  int alignment, currarg, adjusting, argnum, width, precision;
425
470
  char *temppos, padding;
426
470
  zend_string *result;
427
470
  int always_sign;
428
470
  int max_missing_argnum = -1;
429
430
  /* For debugging */
431
470
  const char *format_orig = format;
432
470
  ZEND_IGNORE_VALUE(format_orig);
433
434
470
  result = zend_string_alloc(size, 0);
435
436
470
  currarg = 0;
437
470
  argnum = 0;
438
439
1.81k
  while (format_len) {
440
1.78k
    int expprec;
441
1.78k
    zval *tmp;
442
443
1.78k
    temppos = memchr(format, '%', format_len);
444
1.78k
    if (!temppos) {
445
384
      php_sprintf_appendchars(&result, &outpos, format, format_len);
446
384
      break;
447
1.40k
    } else if (temppos != format) {
448
1.04k
      php_sprintf_appendchars(&result, &outpos, format, temppos - format);
449
1.04k
      format_len -= temppos - format;
450
1.04k
      format = temppos;
451
1.04k
    }
452
1.40k
    format++;     /* skip the '%' */
453
1.40k
    format_len--;
454
455
1.40k
    if (*format == '%') {
456
248
      php_sprintf_appendchar(&result, &outpos, '%');
457
248
      format++;
458
248
      format_len--;
459
1.15k
    } else {
460
      /* starting a new format specifier, reset variables */
461
1.15k
      alignment = ALIGN_RIGHT;
462
1.15k
      adjusting = 0;
463
1.15k
      padding = ' ';
464
1.15k
      always_sign = 0;
465
1.15k
      expprec = 0;
466
467
1.15k
      PRINTF_DEBUG(("sprintf: first looking at '%c', inpos=%zu\n",
468
1.15k
              *format, format - format_orig));
469
1.15k
      if (isalpha((unsigned char)*format)) {
470
160
        width = precision = 0;
471
160
        argnum = ARG_NUM_NEXT;
472
993
      } else {
473
        /* first look for argnum */
474
993
        argnum = php_sprintf_get_argnum(&format, &format_len);
475
993
        if (argnum == ARG_NUM_INVALID) {
476
3
          goto fail;
477
3
        }
478
479
        /* after argnum comes modifiers */
480
990
        PRINTF_DEBUG(("sprintf: looking for modifiers\n"
481
990
                "sprintf: now looking at '%c', inpos=%zu\n",
482
990
                *format, format - format_orig));
483
1.05k
        for (;; format++, format_len--) {
484
1.05k
          if (*format == ' ' || *format == '0') {
485
18
            padding = *format;
486
1.03k
          } else if (*format == '-') {
487
30
            alignment = ALIGN_LEFT;
488
            /* space padding, the default */
489
1.00k
          } else if (*format == '+') {
490
8
            always_sign = 1;
491
994
          } else if (*format == '\'') {
492
5
            if (format_len > 1) {
493
4
              format++;
494
4
              format_len--;
495
4
              padding = *format;
496
4
            } else {
497
1
              zend_value_error("Missing padding character");
498
1
              goto fail;
499
1
            }
500
989
          } else {
501
989
            PRINTF_DEBUG(("sprintf: end of modifiers\n"));
502
989
            break;
503
989
          }
504
1.05k
        }
505
989
        PRINTF_DEBUG(("sprintf: padding='%c'\n", padding));
506
989
        PRINTF_DEBUG(("sprintf: alignment=%s\n",
507
989
                (alignment == ALIGN_LEFT) ? "left" : "right"));
508
509
510
        /* after modifiers comes width */
511
989
        if (*format == '*') {
512
12
          format++;
513
12
          format_len--;
514
515
12
          int width_argnum = php_sprintf_get_argnum(&format, &format_len);
516
12
          if (width_argnum == ARG_NUM_INVALID) {
517
0
            goto fail;
518
0
          }
519
12
          if (width_argnum == ARG_NUM_NEXT) {
520
12
            width_argnum = currarg++;
521
12
          }
522
12
          if (width_argnum >= argc) {
523
3
            max_missing_argnum = MAX(max_missing_argnum, width_argnum);
524
3
            continue;
525
3
          }
526
9
          tmp = &args[width_argnum];
527
9
          ZVAL_DEREF(tmp);
528
9
          if (Z_TYPE_P(tmp) != IS_LONG) {
529
0
            zend_value_error("Width must be an integer");
530
0
            goto fail;
531
0
          }
532
9
          if (Z_LVAL_P(tmp) < 0 || Z_LVAL_P(tmp) > INT_MAX) {
533
0
            zend_value_error("Width must be between 0 and %d", INT_MAX);
534
0
            goto fail;
535
0
          }
536
9
          width = Z_LVAL_P(tmp);
537
9
          adjusting |= ADJ_WIDTH;
538
977
        } else if (isdigit((unsigned char)*format)) {
539
89
          PRINTF_DEBUG(("sprintf: getting width\n"));
540
89
          if ((width = php_sprintf_getnumber(&format, &format_len)) < 0) {
541
3
            zend_value_error("Width must be between 0 and %d", INT_MAX);
542
3
            goto fail;
543
3
          }
544
86
          adjusting |= ADJ_WIDTH;
545
888
        } else {
546
888
          width = 0;
547
888
        }
548
983
        PRINTF_DEBUG(("sprintf: width=%d\n", width));
549
550
        /* after width and argnum comes precision */
551
983
        if (*format == '.') {
552
783
          format++;
553
783
          format_len--;
554
783
          PRINTF_DEBUG(("sprintf: getting precision\n"));
555
783
          if (*format == '*') {
556
0
            format++;
557
0
            format_len--;
558
559
0
            int prec_argnum = php_sprintf_get_argnum(&format, &format_len);
560
0
            if (prec_argnum == ARG_NUM_INVALID) {
561
0
              goto fail;
562
0
            }
563
0
            if (prec_argnum == ARG_NUM_NEXT) {
564
0
              prec_argnum = currarg++;
565
0
            }
566
0
            if (prec_argnum >= argc) {
567
0
              max_missing_argnum = MAX(max_missing_argnum, prec_argnum);
568
0
              continue;
569
0
            }
570
0
            tmp = &args[prec_argnum];
571
0
            ZVAL_DEREF(tmp);
572
0
            if (Z_TYPE_P(tmp) != IS_LONG) {
573
0
              zend_value_error("Precision must be an integer");
574
0
              goto fail;
575
0
            }
576
0
            if (Z_LVAL_P(tmp) < -1 || Z_LVAL_P(tmp) > INT_MAX) {
577
0
              zend_value_error("Precision must be between -1 and %d", INT_MAX);
578
0
              goto fail;
579
0
            }
580
0
            precision = Z_LVAL_P(tmp);
581
0
            adjusting |= ADJ_PRECISION;
582
0
            expprec = 1;
583
783
          } else if (isdigit((unsigned char)*format)) {
584
771
            if ((precision = php_sprintf_getnumber(&format, &format_len)) < 0) {
585
0
              zend_value_error("Precision must be between 0 and %d", INT_MAX);
586
0
              goto fail;
587
0
            }
588
771
            adjusting |= ADJ_PRECISION;
589
771
            expprec = 1;
590
771
          } else {
591
12
            precision = 0;
592
12
            adjusting |= ADJ_PRECISION;
593
12
          }
594
783
        } else {
595
200
          precision = 0;
596
200
        }
597
983
        PRINTF_DEBUG(("sprintf: precision=%d\n", precision));
598
983
      }
599
600
1.14k
      if (*format == 'l') {
601
0
        format++;
602
0
        format_len--;
603
0
      }
604
1.14k
      PRINTF_DEBUG(("sprintf: format character='%c'\n", *format));
605
606
1.14k
      if (argnum == ARG_NUM_NEXT) {
607
1.14k
        argnum = currarg++;
608
1.14k
      }
609
1.14k
      if (argnum >= argc) {
610
159
        max_missing_argnum = MAX(max_missing_argnum, argnum);
611
159
        continue;
612
159
      }
613
614
984
      if (expprec && precision == -1
615
0
          && *format != 'g' && *format != 'G' && *format != 'h' && *format != 'H') {
616
0
        zend_value_error("Precision -1 is only supported for %%g, %%G, %%h and %%H");
617
0
        goto fail;
618
0
      }
619
620
      /* now we expect to find a type specifier */
621
984
      tmp = &args[argnum];
622
984
      switch (*format) {
623
92
        case 's': {
624
92
          zend_string *t;
625
92
          zend_string *str = zval_get_tmp_string(tmp, &t);
626
92
          php_sprintf_appendstring(&result, &outpos,
627
92
                       ZSTR_VAL(str),
628
92
                       width, precision, padding,
629
92
                       alignment,
630
92
                       ZSTR_LEN(str),
631
92
                       /* neg */ false, expprec, 0);
632
92
          zend_tmp_string_release(t);
633
92
          break;
634
0
        }
635
636
12
        case 'd':
637
12
          php_sprintf_appendint(&result, &outpos,
638
12
                      zval_get_long(tmp),
639
12
                      width, padding, alignment,
640
12
                      always_sign);
641
12
          break;
642
643
0
        case 'u':
644
0
          php_sprintf_appenduint(&result, &outpos,
645
0
                      zval_get_long(tmp),
646
0
                      width, padding, alignment);
647
0
          break;
648
649
16
        case 'e':
650
19
        case 'E':
651
799
        case 'f':
652
799
        case 'F':
653
806
        case 'g':
654
806
        case 'G':
655
806
        case 'h':
656
806
        case 'H':
657
806
          php_sprintf_appenddouble(&result, &outpos,
658
806
                       zval_get_double(tmp),
659
806
                       width, padding, alignment,
660
806
                       precision, adjusting,
661
806
                       *format, always_sign
662
806
                      );
663
806
          break;
664
665
0
        case 'c':
666
0
          php_sprintf_appendchar(&result, &outpos,
667
0
                    (char) zval_get_long(tmp));
668
0
          break;
669
670
2
        case 'o':
671
2
          php_sprintf_append2n(&result, &outpos,
672
2
                     zval_get_long(tmp),
673
2
                     width, padding, alignment, 3,
674
2
                     hexchars, expprec);
675
2
          break;
676
677
9
        case 'x':
678
9
          php_sprintf_append2n(&result, &outpos,
679
9
                     zval_get_long(tmp),
680
9
                     width, padding, alignment, 4,
681
9
                     hexchars, expprec);
682
9
          break;
683
684
0
        case 'X':
685
0
          php_sprintf_append2n(&result, &outpos,
686
0
                     zval_get_long(tmp),
687
0
                     width, padding, alignment, 4,
688
0
                     HEXCHARS, expprec);
689
0
          break;
690
691
0
        case 'b':
692
0
          php_sprintf_append2n(&result, &outpos,
693
0
                     zval_get_long(tmp),
694
0
                     width, padding, alignment, 1,
695
0
                     hexchars, expprec);
696
0
          break;
697
698
11
        case '%':
699
11
          php_sprintf_appendchar(&result, &outpos, '%');
700
701
11
          break;
702
703
3
        case '\0':
704
3
          if (!format_len) {
705
0
            zend_value_error("Missing format specifier at end of string");
706
0
            goto fail;
707
0
          }
708
3
          ZEND_FALLTHROUGH;
709
710
52
        default:
711
52
          zend_value_error("Unknown format specifier \"%c\"", *format);
712
52
          goto fail;
713
984
      }
714
932
      format++;
715
932
      format_len--;
716
932
    }
717
1.40k
  }
718
719
411
  if (max_missing_argnum >= 0) {
720
57
    if (nb_additional_parameters == -1) {
721
0
      zend_value_error("The arguments array must contain %d items, %d given", max_missing_argnum + 1, argc);
722
57
    } else {
723
57
      zend_argument_count_error("%d arguments are required, %d given", max_missing_argnum + nb_additional_parameters + 1, argc + nb_additional_parameters);
724
57
    }
725
57
    goto fail;
726
57
  }
727
728
  /* possibly, we have to make sure we have room for the terminating null? */
729
354
  ZSTR_VAL(result)[outpos]=0;
730
354
  ZSTR_LEN(result) = outpos;
731
354
  return result;
732
733
116
fail:
734
116
  zend_string_efree(result);
735
116
  return NULL;
736
411
}
737
/* }}} */
738
739
/* php_formatted_print_get_array() {{{ */
740
static zval *php_formatted_print_get_array(zend_array *array, int *argc)
741
0
{
742
0
  zval *args, *zv;
743
0
  int n;
744
745
0
  n = zend_hash_num_elements(array);
746
0
  args = (zval *)safe_emalloc(n, sizeof(zval), 0);
747
0
  n = 0;
748
0
  ZEND_HASH_FOREACH_VAL(array, zv) {
749
0
    ZVAL_COPY_VALUE(&args[n], zv);
750
0
    n++;
751
0
  } ZEND_HASH_FOREACH_END();
752
753
0
  *argc = n;
754
0
  return args;
755
0
}
756
/* }}} */
757
758
/* {{{ Return a formatted string */
759
PHP_FUNCTION(sprintf)
760
33
{
761
33
  zend_string *result;
762
33
  char *format;
763
33
  size_t format_len;
764
33
  zval *args;
765
33
  int argc;
766
767
99
  ZEND_PARSE_PARAMETERS_START(1, -1)
768
132
    Z_PARAM_STRING(format, format_len)
769
33
    Z_PARAM_VARIADIC('*', args, argc)
770
33
  ZEND_PARSE_PARAMETERS_END();
771
772
33
  result = php_formatted_print(format, format_len, args, argc, 1);
773
33
  if (result == NULL) {
774
7
    RETURN_THROWS();
775
7
  }
776
26
  RETVAL_STR(result);
777
26
}
778
/* }}} */
779
780
/* {{{ Return a formatted string */
781
PHP_FUNCTION(vsprintf)
782
0
{
783
0
  zend_string *result;
784
0
  char *format;
785
0
  size_t format_len;
786
0
  zval *args;
787
0
  zend_array *array;
788
0
  int argc;
789
790
0
  ZEND_PARSE_PARAMETERS_START(2, 2)
791
0
    Z_PARAM_STRING(format, format_len)
792
0
    Z_PARAM_ARRAY_HT(array)
793
0
  ZEND_PARSE_PARAMETERS_END();
794
795
0
  args = php_formatted_print_get_array(array, &argc);
796
797
0
  result = php_formatted_print(format, format_len, args, argc, -1);
798
0
  efree(args);
799
0
  if (result == NULL) {
800
0
    RETURN_THROWS();
801
0
  }
802
0
  RETVAL_STR(result);
803
0
}
804
/* }}} */
805
806
/* {{{ Output a formatted string */
807
PHP_FUNCTION(printf)
808
437
{
809
437
  zend_string *result;
810
437
  size_t rlen;
811
437
  char *format;
812
437
  size_t format_len;
813
437
  zval *args;
814
437
  int argc;
815
816
1.31k
  ZEND_PARSE_PARAMETERS_START(1, -1)
817
1.74k
    Z_PARAM_STRING(format, format_len)
818
437
    Z_PARAM_VARIADIC('*', args, argc)
819
437
  ZEND_PARSE_PARAMETERS_END();
820
821
437
  result = php_formatted_print(format, format_len, args, argc, 1);
822
437
  if (result == NULL) {
823
109
    RETURN_THROWS();
824
109
  }
825
328
  rlen = PHPWRITE(ZSTR_VAL(result), ZSTR_LEN(result));
826
328
  zend_string_efree(result);
827
328
  RETURN_LONG(rlen);
828
328
}
829
/* }}} */
830
831
/* {{{ Output a formatted string */
832
PHP_FUNCTION(vprintf)
833
0
{
834
0
  zend_string *result;
835
0
  size_t rlen;
836
0
  char *format;
837
0
  size_t format_len;
838
0
  zval *args;
839
0
  zend_array *array;
840
0
  int argc;
841
842
0
  ZEND_PARSE_PARAMETERS_START(2, 2)
843
0
    Z_PARAM_STRING(format, format_len)
844
0
    Z_PARAM_ARRAY_HT(array)
845
0
  ZEND_PARSE_PARAMETERS_END();
846
847
0
  args = php_formatted_print_get_array(array, &argc);
848
849
0
  result = php_formatted_print(format, format_len, args, argc, -1);
850
0
  efree(args);
851
0
  if (result == NULL) {
852
0
    RETURN_THROWS();
853
0
  }
854
0
  rlen = PHPWRITE(ZSTR_VAL(result), ZSTR_LEN(result));
855
0
  zend_string_efree(result);
856
0
  RETURN_LONG(rlen);
857
0
}
858
/* }}} */
859
860
/* {{{ Output a formatted string into a stream */
861
PHP_FUNCTION(fprintf)
862
0
{
863
0
  php_stream *stream;
864
0
  char *format;
865
0
  size_t format_len;
866
0
  zval *args = NULL;
867
0
  int argc = 0;
868
0
  zend_string *result;
869
870
0
  ZEND_PARSE_PARAMETERS_START(2, -1)
871
0
    PHP_Z_PARAM_STREAM(stream)
872
0
    Z_PARAM_STRING(format, format_len)
873
0
    Z_PARAM_VARIADIC('*', args, argc)
874
0
  ZEND_PARSE_PARAMETERS_END();
875
876
0
  result = php_formatted_print(format, format_len, args, argc, 2);
877
0
  if (result == NULL) {
878
0
    RETURN_THROWS();
879
0
  }
880
881
0
  php_stream_write(stream, ZSTR_VAL(result), ZSTR_LEN(result));
882
883
0
  RETVAL_LONG(ZSTR_LEN(result));
884
0
  zend_string_efree(result);
885
0
}
886
/* }}} */
887
888
/* {{{ Output a formatted string into a stream */
889
PHP_FUNCTION(vfprintf)
890
0
{
891
0
  php_stream *stream;
892
0
  char *format;
893
0
  size_t format_len;
894
0
  zval *args;
895
0
  zend_array *array;
896
0
  int argc;
897
0
  zend_string *result;
898
899
0
  ZEND_PARSE_PARAMETERS_START(3, 3)
900
0
    PHP_Z_PARAM_STREAM(stream)
901
0
    Z_PARAM_STRING(format, format_len)
902
0
    Z_PARAM_ARRAY_HT(array)
903
0
  ZEND_PARSE_PARAMETERS_END();
904
905
0
  args = php_formatted_print_get_array(array, &argc);
906
907
0
  result = php_formatted_print(format, format_len, args, argc, -1);
908
0
  efree(args);
909
0
  if (result == NULL) {
910
0
    RETURN_THROWS();
911
0
  }
912
913
0
  php_stream_write(stream, ZSTR_VAL(result), ZSTR_LEN(result));
914
915
0
  RETVAL_LONG(ZSTR_LEN(result));
916
0
  zend_string_efree(result);
917
0
}
918
/* }}} */