PageRenderTime 56ms CodeModel.GetById 23ms RepoModel.GetById 1ms app.codeStats 0ms

/ext/iconv/iconv.c

https://github.com/php/php-src
C | 2655 lines | 2089 code | 421 blank | 145 comment | 604 complexity | 523f95cd2722660a696b0842b66fd428 MD5 | raw file
Possible License(s): BSD-2-Clause, BSD-3-Clause, MPL-2.0-no-copyleft-exception, LGPL-2.1

Large files files are truncated, but you can click here to view the full file

  1. /*
  2. +----------------------------------------------------------------------+
  3. | Copyright (c) The PHP Group |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | https://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Rui Hirokawa <rui_hirokawa@ybb.ne.jp> |
  14. | Stig Bakken <ssb@php.net> |
  15. | Moriyoshi Koizumi <moriyoshi@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #ifdef HAVE_CONFIG_H
  19. #include "config.h"
  20. #endif
  21. #include "php.h"
  22. #include "php_globals.h"
  23. #include "ext/standard/info.h"
  24. #include "main/php_output.h"
  25. #include "SAPI.h"
  26. #include "php_ini.h"
  27. #include <stdlib.h>
  28. #include <errno.h>
  29. #include "php_iconv.h"
  30. #ifdef HAVE_ICONV
  31. #include <iconv.h>
  32. #ifdef HAVE_GLIBC_ICONV
  33. #include <gnu/libc-version.h>
  34. #endif
  35. #ifdef HAVE_LIBICONV
  36. #undef iconv
  37. #endif
  38. #include "zend_smart_str.h"
  39. #include "ext/standard/base64.h"
  40. #include "ext/standard/quot_print.h"
  41. #include "iconv_arginfo.h"
  42. #define _php_iconv_memequal(a, b, c) \
  43. (memcmp(a, b, c) == 0)
  44. ZEND_DECLARE_MODULE_GLOBALS(iconv)
  45. static PHP_GINIT_FUNCTION(iconv);
  46. /* {{{ iconv_module_entry */
  47. zend_module_entry iconv_module_entry = {
  48. STANDARD_MODULE_HEADER,
  49. "iconv",
  50. ext_functions,
  51. PHP_MINIT(miconv),
  52. PHP_MSHUTDOWN(miconv),
  53. NULL,
  54. NULL,
  55. PHP_MINFO(miconv),
  56. PHP_ICONV_VERSION,
  57. PHP_MODULE_GLOBALS(iconv),
  58. PHP_GINIT(iconv),
  59. NULL,
  60. NULL,
  61. STANDARD_MODULE_PROPERTIES_EX
  62. };
  63. /* }}} */
  64. #ifdef COMPILE_DL_ICONV
  65. #ifdef ZTS
  66. ZEND_TSRMLS_CACHE_DEFINE()
  67. #endif
  68. ZEND_GET_MODULE(iconv)
  69. #endif
  70. /* {{{ PHP_GINIT_FUNCTION */
  71. static PHP_GINIT_FUNCTION(iconv)
  72. {
  73. #if defined(COMPILE_DL_ICONV) && defined(ZTS)
  74. ZEND_TSRMLS_CACHE_UPDATE();
  75. #endif
  76. iconv_globals->input_encoding = NULL;
  77. iconv_globals->output_encoding = NULL;
  78. iconv_globals->internal_encoding = NULL;
  79. }
  80. /* }}} */
  81. #if defined(HAVE_LIBICONV) && defined(ICONV_ALIASED_LIBICONV)
  82. #define iconv libiconv
  83. #endif
  84. /* {{{ typedef enum php_iconv_enc_scheme_t */
  85. typedef enum _php_iconv_enc_scheme_t {
  86. PHP_ICONV_ENC_SCHEME_BASE64,
  87. PHP_ICONV_ENC_SCHEME_QPRINT
  88. } php_iconv_enc_scheme_t;
  89. /* }}} */
  90. #define PHP_ICONV_MIME_DECODE_STRICT (1<<0)
  91. #define PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR (1<<1)
  92. /* {{{ prototypes */
  93. static php_iconv_err_t _php_iconv_appendl(smart_str *d, const char *s, size_t l, iconv_t cd);
  94. static php_iconv_err_t _php_iconv_appendc(smart_str *d, const char c, iconv_t cd);
  95. static void _php_iconv_show_error(php_iconv_err_t err, const char *out_charset, const char *in_charset);
  96. static php_iconv_err_t _php_iconv_strlen(size_t *pretval, const char *str, size_t nbytes, const char *enc);
  97. static php_iconv_err_t _php_iconv_substr(smart_str *pretval, const char *str, size_t nbytes, zend_long offset, zend_long len, const char *enc);
  98. static php_iconv_err_t _php_iconv_mime_encode(smart_str *pretval, const char *fname, size_t fname_nbytes, const char *fval, size_t fval_nbytes, size_t max_line_len, const char *lfchars, php_iconv_enc_scheme_t enc_scheme, const char *out_charset, const char *enc);
  99. static php_iconv_err_t _php_iconv_mime_decode(smart_str *pretval, const char *str, size_t str_nbytes, const char *enc, const char **next_pos, int mode);
  100. static php_iconv_err_t php_iconv_stream_filter_register_factory(void);
  101. static php_iconv_err_t php_iconv_stream_filter_unregister_factory(void);
  102. static int php_iconv_output_conflict(const char *handler_name, size_t handler_name_len);
  103. static php_output_handler *php_iconv_output_handler_init(const char *name, size_t name_len, size_t chunk_size, int flags);
  104. static int php_iconv_output_handler(void **nothing, php_output_context *output_context);
  105. /* }}} */
  106. /* {{{ static globals */
  107. static const char _generic_superset_name[] = ICONV_UCS4_ENCODING;
  108. #define GENERIC_SUPERSET_NAME _generic_superset_name
  109. #define GENERIC_SUPERSET_NBYTES 4
  110. /* }}} */
  111. static PHP_INI_MH(OnUpdateInputEncoding)
  112. {
  113. if (ZSTR_LEN(new_value) >= ICONV_CSNMAXLEN) {
  114. return FAILURE;
  115. }
  116. if (stage & (PHP_INI_STAGE_ACTIVATE | PHP_INI_STAGE_RUNTIME)) {
  117. php_error_docref("ref.iconv", E_DEPRECATED, "Use of iconv.input_encoding is deprecated");
  118. }
  119. OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  120. return SUCCESS;
  121. }
  122. static PHP_INI_MH(OnUpdateOutputEncoding)
  123. {
  124. if (ZSTR_LEN(new_value) >= ICONV_CSNMAXLEN) {
  125. return FAILURE;
  126. }
  127. if (stage & (PHP_INI_STAGE_ACTIVATE | PHP_INI_STAGE_RUNTIME)) {
  128. php_error_docref("ref.iconv", E_DEPRECATED, "Use of iconv.output_encoding is deprecated");
  129. }
  130. OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  131. return SUCCESS;
  132. }
  133. static PHP_INI_MH(OnUpdateInternalEncoding)
  134. {
  135. if (ZSTR_LEN(new_value) >= ICONV_CSNMAXLEN) {
  136. return FAILURE;
  137. }
  138. if (stage & (PHP_INI_STAGE_ACTIVATE | PHP_INI_STAGE_RUNTIME)) {
  139. php_error_docref("ref.iconv", E_DEPRECATED, "Use of iconv.internal_encoding is deprecated");
  140. }
  141. OnUpdateString(entry, new_value, mh_arg1, mh_arg2, mh_arg3, stage);
  142. return SUCCESS;
  143. }
  144. /* {{{ PHP_INI */
  145. PHP_INI_BEGIN()
  146. STD_PHP_INI_ENTRY("iconv.input_encoding", "", PHP_INI_ALL, OnUpdateInputEncoding, input_encoding, zend_iconv_globals, iconv_globals)
  147. STD_PHP_INI_ENTRY("iconv.output_encoding", "", PHP_INI_ALL, OnUpdateOutputEncoding, output_encoding, zend_iconv_globals, iconv_globals)
  148. STD_PHP_INI_ENTRY("iconv.internal_encoding", "", PHP_INI_ALL, OnUpdateInternalEncoding, internal_encoding, zend_iconv_globals, iconv_globals)
  149. PHP_INI_END()
  150. /* }}} */
  151. /* {{{ PHP_MINIT_FUNCTION */
  152. PHP_MINIT_FUNCTION(miconv)
  153. {
  154. char *version = "unknown";
  155. REGISTER_INI_ENTRIES();
  156. #ifdef HAVE_LIBICONV
  157. {
  158. static char buf[16];
  159. snprintf(buf, sizeof(buf), "%d.%d",
  160. _libiconv_version >> 8, _libiconv_version & 0xff);
  161. version = buf;
  162. }
  163. #elif HAVE_GLIBC_ICONV
  164. version = (char *)gnu_get_libc_version();
  165. #endif
  166. #ifdef PHP_ICONV_IMPL
  167. REGISTER_STRING_CONSTANT("ICONV_IMPL", PHP_ICONV_IMPL, CONST_CS | CONST_PERSISTENT);
  168. #elif HAVE_LIBICONV
  169. REGISTER_STRING_CONSTANT("ICONV_IMPL", "libiconv", CONST_CS | CONST_PERSISTENT);
  170. #else
  171. REGISTER_STRING_CONSTANT("ICONV_IMPL", "unknown", CONST_CS | CONST_PERSISTENT);
  172. #endif
  173. REGISTER_STRING_CONSTANT("ICONV_VERSION", version, CONST_CS | CONST_PERSISTENT);
  174. REGISTER_LONG_CONSTANT("ICONV_MIME_DECODE_STRICT", PHP_ICONV_MIME_DECODE_STRICT, CONST_CS | CONST_PERSISTENT);
  175. REGISTER_LONG_CONSTANT("ICONV_MIME_DECODE_CONTINUE_ON_ERROR", PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR, CONST_CS | CONST_PERSISTENT);
  176. if (php_iconv_stream_filter_register_factory() != PHP_ICONV_ERR_SUCCESS) {
  177. return FAILURE;
  178. }
  179. php_output_handler_alias_register(ZEND_STRL("ob_iconv_handler"), php_iconv_output_handler_init);
  180. php_output_handler_conflict_register(ZEND_STRL("ob_iconv_handler"), php_iconv_output_conflict);
  181. return SUCCESS;
  182. }
  183. /* }}} */
  184. /* {{{ PHP_MSHUTDOWN_FUNCTION */
  185. PHP_MSHUTDOWN_FUNCTION(miconv)
  186. {
  187. php_iconv_stream_filter_unregister_factory();
  188. UNREGISTER_INI_ENTRIES();
  189. return SUCCESS;
  190. }
  191. /* }}} */
  192. /* {{{ PHP_MINFO_FUNCTION */
  193. PHP_MINFO_FUNCTION(miconv)
  194. {
  195. zval *iconv_impl, *iconv_ver;
  196. iconv_impl = zend_get_constant_str("ICONV_IMPL", sizeof("ICONV_IMPL")-1);
  197. iconv_ver = zend_get_constant_str("ICONV_VERSION", sizeof("ICONV_VERSION")-1);
  198. php_info_print_table_start();
  199. php_info_print_table_row(2, "iconv support", "enabled");
  200. php_info_print_table_row(2, "iconv implementation", Z_STRVAL_P(iconv_impl));
  201. php_info_print_table_row(2, "iconv library version", Z_STRVAL_P(iconv_ver));
  202. php_info_print_table_end();
  203. DISPLAY_INI_ENTRIES();
  204. }
  205. /* }}} */
  206. static const char *get_internal_encoding(void) {
  207. if (ICONVG(internal_encoding) && ICONVG(internal_encoding)[0]) {
  208. return ICONVG(internal_encoding);
  209. }
  210. return php_get_internal_encoding();
  211. }
  212. static const char *get_input_encoding(void) {
  213. if (ICONVG(input_encoding) && ICONVG(input_encoding)[0]) {
  214. return ICONVG(input_encoding);
  215. }
  216. return php_get_input_encoding();
  217. }
  218. static const char *get_output_encoding(void) {
  219. if (ICONVG(output_encoding) && ICONVG(output_encoding)[0]) {
  220. return ICONVG(output_encoding);
  221. }
  222. return php_get_output_encoding();
  223. }
  224. static int php_iconv_output_conflict(const char *handler_name, size_t handler_name_len)
  225. {
  226. if (php_output_get_level()) {
  227. if (php_output_handler_conflict(handler_name, handler_name_len, ZEND_STRL("ob_iconv_handler"))
  228. || php_output_handler_conflict(handler_name, handler_name_len, ZEND_STRL("mb_output_handler"))) {
  229. return FAILURE;
  230. }
  231. }
  232. return SUCCESS;
  233. }
  234. static php_output_handler *php_iconv_output_handler_init(const char *handler_name, size_t handler_name_len, size_t chunk_size, int flags)
  235. {
  236. return php_output_handler_create_internal(handler_name, handler_name_len, php_iconv_output_handler, chunk_size, flags);
  237. }
  238. static int php_iconv_output_handler(void **nothing, php_output_context *output_context)
  239. {
  240. char *s, *content_type, *mimetype = NULL;
  241. int output_status, mimetype_len = 0;
  242. if (output_context->op & PHP_OUTPUT_HANDLER_START) {
  243. output_status = php_output_get_status();
  244. if (output_status & PHP_OUTPUT_SENT) {
  245. return FAILURE;
  246. }
  247. if (SG(sapi_headers).mimetype && !strncasecmp(SG(sapi_headers).mimetype, "text/", 5)) {
  248. if ((s = strchr(SG(sapi_headers).mimetype,';')) == NULL){
  249. mimetype = SG(sapi_headers).mimetype;
  250. } else {
  251. mimetype = SG(sapi_headers).mimetype;
  252. mimetype_len = (int)(s - SG(sapi_headers).mimetype);
  253. }
  254. } else if (SG(sapi_headers).send_default_content_type) {
  255. mimetype = SG(default_mimetype) ? SG(default_mimetype) : SAPI_DEFAULT_MIMETYPE;
  256. }
  257. if (mimetype != NULL && !(output_context->op & PHP_OUTPUT_HANDLER_CLEAN)) {
  258. size_t len;
  259. char *p = strstr(get_output_encoding(), "//");
  260. if (p) {
  261. len = spprintf(&content_type, 0, "Content-Type:%.*s; charset=%.*s", mimetype_len ? mimetype_len : (int) strlen(mimetype), mimetype, (int) (p - get_output_encoding()), get_output_encoding());
  262. } else {
  263. len = spprintf(&content_type, 0, "Content-Type:%.*s; charset=%s", mimetype_len ? mimetype_len : (int) strlen(mimetype), mimetype, get_output_encoding());
  264. }
  265. if (content_type && SUCCESS == sapi_add_header(content_type, len, 0)) {
  266. SG(sapi_headers).send_default_content_type = 0;
  267. php_output_handler_hook(PHP_OUTPUT_HANDLER_HOOK_IMMUTABLE, NULL);
  268. }
  269. }
  270. }
  271. if (output_context->in.used) {
  272. zend_string *out;
  273. output_context->out.free = 1;
  274. _php_iconv_show_error(php_iconv_string(output_context->in.data, output_context->in.used, &out, get_output_encoding(), get_internal_encoding()), get_output_encoding(), get_internal_encoding());
  275. if (out) {
  276. output_context->out.data = estrndup(ZSTR_VAL(out), ZSTR_LEN(out));
  277. output_context->out.used = ZSTR_LEN(out);
  278. zend_string_efree(out);
  279. } else {
  280. output_context->out.data = NULL;
  281. output_context->out.used = 0;
  282. }
  283. }
  284. return SUCCESS;
  285. }
  286. /* {{{ _php_iconv_appendl() */
  287. static php_iconv_err_t _php_iconv_appendl(smart_str *d, const char *s, size_t l, iconv_t cd)
  288. {
  289. const char *in_p = s;
  290. size_t in_left = l;
  291. char *out_p;
  292. size_t out_left = 0;
  293. size_t buf_growth = 128;
  294. if (in_p != NULL) {
  295. while (in_left > 0) {
  296. out_left = buf_growth;
  297. smart_str_alloc(d, out_left, 0);
  298. out_p = ZSTR_VAL((d)->s) + ZSTR_LEN((d)->s);
  299. if (iconv(cd, (char **)&in_p, &in_left, (char **) &out_p, &out_left) == (size_t)-1) {
  300. switch (errno) {
  301. case EINVAL:
  302. return PHP_ICONV_ERR_ILLEGAL_CHAR;
  303. case EILSEQ:
  304. return PHP_ICONV_ERR_ILLEGAL_SEQ;
  305. case E2BIG:
  306. break;
  307. default:
  308. return PHP_ICONV_ERR_UNKNOWN;
  309. }
  310. }
  311. ZSTR_LEN((d)->s) += (buf_growth - out_left);
  312. buf_growth <<= 1;
  313. }
  314. } else {
  315. for (;;) {
  316. out_left = buf_growth;
  317. smart_str_alloc(d, out_left, 0);
  318. out_p = ZSTR_VAL((d)->s) + ZSTR_LEN((d)->s);
  319. if (iconv(cd, NULL, NULL, (char **) &out_p, &out_left) == (size_t)0) {
  320. ZSTR_LEN((d)->s) += (buf_growth - out_left);
  321. break;
  322. } else {
  323. if (errno != E2BIG) {
  324. return PHP_ICONV_ERR_UNKNOWN;
  325. }
  326. }
  327. ZSTR_LEN((d)->s) += (buf_growth - out_left);
  328. buf_growth <<= 1;
  329. }
  330. }
  331. return PHP_ICONV_ERR_SUCCESS;
  332. }
  333. /* }}} */
  334. /* {{{ _php_iconv_appendc() */
  335. static php_iconv_err_t _php_iconv_appendc(smart_str *d, const char c, iconv_t cd)
  336. {
  337. return _php_iconv_appendl(d, &c, 1, cd);
  338. }
  339. /* }}} */
  340. /* {{{ */
  341. #ifdef ICONV_BROKEN_IGNORE
  342. static int _php_check_ignore(const char *charset)
  343. {
  344. size_t clen = strlen(charset);
  345. if (clen >= 9 && strcmp("//IGNORE", charset+clen-8) == 0) {
  346. return 1;
  347. }
  348. if (clen >= 19 && strcmp("//IGNORE//TRANSLIT", charset+clen-18) == 0) {
  349. return 1;
  350. }
  351. return 0;
  352. }
  353. #else
  354. #define _php_check_ignore(x) (0)
  355. #endif
  356. /* }}} */
  357. /* {{{ php_iconv_string() */
  358. PHP_ICONV_API php_iconv_err_t php_iconv_string(const char *in_p, size_t in_len, zend_string **out, const char *out_charset, const char *in_charset)
  359. {
  360. iconv_t cd;
  361. size_t in_left, out_size, out_left;
  362. char *out_p;
  363. size_t bsz, result = 0;
  364. php_iconv_err_t retval = PHP_ICONV_ERR_SUCCESS;
  365. zend_string *out_buf;
  366. int ignore_ilseq = _php_check_ignore(out_charset);
  367. *out = NULL;
  368. cd = iconv_open(out_charset, in_charset);
  369. if (cd == (iconv_t)(-1)) {
  370. if (errno == EINVAL) {
  371. return PHP_ICONV_ERR_WRONG_CHARSET;
  372. } else {
  373. return PHP_ICONV_ERR_CONVERTER;
  374. }
  375. }
  376. in_left= in_len;
  377. out_left = in_len + 32; /* Avoid realloc() most cases */
  378. out_size = 0;
  379. bsz = out_left;
  380. out_buf = zend_string_alloc(bsz, 0);
  381. out_p = ZSTR_VAL(out_buf);
  382. while (in_left > 0) {
  383. result = iconv(cd, (char **) &in_p, &in_left, (char **) &out_p, &out_left);
  384. out_size = bsz - out_left;
  385. if (result == (size_t)(-1)) {
  386. if (ignore_ilseq && errno == EILSEQ) {
  387. if (in_left <= 1) {
  388. result = 0;
  389. } else {
  390. errno = 0;
  391. in_p++;
  392. in_left--;
  393. continue;
  394. }
  395. }
  396. if (errno == E2BIG && in_left > 0) {
  397. /* converted string is longer than out buffer */
  398. bsz += in_len;
  399. out_buf = zend_string_extend(out_buf, bsz, 0);
  400. out_p = ZSTR_VAL(out_buf);
  401. out_p += out_size;
  402. out_left = bsz - out_size;
  403. continue;
  404. }
  405. }
  406. break;
  407. }
  408. if (result != (size_t)(-1)) {
  409. /* flush the shift-out sequences */
  410. for (;;) {
  411. result = iconv(cd, NULL, NULL, (char **) &out_p, &out_left);
  412. out_size = bsz - out_left;
  413. if (result != (size_t)(-1)) {
  414. break;
  415. }
  416. if (errno == E2BIG) {
  417. bsz += 16;
  418. out_buf = zend_string_extend(out_buf, bsz, 0);
  419. out_p = ZSTR_VAL(out_buf);
  420. out_p += out_size;
  421. out_left = bsz - out_size;
  422. } else {
  423. break;
  424. }
  425. }
  426. }
  427. iconv_close(cd);
  428. if (result == (size_t)(-1)) {
  429. switch (errno) {
  430. case EINVAL:
  431. retval = PHP_ICONV_ERR_ILLEGAL_CHAR;
  432. break;
  433. case EILSEQ:
  434. retval = PHP_ICONV_ERR_ILLEGAL_SEQ;
  435. break;
  436. case E2BIG:
  437. /* should not happen */
  438. retval = PHP_ICONV_ERR_TOO_BIG;
  439. break;
  440. default:
  441. /* other error */
  442. zend_string_efree(out_buf);
  443. return PHP_ICONV_ERR_UNKNOWN;
  444. }
  445. }
  446. *out_p = '\0';
  447. ZSTR_LEN(out_buf) = out_size;
  448. *out = out_buf;
  449. return retval;
  450. }
  451. /* }}} */
  452. /* {{{ _php_iconv_strlen() */
  453. static php_iconv_err_t _php_iconv_strlen(size_t *pretval, const char *str, size_t nbytes, const char *enc)
  454. {
  455. char buf[GENERIC_SUPERSET_NBYTES*2];
  456. php_iconv_err_t err = PHP_ICONV_ERR_SUCCESS;
  457. iconv_t cd;
  458. const char *in_p;
  459. size_t in_left;
  460. char *out_p;
  461. size_t out_left;
  462. size_t cnt;
  463. int more;
  464. *pretval = (size_t)-1;
  465. cd = iconv_open(GENERIC_SUPERSET_NAME, enc);
  466. if (cd == (iconv_t)(-1)) {
  467. if (errno == EINVAL) {
  468. return PHP_ICONV_ERR_WRONG_CHARSET;
  469. } else {
  470. return PHP_ICONV_ERR_CONVERTER;
  471. }
  472. }
  473. errno = 0;
  474. out_left = 0;
  475. more = nbytes > 0;
  476. for (in_p = str, in_left = nbytes, cnt = 0; more;) {
  477. out_p = buf;
  478. out_left = sizeof(buf);
  479. more = in_left > 0;
  480. iconv(cd, more ? (char **)&in_p : NULL, more ? &in_left : NULL, (char **) &out_p, &out_left);
  481. if (out_left == sizeof(buf)) {
  482. break;
  483. } else {
  484. ZEND_ASSERT((sizeof(buf) - out_left) % GENERIC_SUPERSET_NBYTES == 0);
  485. cnt += (sizeof(buf) - out_left) / GENERIC_SUPERSET_NBYTES;
  486. }
  487. }
  488. switch (errno) {
  489. case EINVAL:
  490. err = PHP_ICONV_ERR_ILLEGAL_CHAR;
  491. break;
  492. case EILSEQ:
  493. err = PHP_ICONV_ERR_ILLEGAL_SEQ;
  494. break;
  495. case E2BIG:
  496. case 0:
  497. *pretval = cnt;
  498. break;
  499. default:
  500. err = PHP_ICONV_ERR_UNKNOWN;
  501. break;
  502. }
  503. iconv_close(cd);
  504. return err;
  505. }
  506. /* }}} */
  507. /* {{{ _php_iconv_substr() */
  508. static php_iconv_err_t _php_iconv_substr(smart_str *pretval,
  509. const char *str, size_t nbytes, zend_long offset, zend_long len, const char *enc)
  510. {
  511. char buf[GENERIC_SUPERSET_NBYTES];
  512. php_iconv_err_t err = PHP_ICONV_ERR_SUCCESS;
  513. iconv_t cd1, cd2;
  514. const char *in_p;
  515. size_t in_left;
  516. char *out_p;
  517. size_t out_left;
  518. size_t cnt;
  519. size_t total_len;
  520. int more;
  521. err = _php_iconv_strlen(&total_len, str, nbytes, enc);
  522. if (err != PHP_ICONV_ERR_SUCCESS) {
  523. return err;
  524. }
  525. if (offset < 0) {
  526. if ((offset += total_len) < 0) {
  527. offset = 0;
  528. }
  529. } else if ((size_t)offset > total_len) {
  530. offset = total_len;
  531. }
  532. if (len < 0) {
  533. if ((len += (total_len - offset)) < 0) {
  534. len = 0;
  535. }
  536. } else if ((size_t)len > total_len) {
  537. len = total_len;
  538. }
  539. if ((size_t)(offset + len) > total_len ) {
  540. /* trying to compute the length */
  541. len = total_len - offset;
  542. }
  543. if (len == 0) {
  544. smart_str_appendl(pretval, "", 0);
  545. smart_str_0(pretval);
  546. return PHP_ICONV_ERR_SUCCESS;
  547. }
  548. cd1 = iconv_open(GENERIC_SUPERSET_NAME, enc);
  549. if (cd1 == (iconv_t)(-1)) {
  550. if (errno == EINVAL) {
  551. return PHP_ICONV_ERR_WRONG_CHARSET;
  552. } else {
  553. return PHP_ICONV_ERR_CONVERTER;
  554. }
  555. }
  556. cd2 = (iconv_t)NULL;
  557. errno = 0;
  558. more = nbytes > 0 && len > 0;
  559. for (in_p = str, in_left = nbytes, cnt = 0; more; ++cnt) {
  560. out_p = buf;
  561. out_left = sizeof(buf);
  562. more = in_left > 0 && len > 0;
  563. iconv(cd1, more ? (char **)&in_p : NULL, more ? &in_left : NULL, (char **) &out_p, &out_left);
  564. if (out_left == sizeof(buf)) {
  565. break;
  566. }
  567. if ((zend_long)cnt >= offset) {
  568. if (cd2 == (iconv_t)NULL) {
  569. cd2 = iconv_open(enc, GENERIC_SUPERSET_NAME);
  570. if (cd2 == (iconv_t)(-1)) {
  571. cd2 = (iconv_t)NULL;
  572. if (errno == EINVAL) {
  573. err = PHP_ICONV_ERR_WRONG_CHARSET;
  574. } else {
  575. err = PHP_ICONV_ERR_CONVERTER;
  576. }
  577. break;
  578. }
  579. }
  580. if (_php_iconv_appendl(pretval, buf, sizeof(buf), cd2) != PHP_ICONV_ERR_SUCCESS) {
  581. break;
  582. }
  583. --len;
  584. }
  585. }
  586. switch (errno) {
  587. case EINVAL:
  588. err = PHP_ICONV_ERR_ILLEGAL_CHAR;
  589. break;
  590. case EILSEQ:
  591. err = PHP_ICONV_ERR_ILLEGAL_SEQ;
  592. break;
  593. case E2BIG:
  594. break;
  595. }
  596. if (err == PHP_ICONV_ERR_SUCCESS) {
  597. if (cd2 != (iconv_t)NULL) {
  598. _php_iconv_appendl(pretval, NULL, 0, cd2);
  599. }
  600. smart_str_0(pretval);
  601. }
  602. if (cd1 != (iconv_t)NULL) {
  603. iconv_close(cd1);
  604. }
  605. if (cd2 != (iconv_t)NULL) {
  606. iconv_close(cd2);
  607. }
  608. return err;
  609. }
  610. /* }}} */
  611. /* {{{ _php_iconv_strpos() */
  612. static php_iconv_err_t _php_iconv_strpos(size_t *pretval,
  613. const char *haystk, size_t haystk_nbytes,
  614. const char *ndl, size_t ndl_nbytes,
  615. size_t offset, const char *enc, bool reverse)
  616. {
  617. char buf[GENERIC_SUPERSET_NBYTES];
  618. php_iconv_err_t err = PHP_ICONV_ERR_SUCCESS;
  619. iconv_t cd;
  620. const char *in_p;
  621. size_t in_left;
  622. char *out_p;
  623. size_t out_left;
  624. size_t cnt;
  625. zend_string *ndl_buf;
  626. const char *ndl_buf_p;
  627. size_t ndl_buf_left;
  628. size_t match_ofs;
  629. int more;
  630. size_t iconv_ret;
  631. *pretval = (size_t)-1;
  632. err = php_iconv_string(ndl, ndl_nbytes, &ndl_buf, GENERIC_SUPERSET_NAME, enc);
  633. if (err != PHP_ICONV_ERR_SUCCESS) {
  634. if (ndl_buf != NULL) {
  635. zend_string_efree(ndl_buf);
  636. }
  637. return err;
  638. }
  639. cd = iconv_open(GENERIC_SUPERSET_NAME, enc);
  640. if (cd == (iconv_t)(-1)) {
  641. if (ndl_buf != NULL) {
  642. zend_string_efree(ndl_buf);
  643. }
  644. if (errno == EINVAL) {
  645. return PHP_ICONV_ERR_WRONG_CHARSET;
  646. } else {
  647. return PHP_ICONV_ERR_CONVERTER;
  648. }
  649. }
  650. ndl_buf_p = ZSTR_VAL(ndl_buf);
  651. ndl_buf_left = ZSTR_LEN(ndl_buf);
  652. match_ofs = (size_t)-1;
  653. more = haystk_nbytes > 0;
  654. for (in_p = haystk, in_left = haystk_nbytes, cnt = 0; more; ++cnt) {
  655. out_p = buf;
  656. out_left = sizeof(buf);
  657. more = in_left > 0;
  658. iconv_ret = iconv(cd, more ? (char **)&in_p : NULL, more ? &in_left : NULL, (char **) &out_p, &out_left);
  659. if (out_left == sizeof(buf)) {
  660. break;
  661. }
  662. if (iconv_ret == (size_t)-1) {
  663. switch (errno) {
  664. case EINVAL:
  665. err = PHP_ICONV_ERR_ILLEGAL_CHAR;
  666. break;
  667. case EILSEQ:
  668. err = PHP_ICONV_ERR_ILLEGAL_SEQ;
  669. break;
  670. case E2BIG:
  671. break;
  672. default:
  673. err = PHP_ICONV_ERR_UNKNOWN;
  674. break;
  675. }
  676. }
  677. if (cnt >= offset) {
  678. if (_php_iconv_memequal(buf, ndl_buf_p, sizeof(buf))) {
  679. if (match_ofs == (size_t)-1) {
  680. match_ofs = cnt;
  681. }
  682. ndl_buf_p += GENERIC_SUPERSET_NBYTES;
  683. ndl_buf_left -= GENERIC_SUPERSET_NBYTES;
  684. if (ndl_buf_left == 0) {
  685. *pretval = match_ofs;
  686. if (reverse) {
  687. /* If searching backward, continue trying to find a later match. */
  688. ndl_buf_p = ZSTR_VAL(ndl_buf);
  689. ndl_buf_left = ZSTR_LEN(ndl_buf);
  690. match_ofs = -1;
  691. } else {
  692. /* If searching forward, stop at first match. */
  693. break;
  694. }
  695. }
  696. } else {
  697. size_t i, j, lim;
  698. i = 0;
  699. j = GENERIC_SUPERSET_NBYTES;
  700. lim = (size_t)(ndl_buf_p - ZSTR_VAL(ndl_buf));
  701. while (j < lim) {
  702. if (_php_iconv_memequal(&ZSTR_VAL(ndl_buf)[j], &ZSTR_VAL(ndl_buf)[i],
  703. GENERIC_SUPERSET_NBYTES)) {
  704. i += GENERIC_SUPERSET_NBYTES;
  705. } else {
  706. j -= i;
  707. i = 0;
  708. }
  709. j += GENERIC_SUPERSET_NBYTES;
  710. }
  711. if (_php_iconv_memequal(buf, &ZSTR_VAL(ndl_buf)[i], sizeof(buf))) {
  712. match_ofs += (lim - i) / GENERIC_SUPERSET_NBYTES;
  713. i += GENERIC_SUPERSET_NBYTES;
  714. ndl_buf_p = &ZSTR_VAL(ndl_buf)[i];
  715. ndl_buf_left = ZSTR_LEN(ndl_buf) - i;
  716. } else {
  717. match_ofs = (size_t)-1;
  718. ndl_buf_p = ZSTR_VAL(ndl_buf);
  719. ndl_buf_left = ZSTR_LEN(ndl_buf);
  720. }
  721. }
  722. }
  723. }
  724. if (ndl_buf) {
  725. zend_string_efree(ndl_buf);
  726. }
  727. iconv_close(cd);
  728. if (err == PHP_ICONV_ERR_SUCCESS && offset > cnt) {
  729. return PHP_ICONV_ERR_OUT_BY_BOUNDS;
  730. }
  731. return err;
  732. }
  733. /* }}} */
  734. /* {{{ _php_iconv_mime_encode() */
  735. static php_iconv_err_t _php_iconv_mime_encode(smart_str *pretval, const char *fname, size_t fname_nbytes, const char *fval, size_t fval_nbytes, size_t max_line_len, const char *lfchars, php_iconv_enc_scheme_t enc_scheme, const char *out_charset, const char *enc)
  736. {
  737. php_iconv_err_t err = PHP_ICONV_ERR_SUCCESS;
  738. iconv_t cd = (iconv_t)(-1), cd_pl = (iconv_t)(-1);
  739. size_t char_cnt = 0;
  740. size_t out_charset_len;
  741. size_t lfchars_len;
  742. char *buf = NULL;
  743. const char *in_p;
  744. size_t in_left;
  745. char *out_p;
  746. size_t out_left;
  747. zend_string *encoded = NULL;
  748. static int qp_table[256] = {
  749. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0x00 */
  750. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0x10 */
  751. 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 0x20 */
  752. 1, 1, 1, 1, 1, 1, 1 ,1, 1, 1, 1, 1, 1, 3, 1, 3, /* 0x30 */
  753. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 0x40 */
  754. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, /* 0x50 */
  755. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 0x60 */
  756. 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, /* 0x70 */
  757. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0x80 */
  758. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0x90 */
  759. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0xA0 */
  760. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0xB0 */
  761. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0xC0 */
  762. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0xD0 */
  763. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, /* 0xE0 */
  764. 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 /* 0xF0 */
  765. };
  766. out_charset_len = strlen(out_charset);
  767. lfchars_len = strlen(lfchars);
  768. if ((fname_nbytes + 2) >= max_line_len
  769. || (out_charset_len + 12) >= max_line_len) {
  770. /* field name is too long */
  771. err = PHP_ICONV_ERR_TOO_BIG;
  772. goto out;
  773. }
  774. cd_pl = iconv_open(ICONV_ASCII_ENCODING, enc);
  775. if (cd_pl == (iconv_t)(-1)) {
  776. if (errno == EINVAL) {
  777. err = PHP_ICONV_ERR_WRONG_CHARSET;
  778. } else {
  779. err = PHP_ICONV_ERR_CONVERTER;
  780. }
  781. goto out;
  782. }
  783. cd = iconv_open(out_charset, enc);
  784. if (cd == (iconv_t)(-1)) {
  785. if (errno == EINVAL) {
  786. err = PHP_ICONV_ERR_WRONG_CHARSET;
  787. } else {
  788. err = PHP_ICONV_ERR_CONVERTER;
  789. }
  790. goto out;
  791. }
  792. buf = safe_emalloc(1, max_line_len, 5);
  793. char_cnt = max_line_len;
  794. _php_iconv_appendl(pretval, fname, fname_nbytes, cd_pl);
  795. char_cnt -= fname_nbytes;
  796. smart_str_appendl(pretval, ": ", sizeof(": ") - 1);
  797. char_cnt -= 2;
  798. in_p = fval;
  799. in_left = fval_nbytes;
  800. do {
  801. size_t prev_in_left;
  802. size_t out_size;
  803. size_t encoded_word_min_len = sizeof("=\?\?X\?\?=")-1 + out_charset_len + (enc_scheme == PHP_ICONV_ENC_SCHEME_BASE64 ? 4 : 3);
  804. if (char_cnt < encoded_word_min_len + lfchars_len + 1) {
  805. /* lfchars must be encoded in ASCII here*/
  806. smart_str_appendl(pretval, lfchars, lfchars_len);
  807. smart_str_appendc(pretval, ' ');
  808. char_cnt = max_line_len - 1;
  809. }
  810. smart_str_appendl(pretval, "=?", sizeof("=?") - 1);
  811. char_cnt -= 2;
  812. smart_str_appendl(pretval, out_charset, out_charset_len);
  813. char_cnt -= out_charset_len;
  814. smart_str_appendc(pretval, '?');
  815. char_cnt --;
  816. switch (enc_scheme) {
  817. case PHP_ICONV_ENC_SCHEME_BASE64: {
  818. size_t ini_in_left;
  819. const char *ini_in_p;
  820. size_t out_reserved = 4;
  821. smart_str_appendc(pretval, 'B');
  822. char_cnt--;
  823. smart_str_appendc(pretval, '?');
  824. char_cnt--;
  825. prev_in_left = ini_in_left = in_left;
  826. ini_in_p = in_p;
  827. out_size = (char_cnt - 2) / 4 * 3;
  828. for (;;) {
  829. out_p = buf;
  830. if (out_size <= out_reserved) {
  831. err = PHP_ICONV_ERR_TOO_BIG;
  832. goto out;
  833. }
  834. out_left = out_size - out_reserved;
  835. if (iconv(cd, (char **)&in_p, &in_left, (char **) &out_p, &out_left) == (size_t)-1) {
  836. switch (errno) {
  837. case EINVAL:
  838. err = PHP_ICONV_ERR_ILLEGAL_CHAR;
  839. goto out;
  840. case EILSEQ:
  841. err = PHP_ICONV_ERR_ILLEGAL_SEQ;
  842. goto out;
  843. case E2BIG:
  844. if (prev_in_left == in_left) {
  845. err = PHP_ICONV_ERR_TOO_BIG;
  846. goto out;
  847. }
  848. break;
  849. default:
  850. err = PHP_ICONV_ERR_UNKNOWN;
  851. goto out;
  852. }
  853. }
  854. out_left += out_reserved;
  855. if (iconv(cd, NULL, NULL, (char **) &out_p, &out_left) == (size_t)-1) {
  856. if (errno != E2BIG) {
  857. err = PHP_ICONV_ERR_UNKNOWN;
  858. goto out;
  859. }
  860. } else {
  861. break;
  862. }
  863. if (iconv(cd, NULL, NULL, NULL, NULL) == (size_t)-1) {
  864. err = PHP_ICONV_ERR_UNKNOWN;
  865. goto out;
  866. }
  867. out_reserved += 4;
  868. in_left = ini_in_left;
  869. in_p = ini_in_p;
  870. }
  871. prev_in_left = in_left;
  872. encoded = php_base64_encode((unsigned char *) buf, (out_size - out_left));
  873. if (char_cnt < ZSTR_LEN(encoded)) {
  874. /* something went wrong! */
  875. err = PHP_ICONV_ERR_UNKNOWN;
  876. goto out;
  877. }
  878. smart_str_appendl(pretval, ZSTR_VAL(encoded), ZSTR_LEN(encoded));
  879. char_cnt -= ZSTR_LEN(encoded);
  880. smart_str_appendl(pretval, "?=", sizeof("?=") - 1);
  881. char_cnt -= 2;
  882. zend_string_release_ex(encoded, 0);
  883. encoded = NULL;
  884. } break; /* case PHP_ICONV_ENC_SCHEME_BASE64: */
  885. case PHP_ICONV_ENC_SCHEME_QPRINT: {
  886. size_t ini_in_left;
  887. const char *ini_in_p;
  888. const unsigned char *p;
  889. size_t nbytes_required;
  890. smart_str_appendc(pretval, 'Q');
  891. char_cnt--;
  892. smart_str_appendc(pretval, '?');
  893. char_cnt--;
  894. prev_in_left = ini_in_left = in_left;
  895. ini_in_p = in_p;
  896. for (out_size = (char_cnt - 2); out_size > 0;) {
  897. nbytes_required = 0;
  898. out_p = buf;
  899. out_left = out_size;
  900. if (iconv(cd, (char **)&in_p, &in_left, (char **) &out_p, &out_left) == (size_t)-1) {
  901. switch (errno) {
  902. case EINVAL:
  903. err = PHP_ICONV_ERR_ILLEGAL_CHAR;
  904. goto out;
  905. case EILSEQ:
  906. err = PHP_ICONV_ERR_ILLEGAL_SEQ;
  907. goto out;
  908. case E2BIG:
  909. if (prev_in_left == in_left) {
  910. err = PHP_ICONV_ERR_UNKNOWN;
  911. goto out;
  912. }
  913. break;
  914. default:
  915. err = PHP_ICONV_ERR_UNKNOWN;
  916. goto out;
  917. }
  918. }
  919. if (iconv(cd, NULL, NULL, (char **) &out_p, &out_left) == (size_t)-1) {
  920. if (errno != E2BIG) {
  921. err = PHP_ICONV_ERR_UNKNOWN;
  922. goto out;
  923. }
  924. }
  925. for (p = (unsigned char *)buf; p < (unsigned char *)out_p; p++) {
  926. nbytes_required += qp_table[*p];
  927. }
  928. if (nbytes_required <= char_cnt - 2) {
  929. break;
  930. }
  931. out_size -= ((nbytes_required - (char_cnt - 2)) + 2) / 3;
  932. in_left = ini_in_left;
  933. in_p = ini_in_p;
  934. }
  935. for (p = (unsigned char *)buf; p < (unsigned char *)out_p; p++) {
  936. if (qp_table[*p] == 1) {
  937. smart_str_appendc(pretval, *(char *)p);
  938. char_cnt--;
  939. } else {
  940. static const char qp_digits[] = "0123456789ABCDEF";
  941. smart_str_appendc(pretval, '=');
  942. smart_str_appendc(pretval, qp_digits[(*p >> 4) & 0x0f]);
  943. smart_str_appendc(pretval, qp_digits[(*p & 0x0f)]);
  944. char_cnt -= 3;
  945. }
  946. }
  947. smart_str_appendl(pretval, "?=", sizeof("?=") - 1);
  948. char_cnt -= 2;
  949. if (iconv(cd, NULL, NULL, NULL, NULL) == (size_t)-1) {
  950. err = PHP_ICONV_ERR_UNKNOWN;
  951. goto out;
  952. }
  953. } break; /* case PHP_ICONV_ENC_SCHEME_QPRINT: */
  954. }
  955. } while (in_left > 0);
  956. smart_str_0(pretval);
  957. out:
  958. if (cd != (iconv_t)(-1)) {
  959. iconv_close(cd);
  960. }
  961. if (cd_pl != (iconv_t)(-1)) {
  962. iconv_close(cd_pl);
  963. }
  964. if (encoded != NULL) {
  965. zend_string_release_ex(encoded, 0);
  966. }
  967. if (buf != NULL) {
  968. efree(buf);
  969. }
  970. return err;
  971. }
  972. /* }}} */
  973. /* {{{ _php_iconv_mime_decode() */
  974. static php_iconv_err_t _php_iconv_mime_decode(smart_str *pretval, const char *str, size_t str_nbytes, const char *enc, const char **next_pos, int mode)
  975. {
  976. php_iconv_err_t err = PHP_ICONV_ERR_SUCCESS;
  977. iconv_t cd = (iconv_t)(-1), cd_pl = (iconv_t)(-1);
  978. const char *p1;
  979. size_t str_left;
  980. unsigned int scan_stat = 0;
  981. const char *csname = NULL;
  982. size_t csname_len;
  983. const char *encoded_text = NULL;
  984. size_t encoded_text_len = 0;
  985. const char *encoded_word = NULL;
  986. const char *spaces = NULL;
  987. php_iconv_enc_scheme_t enc_scheme = PHP_ICONV_ENC_SCHEME_BASE64;
  988. if (next_pos != NULL) {
  989. *next_pos = NULL;
  990. }
  991. cd_pl = iconv_open(enc, ICONV_ASCII_ENCODING);
  992. if (cd_pl == (iconv_t)(-1)) {
  993. if (errno == EINVAL) {
  994. err = PHP_ICONV_ERR_WRONG_CHARSET;
  995. } else {
  996. err = PHP_ICONV_ERR_CONVERTER;
  997. }
  998. goto out;
  999. }
  1000. p1 = str;
  1001. for (str_left = str_nbytes; str_left > 0; str_left--, p1++) {
  1002. int eos = 0;
  1003. switch (scan_stat) {
  1004. case 0: /* expecting any character */
  1005. switch (*p1) {
  1006. case '\r': /* part of an EOL sequence? */
  1007. scan_stat = 7;
  1008. break;
  1009. case '\n':
  1010. scan_stat = 8;
  1011. break;
  1012. case '=': /* first letter of an encoded chunk */
  1013. encoded_word = p1;
  1014. scan_stat = 1;
  1015. break;
  1016. case ' ': case '\t': /* a chunk of whitespaces */
  1017. spaces = p1;
  1018. scan_stat = 11;
  1019. break;
  1020. default: /* first letter of a non-encoded word */
  1021. err = _php_iconv_appendc(pretval, *p1, cd_pl);
  1022. if (err != PHP_ICONV_ERR_SUCCESS) {
  1023. if (mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR) {
  1024. err = PHP_ICONV_ERR_SUCCESS;
  1025. } else {
  1026. goto out;
  1027. }
  1028. }
  1029. encoded_word = NULL;
  1030. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1031. scan_stat = 12;
  1032. }
  1033. break;
  1034. }
  1035. break;
  1036. case 1: /* expecting a delimiter */
  1037. if (*p1 != '?') {
  1038. if (*p1 == '\r' || *p1 == '\n') {
  1039. --p1;
  1040. }
  1041. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1042. if (err != PHP_ICONV_ERR_SUCCESS) {
  1043. goto out;
  1044. }
  1045. encoded_word = NULL;
  1046. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1047. scan_stat = 12;
  1048. } else {
  1049. scan_stat = 0;
  1050. }
  1051. break;
  1052. }
  1053. csname = p1 + 1;
  1054. scan_stat = 2;
  1055. break;
  1056. case 2: /* expecting a charset name */
  1057. switch (*p1) {
  1058. case '?': /* normal delimiter: encoding scheme follows */
  1059. scan_stat = 3;
  1060. break;
  1061. case '*': /* new style delimiter: locale id follows */
  1062. scan_stat = 10;
  1063. break;
  1064. case '\r': case '\n': /* not an encoded-word */
  1065. --p1;
  1066. _php_iconv_appendc(pretval, '=', cd_pl);
  1067. _php_iconv_appendc(pretval, '?', cd_pl);
  1068. err = _php_iconv_appendl(pretval, csname, (size_t)((p1 + 1) - csname), cd_pl);
  1069. if (err != PHP_ICONV_ERR_SUCCESS) {
  1070. goto out;
  1071. }
  1072. csname = NULL;
  1073. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1074. scan_stat = 12;
  1075. }
  1076. else {
  1077. scan_stat = 0;
  1078. }
  1079. continue;
  1080. }
  1081. if (scan_stat != 2) {
  1082. char tmpbuf[80];
  1083. if (csname == NULL) {
  1084. err = PHP_ICONV_ERR_MALFORMED;
  1085. goto out;
  1086. }
  1087. csname_len = (size_t)(p1 - csname);
  1088. if (csname_len > sizeof(tmpbuf) - 1) {
  1089. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1090. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1091. if (err != PHP_ICONV_ERR_SUCCESS) {
  1092. goto out;
  1093. }
  1094. encoded_word = NULL;
  1095. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1096. scan_stat = 12;
  1097. } else {
  1098. scan_stat = 0;
  1099. }
  1100. break;
  1101. } else {
  1102. err = PHP_ICONV_ERR_MALFORMED;
  1103. goto out;
  1104. }
  1105. }
  1106. memcpy(tmpbuf, csname, csname_len);
  1107. tmpbuf[csname_len] = '\0';
  1108. if (cd != (iconv_t)(-1)) {
  1109. iconv_close(cd);
  1110. }
  1111. cd = iconv_open(enc, tmpbuf);
  1112. if (cd == (iconv_t)(-1)) {
  1113. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1114. /* Bad character set, but the user wants us to
  1115. * press on. In this case, we'll just insert the
  1116. * undecoded encoded word, since there isn't really
  1117. * a more sensible behaviour available; the only
  1118. * other options are to swallow the encoded word
  1119. * entirely or decode it with an arbitrarily chosen
  1120. * single byte encoding, both of which seem to have
  1121. * a higher WTF factor than leaving it undecoded.
  1122. *
  1123. * Given this approach, we need to skip ahead to
  1124. * the end of the encoded word. */
  1125. int qmarks = 2;
  1126. while (qmarks > 0 && str_left > 1) {
  1127. if (*(++p1) == '?') {
  1128. --qmarks;
  1129. }
  1130. --str_left;
  1131. }
  1132. /* Look ahead to check for the terminating = that
  1133. * should be there as well; if it's there, we'll
  1134. * also include that. If it's not, there isn't much
  1135. * we can do at this point. */
  1136. if (*(p1 + 1) == '=') {
  1137. ++p1;
  1138. if (str_left > 1) {
  1139. --str_left;
  1140. }
  1141. }
  1142. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1143. if (err != PHP_ICONV_ERR_SUCCESS) {
  1144. goto out;
  1145. }
  1146. /* Let's go back and see if there are further
  1147. * encoded words or bare content, and hope they
  1148. * might actually have a valid character set. */
  1149. scan_stat = 12;
  1150. break;
  1151. } else {
  1152. if (errno == EINVAL) {
  1153. err = PHP_ICONV_ERR_WRONG_CHARSET;
  1154. } else {
  1155. err = PHP_ICONV_ERR_CONVERTER;
  1156. }
  1157. goto out;
  1158. }
  1159. }
  1160. }
  1161. break;
  1162. case 3: /* expecting a encoding scheme specifier */
  1163. switch (*p1) {
  1164. case 'b':
  1165. case 'B':
  1166. enc_scheme = PHP_ICONV_ENC_SCHEME_BASE64;
  1167. scan_stat = 4;
  1168. break;
  1169. case 'q':
  1170. case 'Q':
  1171. enc_scheme = PHP_ICONV_ENC_SCHEME_QPRINT;
  1172. scan_stat = 4;
  1173. break;
  1174. default:
  1175. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1176. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1177. if (err != PHP_ICONV_ERR_SUCCESS) {
  1178. goto out;
  1179. }
  1180. encoded_word = NULL;
  1181. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1182. scan_stat = 12;
  1183. } else {
  1184. scan_stat = 0;
  1185. }
  1186. break;
  1187. } else {
  1188. err = PHP_ICONV_ERR_MALFORMED;
  1189. goto out;
  1190. }
  1191. }
  1192. break;
  1193. case 4: /* expecting a delimiter */
  1194. if (*p1 != '?') {
  1195. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1196. /* pass the entire chunk through the converter */
  1197. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1198. if (err != PHP_ICONV_ERR_SUCCESS) {
  1199. goto out;
  1200. }
  1201. encoded_word = NULL;
  1202. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1203. scan_stat = 12;
  1204. } else {
  1205. scan_stat = 0;
  1206. }
  1207. break;
  1208. } else {
  1209. err = PHP_ICONV_ERR_MALFORMED;
  1210. goto out;
  1211. }
  1212. }
  1213. encoded_text = p1 + 1;
  1214. scan_stat = 5;
  1215. break;
  1216. case 5: /* expecting an encoded portion */
  1217. if (*p1 == '?') {
  1218. encoded_text_len = (size_t)(p1 - encoded_text);
  1219. scan_stat = 6;
  1220. }
  1221. break;
  1222. case 7: /* expecting a "\n" character */
  1223. if (*p1 == '\n') {
  1224. scan_stat = 8;
  1225. } else {
  1226. /* bare CR */
  1227. _php_iconv_appendc(pretval, '\r', cd_pl);
  1228. _php_iconv_appendc(pretval, *p1, cd_pl);
  1229. scan_stat = 0;
  1230. }
  1231. break;
  1232. case 8: /* checking whether the following line is part of a
  1233. folded header */
  1234. if (*p1 != ' ' && *p1 != '\t') {
  1235. --p1;
  1236. str_left = 1; /* quit_loop */
  1237. break;
  1238. }
  1239. if (encoded_word == NULL) {
  1240. _php_iconv_appendc(pretval, ' ', cd_pl);
  1241. }
  1242. spaces = NULL;
  1243. scan_stat = 11;
  1244. break;
  1245. case 6: /* expecting a End-Of-Chunk character "=" */
  1246. if (*p1 != '=') {
  1247. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1248. /* pass the entire chunk through the converter */
  1249. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1250. if (err != PHP_ICONV_ERR_SUCCESS) {
  1251. goto out;
  1252. }
  1253. encoded_word = NULL;
  1254. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1255. scan_stat = 12;
  1256. } else {
  1257. scan_stat = 0;
  1258. }
  1259. break;
  1260. } else {
  1261. err = PHP_ICONV_ERR_MALFORMED;
  1262. goto out;
  1263. }
  1264. }
  1265. scan_stat = 9;
  1266. if (str_left == 1) {
  1267. eos = 1;
  1268. } else {
  1269. break;
  1270. }
  1271. /* TODO might want to rearrange logic so this is more obvious */
  1272. ZEND_FALLTHROUGH;
  1273. case 9: /* choice point, seeing what to do next.*/
  1274. switch (*p1) {
  1275. default:
  1276. /* Handle non-RFC-compliant formats
  1277. *
  1278. * RFC2047 requires the character that comes right
  1279. * after an encoded word (chunk) to be a whitespace,
  1280. * while there are lots of broken implementations that
  1281. * generate such malformed headers that don't fulfill
  1282. * that requirement.
  1283. */
  1284. if (!eos) {
  1285. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1286. /* pass the entire chunk through the converter */
  1287. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1288. if (err != PHP_ICONV_ERR_SUCCESS) {
  1289. goto out;
  1290. }
  1291. scan_stat = 12;
  1292. break;
  1293. }
  1294. }
  1295. ZEND_FALLTHROUGH;
  1296. case '\r': case '\n': case ' ': case '\t': {
  1297. zend_string *decoded_text;
  1298. switch (enc_scheme) {
  1299. case PHP_ICONV_ENC_SCHEME_BASE64:
  1300. decoded_text = php_base64_decode((unsigned char*)encoded_text, encoded_text_len);
  1301. break;
  1302. case PHP_ICONV_ENC_SCHEME_QPRINT:
  1303. decoded_text = php_quot_print_decode((unsigned char*)encoded_text, encoded_text_len, 1);
  1304. break;
  1305. default:
  1306. decoded_text = NULL;
  1307. break;
  1308. }
  1309. if (decoded_text == NULL) {
  1310. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1311. /* pass the entire chunk through the converter */
  1312. err = _php_iconv_appendl(pretval, encoded_word, (size_t)((p1 + 1) - encoded_word), cd_pl);
  1313. if (err != PHP_ICONV_ERR_SUCCESS) {
  1314. goto out;
  1315. }
  1316. encoded_word = NULL;
  1317. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1318. scan_stat = 12;
  1319. } else {
  1320. scan_stat = 0;
  1321. }
  1322. break;
  1323. } else {
  1324. err = PHP_ICONV_ERR_UNKNOWN;
  1325. goto out;
  1326. }
  1327. }
  1328. err = _php_iconv_appendl(pretval, ZSTR_VAL(decoded_text), ZSTR_LEN(decoded_text), cd);
  1329. zend_string_release_ex(decoded_text, 0);
  1330. if (err != PHP_ICONV_ERR_SUCCESS) {
  1331. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1332. /* pass the entire chunk through the converter */
  1333. err = _php_iconv_appendl(pretval, encoded_word, (size_t)(p1 - encoded_word), cd_pl);
  1334. encoded_word = NULL;
  1335. if (err != PHP_ICONV_ERR_SUCCESS) {
  1336. break;
  1337. }
  1338. } else {
  1339. goto out;
  1340. }
  1341. }
  1342. if (eos) { /* reached end-of-string. done. */
  1343. scan_stat = 0;
  1344. break;
  1345. }
  1346. switch (*p1) {
  1347. case '\r': /* part of an EOL sequence? */
  1348. scan_stat = 7;
  1349. break;
  1350. case '\n':
  1351. scan_stat = 8;
  1352. break;
  1353. case '=': /* first letter of an encoded chunk */
  1354. scan_stat = 1;
  1355. break;
  1356. case ' ': case '\t': /* medial whitespaces */
  1357. spaces = p1;
  1358. scan_stat = 11;
  1359. break;
  1360. default: /* first letter of a non-encoded word */
  1361. _php_iconv_appendc(pretval, *p1, cd_pl);
  1362. scan_stat = 12;
  1363. break;
  1364. }
  1365. } break;
  1366. }
  1367. break;
  1368. case 10: /* expects a language specifier. dismiss it for now */
  1369. if (*p1 == '?') {
  1370. scan_stat = 3;
  1371. }
  1372. break;
  1373. case 11: /* expecting a chunk of whitespaces */
  1374. switch (*p1) {
  1375. case '\r': /* part of an EOL sequence? */
  1376. scan_stat = 7;
  1377. break;
  1378. case '\n':
  1379. scan_stat = 8;
  1380. break;
  1381. case '=': /* first letter of an encoded chunk */
  1382. if (spaces != NULL && encoded_word == NULL) {
  1383. _php_iconv_appendl(pretval, spaces, (size_t)(p1 - spaces), cd_pl);
  1384. spaces = NULL;
  1385. }
  1386. encoded_word = p1;
  1387. scan_stat = 1;
  1388. break;
  1389. case ' ': case '\t':
  1390. break;
  1391. default: /* first letter of a non-encoded word */
  1392. if (spaces != NULL) {
  1393. _php_iconv_appendl(pretval, spaces, (size_t)(p1 - spaces), cd_pl);
  1394. spaces = NULL;
  1395. }
  1396. _php_iconv_appendc(pretval, *p1, cd_pl);
  1397. encoded_word = NULL;
  1398. if ((mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1399. scan_stat = 12;
  1400. } else {
  1401. scan_stat = 0;
  1402. }
  1403. break;
  1404. }
  1405. break;
  1406. case 12: /* expecting a non-encoded word */
  1407. switch (*p1) {
  1408. case '\r': /* part of an EOL sequence? */
  1409. scan_stat = 7;
  1410. break;
  1411. case '\n':
  1412. scan_stat = 8;
  1413. break;
  1414. case ' ': case '\t':
  1415. spaces = p1;
  1416. scan_stat = 11;
  1417. break;
  1418. case '=': /* first letter of an encoded chunk */
  1419. if (!(mode & PHP_ICONV_MIME_DECODE_STRICT)) {
  1420. encoded_word = p1;
  1421. scan_stat = 1;
  1422. break;
  1423. }
  1424. ZEND_FALLTHROUGH;
  1425. default:
  1426. _php_iconv_appendc(pretval, *p1, cd_pl);
  1427. break;
  1428. }
  1429. break;
  1430. }
  1431. }
  1432. switch (scan_stat) {
  1433. case 0: case 8: case 11: case 12:
  1434. break;
  1435. default:
  1436. if ((mode & PHP_ICONV_MIME_DECODE_CONTINUE_ON_ERROR)) {
  1437. if (scan_stat == 1) {
  1438. _php_iconv_appendc(pretval, '=', cd_pl);
  1439. }
  1440. err = PHP_ICONV_ERR_SUCCESS;
  1441. } else {
  1442. err = PHP_ICONV_ERR_MALFORMED;
  1443. goto out;
  1444. }
  1445. }
  1446. if (next_pos != NULL) {
  1447. *next_pos = p1;
  1448. }
  1449. if (cd != (iconv_t)(-1)) {
  1450. _php_iconv_appendl(pretval, NULL, 0, cd);
  1451. }
  1452. if (cd_pl != (iconv_t)(-1)) {
  1453. _php_iconv_appendl(pretval, NULL, 0, cd_pl);
  1454. }
  1455. smart_str_0(pretval);
  1456. out:
  1457. if (cd != (iconv_t)(-1)) {
  1458. iconv_close(cd);
  1459. }
  1460. if (cd_pl != (iconv_t)(-1)) {
  1461. iconv_close(cd_pl);
  1462. }
  1463. return err;
  1464. }
  1465. /* }}} */
  1466. /* {{{ php_iconv_show_error() */
  1467. static void _php_iconv_show_error(php_iconv_err_t err, const char *out_charset, const char *in_charset)
  1468. {
  1469. switch (err) {
  1470. case PHP_ICONV_ERR_SUCCESS:
  1471. break;
  1472. case PHP_ICONV_ERR_CONVERTER:
  1473. php_error_docref(NULL, E_WARNING, "Cannot open converter");
  1474. break;
  1475. case PHP_ICONV_ERR_WRONG_CHARSET:
  1476. php_error_docref(NULL, E_WARNING, "Wrong encoding, conversion from \"%s\" to \"%s\" is not allowed",
  1477. in_charset, out_charset);
  1478. break;
  1479. case PHP_ICONV_ERR_ILLEGAL_CHAR:
  1480. php_error_docref(NULL, E_NOTICE, "Detected an incomplete multibyte character in input string");
  1481. break;
  1482. case PHP_ICONV_ERR_ILLEGAL_SEQ:
  1483. php_error_docref(NULL, E_NOTICE, "Detected an illegal character in input string");
  1484. break;
  1485. case PHP_ICONV_ERR_TOO_BIG:
  1486. /* should not happen */
  1487. php_error_docref(NULL, E_WARNING, "Buffer length exceeded");
  1488. break;
  1489. case PHP_ICONV_ERR_MALFORMED:
  1490. php_error_docref(NULL, E_WARNING, "Malformed string");
  1491. break;
  1492. case PHP_ICONV_ERR_OUT_BY_BOUNDS:
  1493. zend_argument_value_error(3, "must be contained in argument #1 ($haystack)");
  1494. break;
  1495. default:
  1496. /* other error */
  1497. php_error_docref(NULL, E_NOTICE, "Unknown error (%d)", errno);
  1498. break;
  1499. }
  1500. }
  1501. /* }}} */
  1502. /* {{{ Returns the character count of str */
  1503. PHP_FUNCTION(iconv_strlen)
  1504. {
  1505. const char *charset = NULL;
  1506. size_t charset_len;
  1507. zend_string *str;
  1508. php_iconv_err_t err;
  1509. size_t retval;
  1510. if (zend_parse_parameters(ZEND_NUM_ARGS(), "S|s!",
  1511. &str, &charset, &charset_len) == FAILURE) {
  1512. RETURN_THROWS();
  1513. }
  1514. if (charset == NULL) {
  1515. charset = get_internal_encoding();
  1516. } else if (charset_len >= ICONV_CSNMAXLEN) {
  1517. php_error_docref(NULL, E_WARNING, "Encoding parameter exceeds the maximum allowed length of %d characters", ICONV_CSNMAXLEN);
  1518. RETURN_FALSE;
  1519. }
  1520. err = _php_iconv_strlen(&retval, ZSTR_VAL(str), ZSTR_LEN(str), charset);
  1521. _php_iconv_show_error(err, GENERIC_SUPERSET_NAME, charset);
  1522. if (err == PHP_ICONV_ERR_SUCCESS) {
  1523. RETVAL_LONG(retval);
  1524. } else {
  1525. RETVAL_FALSE;
  1526. }
  1527. }
  1528. /* }}} */
  1529. /* {{{ Returns specified part of a string */
  1530. PHP_FUNCTION(iconv_substr)
  1531. {
  1532. const char *charset = NULL;
  1533. size_t charset_len;
  1534. zend_string *str;
  1535. zend_long offset, length = 0;
  1536. bool len_is_null = 1;
  1537. php_iconv_err_t err;
  1538. smart_str retval = {0};
  1539. if (zend_parse_parameters(ZEND_NUM_ARGS(), "Sl|l!s!",
  1540. &str, &offset, &length, &len_is_null,
  1541. &charset, &charset_len) == FAILURE) {
  1542. RETURN_THROWS();
  1543. }
  1544. if (charset == NULL) {
  1545. charset = get_internal_encoding();
  1546. } else if (charset_len >= ICONV_CSNMAXLEN) {
  1547. php_error_docref(NULL, E_WARNING, "Encoding parameter exceeds the maximum allowed length of %d characters", ICONV_CSNMAXLEN);
  1548. RETURN_FALSE;
  1549. }
  1550. if (len_is_null) {
  1551. length = ZSTR_LEN(str);
  1552. }
  1553. err = _php_iconv_substr(&retval, ZSTR_VAL(str), ZSTR_LEN(str), offset, length, charset);
  1554. _php_iconv_show_error(err, GENERIC_SUPERSET_NAME, charset);
  1555. if (err == PHP_ICONV_ERR_SUCCESS && retval.s != NULL) {
  1556. RETURN_NEW_STR(retval.s);
  1557. }
  1558. smart_str_free(&retval);
  1559. RETURN_FALSE;
  1560. }
  1561. /* }}} */
  1562. /* {{{ Finds position of first occurrence of needle within part of haystack beginning with offset */
  1563. PHP_FUNCTION(iconv_strpos)
  1564. {
  1565. const char *charset = NULL;
  1566. size_t charset_len, haystk_len;
  1567. zend_string *haystk;
  1568. zend_string *ndl;
  1569. zend_long offset = 0;
  1570. php_iconv_err_t err;
  1571. size_t retval;
  1572. if (zend_parse_parameters(ZEND_NUM_ARGS(), "SS|ls!",
  1573. &haystk, &ndl,
  1574. &offset, &charset, &charset_len) == FAILURE) {
  1575. RETURN_THROWS();
  1576. }
  1577. if (charset == NULL) {
  1578. charset = get_internal_encoding();
  1579. } else if (charset_len >= ICONV_CSNMAXLEN) {
  1580. php_error_docref(NULL, E_WARNING, "Encoding parameter exceeds the maximum allowed length of %d characters", ICONV_CSNMAXLEN);
  1581. RETURN_FALSE;
  1582. }
  1583. if (offset < 0) {
  1584. /* Convert negative offset (counted from the end of string) */
  1585. err = _php_iconv_strlen(&haystk_len, ZSTR_VAL(haystk), ZSTR_LEN(haystk), charset);
  1586. if (err != PHP_ICONV_ERR_SUCCESS) {
  1587. _php_iconv_show_error(err, GENERIC_SUPERSET_NAME, charset);
  1588. RETURN_FALSE;
  1589. }
  1590. offset += haystk_len;
  1591. if (offset < 0) { /* If offset before start */
  1592. zend_argument_value_error(3, "must be contained in argument #1 ($haystack)");
  1593. RETURN_THROWS();
  1594. }
  1595. }
  1596. if (ZSTR_LEN(ndl) < 1) {
  1597. // TODO: Support empty needles!
  1598. RETURN_FALSE;
  1599. }
  1600. err = _php_iconv_strpos(
  1601. &retval, ZSTR_VAL(haystk), ZSTR_LEN(haystk), ZSTR_VAL(ndl), ZSTR_LEN(ndl),
  1602. offset, charset, /* reverse */ false);
  1603. _php_iconv_show_error(err, GENERIC_SUPERSET_NAME, charset);
  1604. if (err == PHP_ICONV_ERR_SUCCESS && retval != (size_t)-1) {
  1605. RETVAL_LONG((zend_long)retval);
  1606. } else {
  1607. RETVAL_FALSE;
  1608. }
  1609. }
  1610. /* }}} */
  1611. /* {{{ Finds position of last occurrence of needle within part of haystack beginning with offset */
  1612. PHP_FUNCTION(iconv_strrpos)
  1613. {
  1614. const char *charset = NULL;
  1615. size_t charset_len;
  1616. zend_string *haystk;
  1617. zend_string *ndl;
  1618. php_iconv_err_t err;
  1619. size_t retval;
  1620. if (zend_parse_parameters(ZEND_NUM_ARGS(), "SS|s!",
  1621. &haystk, &ndl,
  1622. &charset, &charset_len) == FAILURE) {
  1623. RETURN_THROWS();
  1624. }
  1625. if (ZSTR_LEN(ndl) < 1) {
  1626. RETURN_FALSE;
  1627. }
  1628. if (charset == NULL) {
  1629. charset = get_internal_encoding();
  1630. } else if (charset_len >= ICONV_CSNMAXLEN) {
  1631. php_error_docref(NULL, E_WARNING, "Encoding parameter exceeds the maximum allowed length of %d characters", ICONV_CSNMAXLEN);
  1632. RETURN_FALSE;
  1633. }
  1634. err = _php_iconv_strpos(
  1635. &retval, ZSTR_VAL(haystk), ZSTR_LEN(haystk), ZSTR_VAL(ndl), ZSTR_LEN(ndl),
  1636. /* offset */ 0, charset, /* reserve */ true);
  1637. _php_iconv_show_error(err, GENERIC_SUPERSET_NAME, charset);
  1638. if (err == PHP_ICONV_ERR_SUCCESS && retval != (size_t)-1) {
  1639. RETVAL_LONG((zend_long)retval);
  1640. } else {
  1641. RETVAL_FALSE;
  1642. }
  1643. }
  1644. /* }}} */
  1645. /* {{{ Composes a mime header field with field_name and field_value in a specified scheme */
  1646. PHP_FUNCTION(iconv_mime_encode)
  1647. {
  1648. zend_string *field_name = NULL;
  1649. zend_string *field_value = NULL;
  1650. zend_string *tmp_str = NULL;
  1651. zval *pref = NULL;
  1652. smart_str retval = {0};
  1653. php_iconv_err_t err;
  1654. const char *in_charset = get_internal_encoding();
  1655. const char *o

Large files files are truncated, but you can click here to view the full file