summaryrefslogtreecommitdiff
path: root/ext/standard/string.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/standard/string.c')
-rw-r--r--ext/standard/string.c95
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
/*