mirror of
https://github.com/php/php-src.git
synced 2024-09-23 10:57:26 +00:00
8d3f8ca12a
The $Id$ keywords were used in Subversion where they can be substituted with filename, last revision number change, last changed date, and last user who changed it. In Git this functionality is different and can be done with Git attribute ident. These need to be defined manually for each file in the .gitattributes file and are afterwards replaced with 40-character hexadecimal blob object name which is based only on the particular file contents. This patch simplifies handling of $Id$ keywords by removing them since they are not used anymore.
785 lines
20 KiB
C
785 lines
20 KiB
C
/*
|
||
+----------------------------------------------------------------------+
|
||
| PHP Version 7 |
|
||
+----------------------------------------------------------------------+
|
||
| Copyright (c) 1997-2018 The PHP Group |
|
||
+----------------------------------------------------------------------+
|
||
| This source file is subject to version 3.01 of the PHP license, |
|
||
| that is bundled with this package in the file LICENSE, and is |
|
||
| available through the world-wide-web at the following url: |
|
||
| http://www.php.net/license/3_01.txt |
|
||
| If you did not receive a copy of the PHP license and are unable to |
|
||
| obtain it through the world-wide-web, please send a note to |
|
||
| license@php.net so we can mail you a copy immediately. |
|
||
+----------------------------------------------------------------------+
|
||
| Author: Stig S<>ther Bakken <ssb@php.net> |
|
||
+----------------------------------------------------------------------+
|
||
*/
|
||
|
||
#include <math.h> /* modf() */
|
||
#include "php.h"
|
||
#include "ext/standard/head.h"
|
||
#include "php_string.h"
|
||
#include "zend_execute.h"
|
||
#include <stdio.h>
|
||
|
||
#ifdef HAVE_LOCALE_H
|
||
#include <locale.h>
|
||
#ifdef ZTS
|
||
#include "ext/standard/php_string.h"
|
||
#define LCONV_DECIMAL_POINT (*lconv.decimal_point)
|
||
#else
|
||
#define LCONV_DECIMAL_POINT (*lconv->decimal_point)
|
||
#endif
|
||
#else
|
||
#define LCONV_DECIMAL_POINT '.'
|
||
#endif
|
||
|
||
#define ALIGN_LEFT 0
|
||
#define ALIGN_RIGHT 1
|
||
#define ADJ_WIDTH 1
|
||
#define ADJ_PRECISION 2
|
||
#define NUM_BUF_SIZE 500
|
||
#define FLOAT_PRECISION 6
|
||
#define MAX_FLOAT_PRECISION 53
|
||
|
||
#if 0
|
||
/* trick to control varargs functions through cpp */
|
||
# define PRINTF_DEBUG(arg) php_printf arg
|
||
#else
|
||
# define PRINTF_DEBUG(arg)
|
||
#endif
|
||
|
||
static const char hexchars[] = "0123456789abcdef";
|
||
static const char HEXCHARS[] = "0123456789ABCDEF";
|
||
|
||
/* php_spintf_appendchar() {{{ */
|
||
inline static void
|
||
php_sprintf_appendchar(zend_string **buffer, size_t *pos, char add)
|
||
{
|
||
if (!*buffer || (*pos + 1) >= ZSTR_LEN(*buffer)) {
|
||
PRINTF_DEBUG(("%s(): ereallocing buffer to %d bytes\n", get_active_function_name(), ZSTR_LEN(*buffer)));
|
||
*buffer = zend_string_extend(*buffer, ZSTR_LEN(*buffer) << 1, 0);
|
||
}
|
||
PRINTF_DEBUG(("sprintf: appending '%c', pos=\n", add, *pos));
|
||
ZSTR_VAL(*buffer)[(*pos)++] = add;
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_appendstring() {{{ */
|
||
inline static void
|
||
php_sprintf_appendstring(zend_string **buffer, size_t *pos, char *add,
|
||
size_t min_width, size_t max_width, char padding,
|
||
size_t alignment, size_t len, int neg, int expprec, int always_sign)
|
||
{
|
||
register size_t npad;
|
||
size_t req_size;
|
||
size_t copy_len;
|
||
size_t m_width;
|
||
|
||
copy_len = (expprec ? MIN(max_width, len) : len);
|
||
npad = (min_width < copy_len) ? 0 : min_width - copy_len;
|
||
|
||
PRINTF_DEBUG(("sprintf: appendstring(%x, %d, %d, \"%s\", %d, '%c', %d)\n",
|
||
*buffer, *pos, ZSTR_LEN(*buffer), add, min_width, padding, alignment));
|
||
m_width = MAX(min_width, copy_len);
|
||
|
||
if(m_width > INT_MAX - *pos - 1) {
|
||
zend_error_noreturn(E_ERROR, "Field width %zd is too long", m_width);
|
||
}
|
||
|
||
req_size = *pos + m_width + 1;
|
||
|
||
if (!*buffer || req_size > ZSTR_LEN(*buffer)) {
|
||
size_t size = ZSTR_LEN(*buffer);
|
||
while (req_size > size) {
|
||
if (size > ZEND_SIZE_MAX/2) {
|
||
zend_error_noreturn(E_ERROR, "Field width %zd is too long", req_size);
|
||
}
|
||
size <<= 1;
|
||
}
|
||
PRINTF_DEBUG(("sprintf ereallocing buffer to %d bytes\n", size));
|
||
*buffer = zend_string_extend(*buffer, size, 0);
|
||
}
|
||
if (alignment == ALIGN_RIGHT) {
|
||
if ((neg || always_sign) && padding=='0') {
|
||
ZSTR_VAL(*buffer)[(*pos)++] = (neg) ? '-' : '+';
|
||
add++;
|
||
len--;
|
||
copy_len--;
|
||
}
|
||
while (npad-- > 0) {
|
||
ZSTR_VAL(*buffer)[(*pos)++] = padding;
|
||
}
|
||
}
|
||
PRINTF_DEBUG(("sprintf: appending \"%s\"\n", add));
|
||
memcpy(&ZSTR_VAL(*buffer)[*pos], add, copy_len + 1);
|
||
*pos += copy_len;
|
||
if (alignment == ALIGN_LEFT) {
|
||
while (npad--) {
|
||
ZSTR_VAL(*buffer)[(*pos)++] = padding;
|
||
}
|
||
}
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_appendint() {{{ */
|
||
inline static void
|
||
php_sprintf_appendint(zend_string **buffer, size_t *pos, zend_long number,
|
||
size_t width, char padding, size_t alignment,
|
||
int always_sign)
|
||
{
|
||
char numbuf[NUM_BUF_SIZE];
|
||
register zend_ulong magn, nmagn;
|
||
register unsigned int i = NUM_BUF_SIZE - 1, neg = 0;
|
||
|
||
PRINTF_DEBUG(("sprintf: appendint(%x, %x, %x, %d, %d, '%c', %d)\n",
|
||
*buffer, pos, &ZSTR_LEN(*buffer), number, width, padding, alignment));
|
||
if (number < 0) {
|
||
neg = 1;
|
||
magn = ((zend_ulong) -(number + 1)) + 1;
|
||
} else {
|
||
magn = (zend_ulong) number;
|
||
}
|
||
|
||
/* Can't right-pad 0's on integers */
|
||
if(alignment==0 && padding=='0') padding=' ';
|
||
|
||
numbuf[i] = '\0';
|
||
|
||
do {
|
||
nmagn = magn / 10;
|
||
|
||
numbuf[--i] = (unsigned char)(magn - (nmagn * 10)) + '0';
|
||
magn = nmagn;
|
||
}
|
||
while (magn > 0 && i > 1);
|
||
if (neg) {
|
||
numbuf[--i] = '-';
|
||
} else if (always_sign) {
|
||
numbuf[--i] = '+';
|
||
}
|
||
PRINTF_DEBUG(("sprintf: appending %d as \"%s\", i=%d\n",
|
||
number, &numbuf[i], i));
|
||
php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
|
||
padding, alignment, (NUM_BUF_SIZE - 1) - i,
|
||
neg, 0, always_sign);
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_appenduint() {{{ */
|
||
inline static void
|
||
php_sprintf_appenduint(zend_string **buffer, size_t *pos,
|
||
zend_ulong number,
|
||
size_t width, char padding, size_t alignment)
|
||
{
|
||
char numbuf[NUM_BUF_SIZE];
|
||
register zend_ulong magn, nmagn;
|
||
register unsigned int i = NUM_BUF_SIZE - 1;
|
||
|
||
PRINTF_DEBUG(("sprintf: appenduint(%x, %x, %x, %d, %d, '%c', %d)\n",
|
||
*buffer, pos, &ZSTR_LEN(*buffer), number, width, padding, alignment));
|
||
magn = (zend_ulong) number;
|
||
|
||
/* Can't right-pad 0's on integers */
|
||
if (alignment == 0 && padding == '0') padding = ' ';
|
||
|
||
numbuf[i] = '\0';
|
||
|
||
do {
|
||
nmagn = magn / 10;
|
||
|
||
numbuf[--i] = (unsigned char)(magn - (nmagn * 10)) + '0';
|
||
magn = nmagn;
|
||
} while (magn > 0 && i > 0);
|
||
|
||
PRINTF_DEBUG(("sprintf: appending %d as \"%s\", i=%d\n", number, &numbuf[i], i));
|
||
php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
|
||
padding, alignment, (NUM_BUF_SIZE - 1) - i, 0, 0, 0);
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_appenddouble() {{{ */
|
||
inline static void
|
||
php_sprintf_appenddouble(zend_string **buffer, size_t *pos,
|
||
double number,
|
||
size_t width, char padding,
|
||
size_t alignment, int precision,
|
||
int adjust, char fmt,
|
||
int always_sign
|
||
)
|
||
{
|
||
char num_buf[NUM_BUF_SIZE];
|
||
char *s = NULL;
|
||
size_t s_len = 0;
|
||
int is_negative = 0;
|
||
#ifdef HAVE_LOCALE_H
|
||
#ifdef ZTS
|
||
struct lconv lconv;
|
||
#else
|
||
struct lconv *lconv;
|
||
#endif
|
||
#endif
|
||
|
||
PRINTF_DEBUG(("sprintf: appenddouble(%x, %x, %x, %f, %d, '%c', %d, %c)\n",
|
||
*buffer, pos, &ZSTR_LEN(*buffer), number, width, padding, alignment, fmt));
|
||
if ((adjust & ADJ_PRECISION) == 0) {
|
||
precision = FLOAT_PRECISION;
|
||
} else if (precision > MAX_FLOAT_PRECISION) {
|
||
php_error_docref(NULL, E_NOTICE, "Requested precision of %d digits was truncated to PHP maximum of %d digits", precision, MAX_FLOAT_PRECISION);
|
||
precision = MAX_FLOAT_PRECISION;
|
||
}
|
||
|
||
if (zend_isnan(number)) {
|
||
is_negative = (number<0);
|
||
php_sprintf_appendstring(buffer, pos, "NaN", 3, 0, padding,
|
||
alignment, 3, is_negative, 0, always_sign);
|
||
return;
|
||
}
|
||
|
||
if (zend_isinf(number)) {
|
||
is_negative = (number<0);
|
||
php_sprintf_appendstring(buffer, pos, "INF", 3, 0, padding,
|
||
alignment, 3, is_negative, 0, always_sign);
|
||
return;
|
||
}
|
||
|
||
switch (fmt) {
|
||
case 'e':
|
||
case 'E':
|
||
case 'f':
|
||
case 'F':
|
||
#ifdef HAVE_LOCALE_H
|
||
#ifdef ZTS
|
||
localeconv_r(&lconv);
|
||
#else
|
||
lconv = localeconv();
|
||
#endif
|
||
#endif
|
||
s = php_conv_fp((fmt == 'f')?'F':fmt, number, 0, precision,
|
||
(fmt == 'f')?LCONV_DECIMAL_POINT:'.',
|
||
&is_negative, &num_buf[1], &s_len);
|
||
if (is_negative) {
|
||
num_buf[0] = '-';
|
||
s = num_buf;
|
||
s_len++;
|
||
} else if (always_sign) {
|
||
num_buf[0] = '+';
|
||
s = num_buf;
|
||
s_len++;
|
||
}
|
||
break;
|
||
|
||
case 'g':
|
||
case 'G':
|
||
if (precision == 0)
|
||
precision = 1;
|
||
/*
|
||
* * We use &num_buf[ 1 ], so that we have room for the sign
|
||
*/
|
||
#ifdef HAVE_LOCALE_H
|
||
#ifdef ZTS
|
||
localeconv_r(&lconv);
|
||
#else
|
||
lconv = localeconv();
|
||
#endif
|
||
#endif
|
||
s = php_gcvt(number, precision, LCONV_DECIMAL_POINT, (fmt == 'G')?'E':'e', &num_buf[1]);
|
||
is_negative = 0;
|
||
if (*s == '-') {
|
||
is_negative = 1;
|
||
s = &num_buf[1];
|
||
} else if (always_sign) {
|
||
num_buf[0] = '+';
|
||
s = num_buf;
|
||
}
|
||
|
||
s_len = strlen(s);
|
||
break;
|
||
}
|
||
|
||
php_sprintf_appendstring(buffer, pos, s, width, 0, padding,
|
||
alignment, s_len, is_negative, 0, always_sign);
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_appendd2n() {{{ */
|
||
inline static void
|
||
php_sprintf_append2n(zend_string **buffer, size_t *pos, zend_long number,
|
||
size_t width, char padding, size_t alignment, int n,
|
||
const char *chartable, int expprec)
|
||
{
|
||
char numbuf[NUM_BUF_SIZE];
|
||
register zend_ulong num;
|
||
register zend_ulong i = NUM_BUF_SIZE - 1;
|
||
register int andbits = (1 << n) - 1;
|
||
|
||
PRINTF_DEBUG(("sprintf: append2n(%x, %x, %x, %d, %d, '%c', %d, %d, %x)\n",
|
||
*buffer, pos, &ZSTR_LEN(*buffer), number, width, padding, alignment, n,
|
||
chartable));
|
||
PRINTF_DEBUG(("sprintf: append2n 2^%d andbits=%x\n", n, andbits));
|
||
|
||
num = (zend_ulong) number;
|
||
numbuf[i] = '\0';
|
||
|
||
do {
|
||
numbuf[--i] = chartable[(num & andbits)];
|
||
num >>= n;
|
||
}
|
||
while (num > 0);
|
||
|
||
php_sprintf_appendstring(buffer, pos, &numbuf[i], width, 0,
|
||
padding, alignment, (NUM_BUF_SIZE - 1) - i,
|
||
0, expprec, 0);
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_spintf_getnumber() {{{ */
|
||
inline static int
|
||
php_sprintf_getnumber(char *buffer, size_t *pos)
|
||
{
|
||
char *endptr;
|
||
register zend_long num = ZEND_STRTOL(&buffer[*pos], &endptr, 10);
|
||
register size_t i = 0;
|
||
|
||
if (endptr != NULL) {
|
||
i = (endptr - &buffer[*pos]);
|
||
}
|
||
PRINTF_DEBUG(("sprintf_getnumber: number was %d bytes long\n", i));
|
||
*pos += i;
|
||
|
||
if (num >= INT_MAX || num < 0) {
|
||
return -1;
|
||
} else {
|
||
return (int) num;
|
||
}
|
||
}
|
||
/* }}} */
|
||
|
||
/* php_formatted_print() {{{
|
||
* New sprintf implementation for PHP.
|
||
*
|
||
* Modifiers:
|
||
*
|
||
* " " pad integers with spaces
|
||
* "-" left adjusted field
|
||
* n field size
|
||
* "."n precision (floats only)
|
||
* "+" Always place a sign (+ or -) in front of a number
|
||
*
|
||
* Type specifiers:
|
||
*
|
||
* "%" literal "%", modifiers are ignored.
|
||
* "b" integer argument is printed as binary
|
||
* "c" integer argument is printed as a single character
|
||
* "d" argument is an integer
|
||
* "f" the argument is a float
|
||
* "o" integer argument is printed as octal
|
||
* "s" argument is a string
|
||
* "x" integer argument is printed as lowercase hexadecimal
|
||
* "X" integer argument is printed as uppercase hexadecimal
|
||
*
|
||
*/
|
||
static zend_string *
|
||
php_formatted_print(zend_execute_data *execute_data, int use_array, int format_offset)
|
||
{
|
||
zval *newargs = NULL;
|
||
zval *args, *z_format;
|
||
int argc;
|
||
size_t size = 240, inpos = 0, outpos = 0, temppos;
|
||
int alignment, currarg, adjusting, argnum, width, precision;
|
||
char *format, padding;
|
||
zend_string *result;
|
||
int always_sign;
|
||
size_t format_len;
|
||
|
||
ZEND_PARSE_PARAMETERS_START(1, -1)
|
||
Z_PARAM_VARIADIC('+', args, argc)
|
||
ZEND_PARSE_PARAMETERS_END_EX(return NULL);
|
||
|
||
/* verify the number of args */
|
||
if ((use_array && argc != (2 + format_offset))
|
||
|| (!use_array && argc < (1 + format_offset))) {
|
||
WRONG_PARAM_COUNT_WITH_RETVAL(NULL);
|
||
}
|
||
|
||
convert_to_string_ex(&args[format_offset]);
|
||
if (use_array) {
|
||
int i = 1;
|
||
zval *zv;
|
||
zval *array;
|
||
|
||
z_format = &args[format_offset];
|
||
array = &args[1 + format_offset];
|
||
if (Z_TYPE_P(array) != IS_ARRAY) {
|
||
convert_to_array(array);
|
||
}
|
||
|
||
argc = 1 + zend_hash_num_elements(Z_ARRVAL_P(array));
|
||
newargs = (zval *)safe_emalloc(argc, sizeof(zval), 0);
|
||
ZVAL_COPY_VALUE(&newargs[0], z_format);
|
||
|
||
ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(array), zv) {
|
||
ZVAL_COPY_VALUE(&newargs[i], zv);
|
||
i++;
|
||
} ZEND_HASH_FOREACH_END();
|
||
args = newargs;
|
||
format_offset = 0;
|
||
}
|
||
|
||
format = Z_STRVAL(args[format_offset]);
|
||
format_len = Z_STRLEN(args[format_offset]);
|
||
result = zend_string_alloc(size, 0);
|
||
|
||
currarg = 1;
|
||
|
||
while (inpos < Z_STRLEN(args[format_offset])) {
|
||
int expprec = 0;
|
||
zval *tmp;
|
||
|
||
PRINTF_DEBUG(("sprintf: format[%d]='%c'\n", inpos, format[inpos]));
|
||
PRINTF_DEBUG(("sprintf: outpos=%d\n", outpos));
|
||
if (format[inpos] != '%') {
|
||
php_sprintf_appendchar(&result, &outpos, format[inpos++]);
|
||
} else if (format[inpos + 1] == '%') {
|
||
php_sprintf_appendchar(&result, &outpos, '%');
|
||
inpos += 2;
|
||
} else {
|
||
/* starting a new format specifier, reset variables */
|
||
alignment = ALIGN_RIGHT;
|
||
adjusting = 0;
|
||
padding = ' ';
|
||
always_sign = 0;
|
||
inpos++; /* skip the '%' */
|
||
|
||
PRINTF_DEBUG(("sprintf: first looking at '%c', inpos=%d\n",
|
||
format[inpos], inpos));
|
||
if (isascii((int)format[inpos]) && !isalpha((int)format[inpos])) {
|
||
/* first look for argnum */
|
||
temppos = inpos;
|
||
while (isdigit((int)format[temppos])) temppos++;
|
||
if (format[temppos] == '$') {
|
||
argnum = php_sprintf_getnumber(format, &inpos);
|
||
|
||
if (argnum <= 0) {
|
||
efree(result);
|
||
if (newargs) {
|
||
efree(newargs);
|
||
}
|
||
php_error_docref(NULL, E_WARNING, "Argument number must be greater than zero");
|
||
return NULL;
|
||
}
|
||
|
||
inpos++; /* skip the '$' */
|
||
} else {
|
||
argnum = currarg++;
|
||
}
|
||
|
||
argnum += format_offset;
|
||
|
||
/* after argnum comes modifiers */
|
||
PRINTF_DEBUG(("sprintf: looking for modifiers\n"
|
||
"sprintf: now looking at '%c', inpos=%d\n",
|
||
format[inpos], inpos));
|
||
for (;; inpos++) {
|
||
if (format[inpos] == ' ' || format[inpos] == '0') {
|
||
padding = format[inpos];
|
||
} else if (format[inpos] == '-') {
|
||
alignment = ALIGN_LEFT;
|
||
/* space padding, the default */
|
||
} else if (format[inpos] == '+') {
|
||
always_sign = 1;
|
||
} else if (format[inpos] == '\'' && inpos+1<format_len) {
|
||
padding = format[++inpos];
|
||
} else {
|
||
PRINTF_DEBUG(("sprintf: end of modifiers\n"));
|
||
break;
|
||
}
|
||
}
|
||
PRINTF_DEBUG(("sprintf: padding='%c'\n", padding));
|
||
PRINTF_DEBUG(("sprintf: alignment=%s\n",
|
||
(alignment == ALIGN_LEFT) ? "left" : "right"));
|
||
|
||
|
||
/* after modifiers comes width */
|
||
if (isdigit((int)format[inpos])) {
|
||
PRINTF_DEBUG(("sprintf: getting width\n"));
|
||
if ((width = php_sprintf_getnumber(format, &inpos)) < 0) {
|
||
efree(result);
|
||
if (newargs) {
|
||
efree(newargs);
|
||
}
|
||
php_error_docref(NULL, E_WARNING, "Width must be greater than zero and less than %d", INT_MAX);
|
||
return NULL;
|
||
}
|
||
adjusting |= ADJ_WIDTH;
|
||
} else {
|
||
width = 0;
|
||
}
|
||
PRINTF_DEBUG(("sprintf: width=%d\n", width));
|
||
|
||
/* after width and argnum comes precision */
|
||
if (format[inpos] == '.') {
|
||
inpos++;
|
||
PRINTF_DEBUG(("sprintf: getting precision\n"));
|
||
if (isdigit((int)format[inpos])) {
|
||
if ((precision = php_sprintf_getnumber(format, &inpos)) < 0) {
|
||
efree(result);
|
||
if (newargs) {
|
||
efree(newargs);
|
||
}
|
||
php_error_docref(NULL, E_WARNING, "Precision must be greater than zero and less than %d", INT_MAX);
|
||
return NULL;
|
||
}
|
||
adjusting |= ADJ_PRECISION;
|
||
expprec = 1;
|
||
} else {
|
||
precision = 0;
|
||
}
|
||
} else {
|
||
precision = 0;
|
||
}
|
||
PRINTF_DEBUG(("sprintf: precision=%d\n", precision));
|
||
} else {
|
||
width = precision = 0;
|
||
argnum = currarg++ + format_offset;
|
||
}
|
||
|
||
if (argnum >= argc) {
|
||
efree(result);
|
||
if (newargs) {
|
||
efree(newargs);
|
||
}
|
||
php_error_docref(NULL, E_WARNING, "Too few arguments");
|
||
return NULL;
|
||
}
|
||
|
||
if (format[inpos] == 'l') {
|
||
inpos++;
|
||
}
|
||
PRINTF_DEBUG(("sprintf: format character='%c'\n", format[inpos]));
|
||
/* now we expect to find a type specifier */
|
||
tmp = &args[argnum];
|
||
switch (format[inpos]) {
|
||
case 's': {
|
||
zend_string *t;
|
||
zend_string *str = zval_get_tmp_string(tmp, &t);
|
||
php_sprintf_appendstring(&result, &outpos,
|
||
ZSTR_VAL(str),
|
||
width, precision, padding,
|
||
alignment,
|
||
ZSTR_LEN(str),
|
||
0, expprec, 0);
|
||
zend_tmp_string_release(t);
|
||
break;
|
||
}
|
||
|
||
case 'd':
|
||
php_sprintf_appendint(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment,
|
||
always_sign);
|
||
break;
|
||
|
||
case 'u':
|
||
php_sprintf_appenduint(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment);
|
||
break;
|
||
|
||
case 'g':
|
||
case 'G':
|
||
case 'e':
|
||
case 'E':
|
||
case 'f':
|
||
case 'F':
|
||
php_sprintf_appenddouble(&result, &outpos,
|
||
zval_get_double(tmp),
|
||
width, padding, alignment,
|
||
precision, adjusting,
|
||
format[inpos], always_sign
|
||
);
|
||
break;
|
||
|
||
case 'c':
|
||
php_sprintf_appendchar(&result, &outpos,
|
||
(char) zval_get_long(tmp));
|
||
break;
|
||
|
||
case 'o':
|
||
php_sprintf_append2n(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment, 3,
|
||
hexchars, expprec);
|
||
break;
|
||
|
||
case 'x':
|
||
php_sprintf_append2n(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment, 4,
|
||
hexchars, expprec);
|
||
break;
|
||
|
||
case 'X':
|
||
php_sprintf_append2n(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment, 4,
|
||
HEXCHARS, expprec);
|
||
break;
|
||
|
||
case 'b':
|
||
php_sprintf_append2n(&result, &outpos,
|
||
zval_get_long(tmp),
|
||
width, padding, alignment, 1,
|
||
hexchars, expprec);
|
||
break;
|
||
|
||
case '%':
|
||
php_sprintf_appendchar(&result, &outpos, '%');
|
||
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
inpos++;
|
||
}
|
||
}
|
||
|
||
if (newargs) {
|
||
efree(newargs);
|
||
}
|
||
|
||
/* possibly, we have to make sure we have room for the terminating null? */
|
||
ZSTR_VAL(result)[outpos]=0;
|
||
ZSTR_LEN(result) = outpos;
|
||
return result;
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto string sprintf(string format [, mixed arg1 [, mixed ...]])
|
||
Return a formatted string */
|
||
PHP_FUNCTION(user_sprintf)
|
||
{
|
||
zend_string *result;
|
||
|
||
if ((result=php_formatted_print(execute_data, 0, 0))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
RETVAL_STR(result);
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto string vsprintf(string format, array args)
|
||
Return a formatted string */
|
||
PHP_FUNCTION(vsprintf)
|
||
{
|
||
zend_string *result;
|
||
|
||
if ((result=php_formatted_print(execute_data, 1, 0))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
RETVAL_STR(result);
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto int printf(string format [, mixed arg1 [, mixed ...]])
|
||
Output a formatted string */
|
||
PHP_FUNCTION(user_printf)
|
||
{
|
||
zend_string *result;
|
||
size_t rlen;
|
||
|
||
if ((result=php_formatted_print(execute_data, 0, 0))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
rlen = PHPWRITE(ZSTR_VAL(result), ZSTR_LEN(result));
|
||
zend_string_free(result);
|
||
RETURN_LONG(rlen);
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto int vprintf(string format, array args)
|
||
Output a formatted string */
|
||
PHP_FUNCTION(vprintf)
|
||
{
|
||
zend_string *result;
|
||
size_t rlen;
|
||
|
||
if ((result=php_formatted_print(execute_data, 1, 0))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
rlen = PHPWRITE(ZSTR_VAL(result), ZSTR_LEN(result));
|
||
zend_string_free(result);
|
||
RETURN_LONG(rlen);
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto int fprintf(resource stream, string format [, mixed arg1 [, mixed ...]])
|
||
Output a formatted string into a stream */
|
||
PHP_FUNCTION(fprintf)
|
||
{
|
||
php_stream *stream;
|
||
zval *arg1;
|
||
zend_string *result;
|
||
|
||
if (ZEND_NUM_ARGS() < 2) {
|
||
WRONG_PARAM_COUNT;
|
||
}
|
||
|
||
ZEND_PARSE_PARAMETERS_START(1, -1)
|
||
Z_PARAM_RESOURCE(arg1)
|
||
/* php_formatted_print does its own zpp for extra args */
|
||
ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
|
||
|
||
php_stream_from_zval(stream, arg1);
|
||
|
||
if ((result=php_formatted_print(execute_data, 0, 1))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
|
||
php_stream_write(stream, ZSTR_VAL(result), ZSTR_LEN(result));
|
||
|
||
RETVAL_LONG(ZSTR_LEN(result));
|
||
zend_string_free(result);
|
||
}
|
||
/* }}} */
|
||
|
||
/* {{{ proto int vfprintf(resource stream, string format, array args)
|
||
Output a formatted string into a stream */
|
||
PHP_FUNCTION(vfprintf)
|
||
{
|
||
php_stream *stream;
|
||
zval *arg1;
|
||
zend_string *result;
|
||
|
||
if (ZEND_NUM_ARGS() != 3) {
|
||
WRONG_PARAM_COUNT;
|
||
}
|
||
|
||
ZEND_PARSE_PARAMETERS_START(1, -1)
|
||
Z_PARAM_RESOURCE(arg1)
|
||
/* php_formatted_print does its own zpp for extra args */
|
||
ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
|
||
|
||
php_stream_from_zval(stream, arg1);
|
||
|
||
if ((result=php_formatted_print(execute_data, 1, 1))==NULL) {
|
||
RETURN_FALSE;
|
||
}
|
||
|
||
php_stream_write(stream, ZSTR_VAL(result), ZSTR_LEN(result));
|
||
|
||
RETVAL_LONG(ZSTR_LEN(result));
|
||
zend_string_free(result);
|
||
}
|
||
/* }}} */
|
||
|
||
/*
|
||
* Local variables:
|
||
* tab-width: 4
|
||
* c-basic-offset: 4
|
||
* End:
|
||
* vim600: sw=4 ts=4 fdm=marker
|
||
* vim<600: sw=4 ts=4
|
||
*/
|