spprintf.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Marcus Boerger <helly@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. /* This is the spprintf implementation.
  20. * It has emerged from apache snprintf. See original header:
  21. */
  22. /* ====================================================================
  23. * Copyright (c) 1995-1998 The Apache Group. All rights reserved.
  24. *
  25. * Redistribution and use in source and binary forms, with or without
  26. * modification, are permitted provided that the following conditions
  27. * are met:
  28. *
  29. * 1. Redistributions of source code must retain the above copyright
  30. * notice, this list of conditions and the following disclaimer.
  31. *
  32. * 2. Redistributions in binary form must reproduce the above copyright
  33. * notice, this list of conditions and the following disclaimer in
  34. * the documentation and/or other materials provided with the
  35. * distribution.
  36. *
  37. * 3. All advertising materials mentioning features or use of this
  38. * software must display the following acknowledgment:
  39. * "This product includes software developed by the Apache Group
  40. * for use in the Apache HTTP server project (http://www.apache.org/)."
  41. *
  42. * 4. The names "Apache Server" and "Apache Group" must not be used to
  43. * endorse or promote products derived from this software without
  44. * prior written permission.
  45. *
  46. * 5. Redistributions of any form whatsoever must retain the following
  47. * acknowledgment:
  48. * "This product includes software developed by the Apache Group
  49. * for use in the Apache HTTP server project (http://www.apache.org/)."
  50. *
  51. * THIS SOFTWARE IS PROVIDED BY THE APACHE GROUP ``AS IS'' AND ANY
  52. * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  53. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  54. * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE APACHE GROUP OR
  55. * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  56. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  57. * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  58. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  59. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
  60. * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  61. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
  62. * OF THE POSSIBILITY OF SUCH DAMAGE.
  63. * ====================================================================
  64. *
  65. * This software consists of voluntary contributions made by many
  66. * individuals on behalf of the Apache Group and was originally based
  67. * on public domain software written at the National Center for
  68. * Supercomputing Applications, University of Illinois, Urbana-Champaign.
  69. * For more information on the Apache Group and the Apache HTTP server
  70. * project, please see <http://www.apache.org/>.
  71. *
  72. * This code is based on, and used with the permission of, the
  73. * SIO stdio-replacement strx_* functions by Panos Tsirigotis
  74. * <panos@alumni.cs.colorado.edu> for xinetd.
  75. */
  76. #define _GNU_SOURCE
  77. #include "php.h"
  78. #include <stddef.h>
  79. #include <stdio.h>
  80. #include <ctype.h>
  81. #include <sys/types.h>
  82. #include <stdarg.h>
  83. #include <string.h>
  84. #include <stdlib.h>
  85. #include <math.h>
  86. #ifdef HAVE_INTTYPES_H
  87. #include <inttypes.h>
  88. #endif
  89. #ifdef HAVE_LOCALE_H
  90. #include <locale.h>
  91. #ifdef ZTS
  92. #include "ext/standard/php_string.h"
  93. #define LCONV_DECIMAL_POINT (*lconv.decimal_point)
  94. #else
  95. #define LCONV_DECIMAL_POINT (*lconv->decimal_point)
  96. #endif
  97. #else
  98. #define LCONV_DECIMAL_POINT '.'
  99. #endif
  100. #include "snprintf.h"
  101. #define FALSE 0
  102. #define TRUE 1
  103. #define NUL '\0'
  104. #define INT_NULL ((int *)0)
  105. #define S_NULL "(null)"
  106. #define S_NULL_LEN 6
  107. #define FLOAT_DIGITS 6
  108. #define EXPONENT_LENGTH 10
  109. #include "ext/standard/php_smart_str.h"
  110. /* {{{ macros */
  111. /*
  112. * NUM_BUF_SIZE is the size of the buffer used for arithmetic conversions
  113. *
  114. * XXX: this is a magic number; do not decrease it
  115. * Emax = 1023
  116. * NDIG = 320
  117. * NUM_BUF_SIZE >= strlen("-") + Emax + strlrn(".") + NDIG + strlen("E+1023") + 1;
  118. */
  119. #define NUM_BUF_SIZE 2048
  120. /*
  121. * The INS_CHAR macro inserts a character in the buffer.
  122. *
  123. * NOTE: Evaluation of the ch argument should not have any side-effects
  124. */
  125. #define INS_CHAR_NR(xbuf, ch) do { \
  126. smart_str_appendc(xbuf, ch); \
  127. } while (0)
  128. #define INS_STRING(xbuf, s, slen) do { \
  129. smart_str_appendl(xbuf, s, slen); \
  130. } while (0)
  131. #define INS_CHAR(xbuf, ch) \
  132. INS_CHAR_NR(xbuf, ch)
  133. /*
  134. * Macro that does padding. The padding is done by printing
  135. * the character ch.
  136. */
  137. #define PAD(xbuf, count, ch) do { \
  138. if ((count) > 0) { \
  139. size_t newlen; \
  140. smart_str_alloc(xbuf, (count), 0); \
  141. memset(xbuf->c + xbuf->len, ch, (count)); \
  142. xbuf->len += (count); \
  143. } \
  144. } while (0)
  145. #define NUM(c) (c - '0')
  146. #define STR_TO_DEC(str, num) do { \
  147. num = NUM(*str++); \
  148. while (isdigit((int)*str)) { \
  149. num *= 10; \
  150. num += NUM(*str++); \
  151. if (num >= INT_MAX / 10) { \
  152. while (isdigit((int)*str++)); \
  153. break; \
  154. } \
  155. } \
  156. } while (0)
  157. /*
  158. * This macro does zero padding so that the precision
  159. * requirement is satisfied. The padding is done by
  160. * adding '0's to the left of the string that is going
  161. * to be printed.
  162. */
  163. #define FIX_PRECISION(adjust, precision, s, s_len) do { \
  164. if (adjust) \
  165. while (s_len < precision) { \
  166. *--s = '0'; \
  167. s_len++; \
  168. } \
  169. } while (0)
  170. /* }}} */
  171. #if !HAVE_STRNLEN
  172. static size_t strnlen(const char *s, size_t maxlen) {
  173. char *r = memchr(s, '\0', maxlen);
  174. return r ? r-s : maxlen;
  175. }
  176. #endif
  177. /*
  178. * Do format conversion placing the output in buffer
  179. */
  180. static void xbuf_format_converter(smart_str *xbuf, const char *fmt, va_list ap) /* {{{ */
  181. {
  182. char *s = NULL;
  183. int s_len, free_zcopy;
  184. zval *zvp, zcopy;
  185. int min_width = 0;
  186. int precision = 0;
  187. enum {
  188. LEFT, RIGHT
  189. } adjust;
  190. char pad_char;
  191. char prefix_char;
  192. double fp_num;
  193. wide_int i_num = (wide_int) 0;
  194. u_wide_int ui_num = (u_wide_int) 0;
  195. char num_buf[NUM_BUF_SIZE];
  196. char char_buf[2]; /* for printing %% and %<unknown> */
  197. #ifdef HAVE_LOCALE_H
  198. #ifdef ZTS
  199. struct lconv lconv;
  200. #else
  201. struct lconv *lconv = NULL;
  202. #endif
  203. #endif
  204. /*
  205. * Flag variables
  206. */
  207. length_modifier_e modifier;
  208. boolean_e alternate_form;
  209. boolean_e print_sign;
  210. boolean_e print_blank;
  211. boolean_e adjust_precision;
  212. boolean_e adjust_width;
  213. bool_int is_negative;
  214. while (*fmt) {
  215. if (*fmt != '%') {
  216. INS_CHAR(xbuf, *fmt);
  217. } else {
  218. /*
  219. * Default variable settings
  220. */
  221. adjust = RIGHT;
  222. alternate_form = print_sign = print_blank = NO;
  223. pad_char = ' ';
  224. prefix_char = NUL;
  225. free_zcopy = 0;
  226. fmt++;
  227. /*
  228. * Try to avoid checking for flags, width or precision
  229. */
  230. if (isascii((int)*fmt) && !islower((int)*fmt)) {
  231. /*
  232. * Recognize flags: -, #, BLANK, +
  233. */
  234. for (;; fmt++) {
  235. if (*fmt == '-')
  236. adjust = LEFT;
  237. else if (*fmt == '+')
  238. print_sign = YES;
  239. else if (*fmt == '#')
  240. alternate_form = YES;
  241. else if (*fmt == ' ')
  242. print_blank = YES;
  243. else if (*fmt == '0')
  244. pad_char = '0';
  245. else
  246. break;
  247. }
  248. /*
  249. * Check if a width was specified
  250. */
  251. if (isdigit((int)*fmt)) {
  252. STR_TO_DEC(fmt, min_width);
  253. adjust_width = YES;
  254. } else if (*fmt == '*') {
  255. min_width = va_arg(ap, int);
  256. fmt++;
  257. adjust_width = YES;
  258. if (min_width < 0) {
  259. adjust = LEFT;
  260. min_width = -min_width;
  261. }
  262. } else
  263. adjust_width = NO;
  264. /*
  265. * Check if a precision was specified
  266. */
  267. if (*fmt == '.') {
  268. adjust_precision = YES;
  269. fmt++;
  270. if (isdigit((int)*fmt)) {
  271. STR_TO_DEC(fmt, precision);
  272. } else if (*fmt == '*') {
  273. precision = va_arg(ap, int);
  274. fmt++;
  275. if (precision < 0)
  276. precision = 0;
  277. } else
  278. precision = 0;
  279. if (precision > FORMAT_CONV_MAX_PRECISION) {
  280. precision = FORMAT_CONV_MAX_PRECISION;
  281. }
  282. } else
  283. adjust_precision = NO;
  284. } else
  285. adjust_precision = adjust_width = NO;
  286. /*
  287. * Modifier check
  288. */
  289. switch (*fmt) {
  290. case 'L':
  291. fmt++;
  292. modifier = LM_LONG_DOUBLE;
  293. break;
  294. case 'I':
  295. fmt++;
  296. #if SIZEOF_LONG_LONG
  297. if (*fmt == '6' && *(fmt+1) == '4') {
  298. fmt += 2;
  299. modifier = LM_LONG_LONG;
  300. } else
  301. #endif
  302. if (*fmt == '3' && *(fmt+1) == '2') {
  303. fmt += 2;
  304. modifier = LM_LONG;
  305. } else {
  306. #ifdef _WIN64
  307. modifier = LM_LONG_LONG;
  308. #else
  309. modifier = LM_LONG;
  310. #endif
  311. }
  312. break;
  313. case 'l':
  314. fmt++;
  315. #if SIZEOF_LONG_LONG
  316. if (*fmt == 'l') {
  317. fmt++;
  318. modifier = LM_LONG_LONG;
  319. } else
  320. #endif
  321. modifier = LM_LONG;
  322. break;
  323. case 'z':
  324. fmt++;
  325. modifier = LM_SIZE_T;
  326. break;
  327. case 'j':
  328. fmt++;
  329. #if SIZEOF_INTMAX_T
  330. modifier = LM_INTMAX_T;
  331. #else
  332. modifier = LM_SIZE_T;
  333. #endif
  334. break;
  335. case 't':
  336. fmt++;
  337. #if SIZEOF_PTRDIFF_T
  338. modifier = LM_PTRDIFF_T;
  339. #else
  340. modifier = LM_SIZE_T;
  341. #endif
  342. break;
  343. case 'h':
  344. fmt++;
  345. if (*fmt == 'h') {
  346. fmt++;
  347. }
  348. /* these are promoted to int, so no break */
  349. default:
  350. modifier = LM_STD;
  351. break;
  352. }
  353. /*
  354. * Argument extraction and printing.
  355. * First we determine the argument type.
  356. * Then, we convert the argument to a string.
  357. * On exit from the switch, s points to the string that
  358. * must be printed, s_len has the length of the string
  359. * The precision requirements, if any, are reflected in s_len.
  360. *
  361. * NOTE: pad_char may be set to '0' because of the 0 flag.
  362. * It is reset to ' ' by non-numeric formats
  363. */
  364. switch (*fmt) {
  365. case 'Z':
  366. zvp = (zval*) va_arg(ap, zval*);
  367. zend_make_printable_zval(zvp, &zcopy, &free_zcopy);
  368. if (free_zcopy) {
  369. zvp = &zcopy;
  370. }
  371. s_len = Z_STRLEN_P(zvp);
  372. s = Z_STRVAL_P(zvp);
  373. if (adjust_precision && precision < s_len) {
  374. s_len = precision;
  375. }
  376. break;
  377. case 'u':
  378. switch(modifier) {
  379. default:
  380. i_num = (wide_int) va_arg(ap, unsigned int);
  381. break;
  382. case LM_LONG_DOUBLE:
  383. goto fmt_error;
  384. case LM_LONG:
  385. i_num = (wide_int) va_arg(ap, unsigned long int);
  386. break;
  387. case LM_SIZE_T:
  388. i_num = (wide_int) va_arg(ap, size_t);
  389. break;
  390. #if SIZEOF_LONG_LONG
  391. case LM_LONG_LONG:
  392. i_num = (wide_int) va_arg(ap, u_wide_int);
  393. break;
  394. #endif
  395. #if SIZEOF_INTMAX_T
  396. case LM_INTMAX_T:
  397. i_num = (wide_int) va_arg(ap, uintmax_t);
  398. break;
  399. #endif
  400. #if SIZEOF_PTRDIFF_T
  401. case LM_PTRDIFF_T:
  402. i_num = (wide_int) va_arg(ap, ptrdiff_t);
  403. break;
  404. #endif
  405. }
  406. /*
  407. * The rest also applies to other integer formats, so fall
  408. * into that case.
  409. */
  410. case 'd':
  411. case 'i':
  412. /*
  413. * Get the arg if we haven't already.
  414. */
  415. if ((*fmt) != 'u') {
  416. switch(modifier) {
  417. default:
  418. i_num = (wide_int) va_arg(ap, int);
  419. break;
  420. case LM_LONG_DOUBLE:
  421. goto fmt_error;
  422. case LM_LONG:
  423. i_num = (wide_int) va_arg(ap, long int);
  424. break;
  425. case LM_SIZE_T:
  426. #if SIZEOF_SSIZE_T
  427. i_num = (wide_int) va_arg(ap, ssize_t);
  428. #else
  429. i_num = (wide_int) va_arg(ap, size_t);
  430. #endif
  431. break;
  432. #if SIZEOF_LONG_LONG
  433. case LM_LONG_LONG:
  434. i_num = (wide_int) va_arg(ap, wide_int);
  435. break;
  436. #endif
  437. #if SIZEOF_INTMAX_T
  438. case LM_INTMAX_T:
  439. i_num = (wide_int) va_arg(ap, intmax_t);
  440. break;
  441. #endif
  442. #if SIZEOF_PTRDIFF_T
  443. case LM_PTRDIFF_T:
  444. i_num = (wide_int) va_arg(ap, ptrdiff_t);
  445. break;
  446. #endif
  447. }
  448. }
  449. s = ap_php_conv_10(i_num, (*fmt) == 'u', &is_negative,
  450. &num_buf[NUM_BUF_SIZE], &s_len);
  451. FIX_PRECISION(adjust_precision, precision, s, s_len);
  452. if (*fmt != 'u') {
  453. if (is_negative)
  454. prefix_char = '-';
  455. else if (print_sign)
  456. prefix_char = '+';
  457. else if (print_blank)
  458. prefix_char = ' ';
  459. }
  460. break;
  461. case 'o':
  462. switch(modifier) {
  463. default:
  464. ui_num = (u_wide_int) va_arg(ap, unsigned int);
  465. break;
  466. case LM_LONG_DOUBLE:
  467. goto fmt_error;
  468. case LM_LONG:
  469. ui_num = (u_wide_int) va_arg(ap, unsigned long int);
  470. break;
  471. case LM_SIZE_T:
  472. ui_num = (u_wide_int) va_arg(ap, size_t);
  473. break;
  474. #if SIZEOF_LONG_LONG
  475. case LM_LONG_LONG:
  476. ui_num = (u_wide_int) va_arg(ap, u_wide_int);
  477. break;
  478. #endif
  479. #if SIZEOF_INTMAX_T
  480. case LM_INTMAX_T:
  481. ui_num = (u_wide_int) va_arg(ap, uintmax_t);
  482. break;
  483. #endif
  484. #if SIZEOF_PTRDIFF_T
  485. case LM_PTRDIFF_T:
  486. ui_num = (u_wide_int) va_arg(ap, ptrdiff_t);
  487. break;
  488. #endif
  489. }
  490. s = ap_php_conv_p2(ui_num, 3, *fmt,
  491. &num_buf[NUM_BUF_SIZE], &s_len);
  492. FIX_PRECISION(adjust_precision, precision, s, s_len);
  493. if (alternate_form && *s != '0') {
  494. *--s = '0';
  495. s_len++;
  496. }
  497. break;
  498. case 'x':
  499. case 'X':
  500. switch(modifier) {
  501. default:
  502. ui_num = (u_wide_int) va_arg(ap, unsigned int);
  503. break;
  504. case LM_LONG_DOUBLE:
  505. goto fmt_error;
  506. case LM_LONG:
  507. ui_num = (u_wide_int) va_arg(ap, unsigned long int);
  508. break;
  509. case LM_SIZE_T:
  510. ui_num = (u_wide_int) va_arg(ap, size_t);
  511. break;
  512. #if SIZEOF_LONG_LONG
  513. case LM_LONG_LONG:
  514. ui_num = (u_wide_int) va_arg(ap, u_wide_int);
  515. break;
  516. #endif
  517. #if SIZEOF_INTMAX_T
  518. case LM_INTMAX_T:
  519. ui_num = (u_wide_int) va_arg(ap, uintmax_t);
  520. break;
  521. #endif
  522. #if SIZEOF_PTRDIFF_T
  523. case LM_PTRDIFF_T:
  524. ui_num = (u_wide_int) va_arg(ap, ptrdiff_t);
  525. break;
  526. #endif
  527. }
  528. s = ap_php_conv_p2(ui_num, 4, *fmt,
  529. &num_buf[NUM_BUF_SIZE], &s_len);
  530. FIX_PRECISION(adjust_precision, precision, s, s_len);
  531. if (alternate_form && ui_num != 0) {
  532. *--s = *fmt; /* 'x' or 'X' */
  533. *--s = '0';
  534. s_len += 2;
  535. }
  536. break;
  537. case 's':
  538. case 'v':
  539. s = va_arg(ap, char *);
  540. if (s != NULL) {
  541. if (!adjust_precision) {
  542. s_len = strlen(s);
  543. } else {
  544. s_len = strnlen(s, precision);
  545. }
  546. } else {
  547. s = S_NULL;
  548. s_len = S_NULL_LEN;
  549. }
  550. pad_char = ' ';
  551. break;
  552. case 'f':
  553. case 'F':
  554. case 'e':
  555. case 'E':
  556. switch(modifier) {
  557. case LM_LONG_DOUBLE:
  558. fp_num = (double) va_arg(ap, long double);
  559. break;
  560. case LM_STD:
  561. fp_num = va_arg(ap, double);
  562. break;
  563. default:
  564. goto fmt_error;
  565. }
  566. if (zend_isnan(fp_num)) {
  567. s = "nan";
  568. s_len = 3;
  569. } else if (zend_isinf(fp_num)) {
  570. s = "inf";
  571. s_len = 3;
  572. } else {
  573. #ifdef HAVE_LOCALE_H
  574. #ifdef ZTS
  575. localeconv_r(&lconv);
  576. #else
  577. if (!lconv) {
  578. lconv = localeconv();
  579. }
  580. #endif
  581. #endif
  582. s = php_conv_fp((*fmt == 'f')?'F':*fmt, fp_num, alternate_form,
  583. (adjust_precision == NO) ? FLOAT_DIGITS : precision,
  584. (*fmt == 'f')?LCONV_DECIMAL_POINT:'.',
  585. &is_negative, &num_buf[1], &s_len);
  586. if (is_negative)
  587. prefix_char = '-';
  588. else if (print_sign)
  589. prefix_char = '+';
  590. else if (print_blank)
  591. prefix_char = ' ';
  592. }
  593. break;
  594. case 'g':
  595. case 'k':
  596. case 'G':
  597. case 'H':
  598. switch(modifier) {
  599. case LM_LONG_DOUBLE:
  600. fp_num = (double) va_arg(ap, long double);
  601. break;
  602. case LM_STD:
  603. fp_num = va_arg(ap, double);
  604. break;
  605. default:
  606. goto fmt_error;
  607. }
  608. if (zend_isnan(fp_num)) {
  609. s = "NAN";
  610. s_len = 3;
  611. break;
  612. } else if (zend_isinf(fp_num)) {
  613. if (fp_num > 0) {
  614. s = "INF";
  615. s_len = 3;
  616. } else {
  617. s = "-INF";
  618. s_len = 4;
  619. }
  620. break;
  621. }
  622. if (adjust_precision == NO)
  623. precision = FLOAT_DIGITS;
  624. else if (precision == 0)
  625. precision = 1;
  626. /*
  627. * * We use &num_buf[ 1 ], so that we have room for the sign
  628. */
  629. #ifdef HAVE_LOCALE_H
  630. #ifdef ZTS
  631. localeconv_r(&lconv);
  632. #else
  633. if (!lconv) {
  634. lconv = localeconv();
  635. }
  636. #endif
  637. #endif
  638. s = php_gcvt(fp_num, precision, (*fmt=='H' || *fmt == 'k') ? '.' : LCONV_DECIMAL_POINT, (*fmt == 'G' || *fmt == 'H')?'E':'e', &num_buf[1]);
  639. if (*s == '-')
  640. prefix_char = *s++;
  641. else if (print_sign)
  642. prefix_char = '+';
  643. else if (print_blank)
  644. prefix_char = ' ';
  645. s_len = strlen(s);
  646. if (alternate_form && (strchr(s, '.')) == NULL)
  647. s[s_len++] = '.';
  648. break;
  649. case 'c':
  650. char_buf[0] = (char) (va_arg(ap, int));
  651. s = &char_buf[0];
  652. s_len = 1;
  653. pad_char = ' ';
  654. break;
  655. case '%':
  656. char_buf[0] = '%';
  657. s = &char_buf[0];
  658. s_len = 1;
  659. pad_char = ' ';
  660. break;
  661. case 'n':
  662. *(va_arg(ap, int *)) = xbuf->len;
  663. goto skip_output;
  664. /*
  665. * Always extract the argument as a "char *" pointer. We
  666. * should be using "void *" but there are still machines
  667. * that don't understand it.
  668. * If the pointer size is equal to the size of an unsigned
  669. * integer we convert the pointer to a hex number, otherwise
  670. * we print "%p" to indicate that we don't handle "%p".
  671. */
  672. case 'p':
  673. if (sizeof(char *) <= sizeof(u_wide_int)) {
  674. ui_num = (u_wide_int)((size_t) va_arg(ap, char *));
  675. s = ap_php_conv_p2(ui_num, 4, 'x',
  676. &num_buf[NUM_BUF_SIZE], &s_len);
  677. if (ui_num != 0) {
  678. *--s = 'x';
  679. *--s = '0';
  680. s_len += 2;
  681. }
  682. } else {
  683. s = "%p";
  684. s_len = 2;
  685. }
  686. pad_char = ' ';
  687. break;
  688. case NUL:
  689. /*
  690. * The last character of the format string was %.
  691. * We ignore it.
  692. */
  693. continue;
  694. fmt_error:
  695. php_error(E_ERROR, "Illegal length modifier specified '%c' in s[np]printf call", *fmt);
  696. /*
  697. * The default case is for unrecognized %'s.
  698. * We print %<char> to help the user identify what
  699. * option is not understood.
  700. * This is also useful in case the user wants to pass
  701. * the output of format_converter to another function
  702. * that understands some other %<char> (like syslog).
  703. * Note that we can't point s inside fmt because the
  704. * unknown <char> could be preceded by width etc.
  705. */
  706. default:
  707. char_buf[0] = '%';
  708. char_buf[1] = *fmt;
  709. s = char_buf;
  710. s_len = 2;
  711. pad_char = ' ';
  712. break;
  713. }
  714. if (prefix_char != NUL) {
  715. *--s = prefix_char;
  716. s_len++;
  717. }
  718. if (adjust_width && adjust == RIGHT && min_width > s_len) {
  719. if (pad_char == '0' && prefix_char != NUL) {
  720. INS_CHAR(xbuf, *s);
  721. s++;
  722. s_len--;
  723. min_width--;
  724. }
  725. PAD(xbuf, min_width - s_len, pad_char);
  726. }
  727. /*
  728. * Print the string s.
  729. */
  730. INS_STRING(xbuf, s, s_len);
  731. if (adjust_width && adjust == LEFT && min_width > s_len)
  732. PAD(xbuf, min_width - s_len, pad_char);
  733. if (free_zcopy) {
  734. zval_dtor(&zcopy);
  735. }
  736. }
  737. skip_output:
  738. fmt++;
  739. }
  740. return;
  741. }
  742. /* }}} */
  743. /*
  744. * This is the general purpose conversion function.
  745. */
  746. PHPAPI int vspprintf(char **pbuf, size_t max_len, const char *format, va_list ap) /* {{{ */
  747. {
  748. smart_str xbuf = {0};
  749. /* since there are places where (v)spprintf called without checking for null,
  750. a bit of defensive coding here */
  751. if(!pbuf) {
  752. return 0;
  753. }
  754. xbuf_format_converter(&xbuf, format, ap);
  755. if (max_len && xbuf.len > max_len) {
  756. xbuf.len = max_len;
  757. }
  758. smart_str_0(&xbuf);
  759. *pbuf = xbuf.c;
  760. return xbuf.len;
  761. }
  762. /* }}} */
  763. PHPAPI int spprintf(char **pbuf, size_t max_len, const char *format, ...) /* {{{ */
  764. {
  765. int cc;
  766. va_list ap;
  767. va_start(ap, format);
  768. cc = vspprintf(pbuf, max_len, format, ap);
  769. va_end(ap);
  770. return (cc);
  771. }
  772. /* }}} */
  773. /*
  774. * Local variables:
  775. * tab-width: 4
  776. * c-basic-offset: 4
  777. * End:
  778. * vim600: sw=4 ts=4 fdm=marker
  779. * vim<600: sw=4 ts=4
  780. */