diff options
Diffstat (limited to 'ext/standard/string.c')
| -rw-r--r-- | ext/standard/string.c | 95 |
1 files changed, 70 insertions, 25 deletions
diff --git a/ext/standard/string.c b/ext/standard/string.c index f6010e71dc..34dc4cf6e4 100644 --- a/ext/standard/string.c +++ b/ext/standard/string.c @@ -32,6 +32,7 @@ #endif #ifdef HAVE_ICONV # include <iconv.h> +# include "SAPI.h" #endif #include "scanf.h" #include "zend_API.h" @@ -71,6 +72,7 @@ void register_string_constants(INIT_FUNC_ARGS) } int php_tag_find(char *tag, int len, char *set); +int php_iconv_string(char *, char **, char *, char *); /* this is read-only, so it's ok */ static char hexconvtab[] = "0123456789abcdef"; @@ -3003,7 +3005,44 @@ PHP_FUNCTION(sscanf) /* }}} */ #ifdef HAVE_ICONV -/* {{{ proto iconv(string in_charset, string out_charset, string str) + +int php_iconv_string(char *in_p, char **out, char *in_charset, char *out_charset) { + unsigned int in_size, out_size; + char *out_buffer, *out_p; + iconv_t cd; + size_t result; + typedef unsigned int ucs4_t; + + in_size = strlen(in_p) * sizeof(char) + 1; + out_size = strlen(in_p) * sizeof(ucs4_t) + 1; + + out_buffer = (char *) emalloc(out_size); + *out = out_buffer; + out_p = out_buffer; + + cd = iconv_open(out_charset, in_charset); + + if (cd == (iconv_t)(-1)) { + php_error(E_WARNING, "iconv: cannot convert from `%s' to `%s'", + in_charset, out_charset); + efree(out_buffer); + return -1; + } + + result = iconv(cd, (const char **) &in_p, &in_size, (char **) + &out_p, &out_size); + + if (result == (size_t)(-1)) { + sprintf(out_buffer, "???") ; + return -1; + } + + iconv_close(cd); + + return SUCCESS; +} + +/* {{{ proto string iconv(string in_charset, string out_charset, string str) Returns str converted to the out_charset character set. */ PHP_FUNCTION(iconv) @@ -3023,34 +3062,40 @@ PHP_FUNCTION(iconv) convert_to_string_ex(out_charset); convert_to_string_ex(in_buffer); - in_size = Z_STRLEN_PP(in_buffer) * sizeof(char) + 1; - out_size = Z_STRLEN_PP(in_buffer) * sizeof(ucs4_t) + 1; - - out_buffer = (char *) emalloc(out_size); - - in_p = Z_STRVAL_PP(in_buffer); - out_p = out_buffer; - - cd = iconv_open(Z_STRVAL_PP(out_charset), Z_STRVAL_PP(in_charset)); - - if (cd == (iconv_t)(-1)) { - php_error(E_WARNING, "iconv: cannot convert from `%s' to `%s'", - Z_STRVAL_PP(in_charset), Z_STRVAL_PP(out_charset)); - efree(out_buffer); - RETURN_FALSE; - } - - result = iconv(cd, (const char **) &in_p, &in_size, (char **) - &out_p, &out_size); + if (php_iconv_string(Z_STRVAL_PP(in_buffer), &out_buffer, + Z_STRVAL_PP(in_charset), Z_STRVAL_PP(out_charset)) == SUCCESS) { + RETVAL_STRING(out_buffer, 0); + } else { + RETURN_FALSE; + } +} - if (result == (size_t)(-1)) { - sprintf(out_buffer, "???") ; - } +/* {{{ proto string ob_iconv_handler(string contents) + Returns str in output buffer converted to the iconv.output_encoding + character set. +*/ +PHP_FUNCTION(ob_iconv_handler) +{ + int coding; + char *out_buffer; + zval **zv_string; - iconv_close(cd); + if (ZEND_NUM_ARGS()!=1 || zend_get_parameters_ex(1, &zv_string)==FAILURE) { + ZEND_WRONG_PARAM_COUNT(); + } - RETVAL_STRING(out_buffer, 0); + if (php_iconv_string(Z_STRVAL_PP(zv_string), &out_buffer, + BG(iconv_internal_encoding), + BG(iconv_output_encoding))==SUCCESS) { + RETVAL_STRING(out_buffer, 0); + } else { + zval_dtor(return_value); + *return_value = **zv_string; + zval_copy_ctor(return_value); + } + } + #endif /* |
