xml.c 49 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 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. | Authors: Stig Sæther Bakken <ssb@php.net> |
  16. | Thies C. Arntzen <thies@thieso.net> |
  17. | Sterling Hughes <sterling@php.net> |
  18. +----------------------------------------------------------------------+
  19. */
  20. #define IS_EXT_MODULE
  21. #ifdef HAVE_CONFIG_H
  22. #include "config.h"
  23. #endif
  24. #include "php.h"
  25. #define PHP_XML_INTERNAL
  26. #include "zend_variables.h"
  27. #include "ext/standard/php_string.h"
  28. #include "ext/standard/info.h"
  29. #include "ext/standard/html.h"
  30. #if HAVE_XML
  31. #include "php_xml.h"
  32. # include "ext/standard/head.h"
  33. #ifdef LIBXML_EXPAT_COMPAT
  34. #include "ext/libxml/php_libxml.h"
  35. #endif
  36. /* Short-term TODO list:
  37. * - Implement XML_ExternalEntityParserCreate()
  38. * - XML_SetCommentHandler
  39. * - XML_SetCdataSectionHandler
  40. * - XML_SetParamEntityParsing
  41. */
  42. /* Long-term TODO list:
  43. * - Fix the expat library so you can install your own memory manager
  44. * functions
  45. */
  46. /* Known bugs:
  47. * - Weird things happen with <![CDATA[]]> sections.
  48. */
  49. ZEND_DECLARE_MODULE_GLOBALS(xml)
  50. /* {{{ dynamically loadable module stuff */
  51. #ifdef COMPILE_DL_XML
  52. #ifdef ZTS
  53. ZEND_TSRMLS_CACHE_DEFINE()
  54. #endif
  55. ZEND_GET_MODULE(xml)
  56. #endif /* COMPILE_DL_XML */
  57. /* }}} */
  58. #define SKIP_TAGSTART(str) ((str) + (parser->toffset > (int)strlen(str) ? strlen(str) : parser->toffset))
  59. /* {{{ function prototypes */
  60. PHP_MINIT_FUNCTION(xml);
  61. PHP_MINFO_FUNCTION(xml);
  62. static PHP_GINIT_FUNCTION(xml);
  63. static void xml_parser_dtor(zend_resource *rsrc);
  64. static void xml_set_handler(zval *, zval *);
  65. inline static unsigned short xml_encode_iso_8859_1(unsigned char);
  66. inline static char xml_decode_iso_8859_1(unsigned short);
  67. inline static unsigned short xml_encode_us_ascii(unsigned char);
  68. inline static char xml_decode_us_ascii(unsigned short);
  69. static void xml_call_handler(xml_parser *, zval *, zend_function *, int, zval *, zval *);
  70. static void _xml_xmlchar_zval(const XML_Char *, int, const XML_Char *, zval *);
  71. static int _xml_xmlcharlen(const XML_Char *);
  72. static void _xml_add_to_info(xml_parser *parser,char *name);
  73. inline static zend_string *_xml_decode_tag(xml_parser *parser, const char *tag);
  74. void _xml_startElementHandler(void *, const XML_Char *, const XML_Char **);
  75. void _xml_endElementHandler(void *, const XML_Char *);
  76. void _xml_characterDataHandler(void *, const XML_Char *, int);
  77. void _xml_processingInstructionHandler(void *, const XML_Char *, const XML_Char *);
  78. void _xml_defaultHandler(void *, const XML_Char *, int);
  79. void _xml_unparsedEntityDeclHandler(void *, const XML_Char *, const XML_Char *, const XML_Char *, const XML_Char *, const XML_Char *);
  80. void _xml_notationDeclHandler(void *, const XML_Char *, const XML_Char *, const XML_Char *, const XML_Char *);
  81. int _xml_externalEntityRefHandler(XML_Parser, const XML_Char *, const XML_Char *, const XML_Char *, const XML_Char *);
  82. void _xml_startNamespaceDeclHandler(void *, const XML_Char *, const XML_Char *);
  83. void _xml_endNamespaceDeclHandler(void *, const XML_Char *);
  84. /* }}} */
  85. /* {{{ extension definition structures */
  86. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parser_create, 0, 0, 0)
  87. ZEND_ARG_INFO(0, encoding)
  88. ZEND_END_ARG_INFO()
  89. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parser_create_ns, 0, 0, 0)
  90. ZEND_ARG_INFO(0, encoding)
  91. ZEND_ARG_INFO(0, sep)
  92. ZEND_END_ARG_INFO()
  93. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_object, 0, 0, 2)
  94. ZEND_ARG_INFO(0, parser)
  95. ZEND_ARG_INFO(0, obj)
  96. ZEND_END_ARG_INFO()
  97. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_element_handler, 0, 0, 3)
  98. ZEND_ARG_INFO(0, parser)
  99. ZEND_ARG_INFO(0, shdl)
  100. ZEND_ARG_INFO(0, ehdl)
  101. ZEND_END_ARG_INFO()
  102. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_character_data_handler, 0, 0, 2)
  103. ZEND_ARG_INFO(0, parser)
  104. ZEND_ARG_INFO(0, hdl)
  105. ZEND_END_ARG_INFO()
  106. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_processing_instruction_handler, 0, 0, 2)
  107. ZEND_ARG_INFO(0, parser)
  108. ZEND_ARG_INFO(0, hdl)
  109. ZEND_END_ARG_INFO()
  110. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_default_handler, 0, 0, 2)
  111. ZEND_ARG_INFO(0, parser)
  112. ZEND_ARG_INFO(0, hdl)
  113. ZEND_END_ARG_INFO()
  114. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_unparsed_entity_decl_handler, 0, 0, 2)
  115. ZEND_ARG_INFO(0, parser)
  116. ZEND_ARG_INFO(0, hdl)
  117. ZEND_END_ARG_INFO()
  118. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_notation_decl_handler, 0, 0, 2)
  119. ZEND_ARG_INFO(0, parser)
  120. ZEND_ARG_INFO(0, hdl)
  121. ZEND_END_ARG_INFO()
  122. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_external_entity_ref_handler, 0, 0, 2)
  123. ZEND_ARG_INFO(0, parser)
  124. ZEND_ARG_INFO(0, hdl)
  125. ZEND_END_ARG_INFO()
  126. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_start_namespace_decl_handler, 0, 0, 2)
  127. ZEND_ARG_INFO(0, parser)
  128. ZEND_ARG_INFO(0, hdl)
  129. ZEND_END_ARG_INFO()
  130. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_set_end_namespace_decl_handler, 0, 0, 2)
  131. ZEND_ARG_INFO(0, parser)
  132. ZEND_ARG_INFO(0, hdl)
  133. ZEND_END_ARG_INFO()
  134. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parse, 0, 0, 2)
  135. ZEND_ARG_INFO(0, parser)
  136. ZEND_ARG_INFO(0, data)
  137. ZEND_ARG_INFO(0, isfinal)
  138. ZEND_END_ARG_INFO()
  139. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parse_into_struct, 0, 0, 3)
  140. ZEND_ARG_INFO(0, parser)
  141. ZEND_ARG_INFO(0, data)
  142. ZEND_ARG_INFO(1, values)
  143. ZEND_ARG_INFO(1, index)
  144. ZEND_END_ARG_INFO()
  145. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_get_error_code, 0, 0, 1)
  146. ZEND_ARG_INFO(0, parser)
  147. ZEND_END_ARG_INFO()
  148. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_error_string, 0, 0, 1)
  149. ZEND_ARG_INFO(0, code)
  150. ZEND_END_ARG_INFO()
  151. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_get_current_line_number, 0, 0, 1)
  152. ZEND_ARG_INFO(0, parser)
  153. ZEND_END_ARG_INFO()
  154. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_get_current_column_number, 0, 0, 1)
  155. ZEND_ARG_INFO(0, parser)
  156. ZEND_END_ARG_INFO()
  157. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_get_current_byte_index, 0, 0, 1)
  158. ZEND_ARG_INFO(0, parser)
  159. ZEND_END_ARG_INFO()
  160. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parser_free, 0, 0, 1)
  161. ZEND_ARG_INFO(0, parser)
  162. ZEND_END_ARG_INFO()
  163. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parser_set_option, 0, 0, 3)
  164. ZEND_ARG_INFO(0, parser)
  165. ZEND_ARG_INFO(0, option)
  166. ZEND_ARG_INFO(0, value)
  167. ZEND_END_ARG_INFO()
  168. ZEND_BEGIN_ARG_INFO_EX(arginfo_xml_parser_get_option, 0, 0, 2)
  169. ZEND_ARG_INFO(0, parser)
  170. ZEND_ARG_INFO(0, option)
  171. ZEND_END_ARG_INFO()
  172. static const zend_function_entry xml_functions[] = {
  173. PHP_FE(xml_parser_create, arginfo_xml_parser_create)
  174. PHP_FE(xml_parser_create_ns, arginfo_xml_parser_create_ns)
  175. PHP_FE(xml_set_object, arginfo_xml_set_object)
  176. PHP_FE(xml_set_element_handler, arginfo_xml_set_element_handler)
  177. PHP_FE(xml_set_character_data_handler, arginfo_xml_set_character_data_handler)
  178. PHP_FE(xml_set_processing_instruction_handler, arginfo_xml_set_processing_instruction_handler)
  179. PHP_FE(xml_set_default_handler, arginfo_xml_set_default_handler)
  180. PHP_FE(xml_set_unparsed_entity_decl_handler,arginfo_xml_set_unparsed_entity_decl_handler)
  181. PHP_FE(xml_set_notation_decl_handler, arginfo_xml_set_notation_decl_handler)
  182. PHP_FE(xml_set_external_entity_ref_handler, arginfo_xml_set_external_entity_ref_handler)
  183. PHP_FE(xml_set_start_namespace_decl_handler,arginfo_xml_set_start_namespace_decl_handler)
  184. PHP_FE(xml_set_end_namespace_decl_handler, arginfo_xml_set_end_namespace_decl_handler)
  185. PHP_FE(xml_parse, arginfo_xml_parse)
  186. PHP_FE(xml_parse_into_struct, arginfo_xml_parse_into_struct)
  187. PHP_FE(xml_get_error_code, arginfo_xml_get_error_code)
  188. PHP_FE(xml_error_string, arginfo_xml_error_string)
  189. PHP_FE(xml_get_current_line_number, arginfo_xml_get_current_line_number)
  190. PHP_FE(xml_get_current_column_number, arginfo_xml_get_current_column_number)
  191. PHP_FE(xml_get_current_byte_index, arginfo_xml_get_current_byte_index)
  192. PHP_FE(xml_parser_free, arginfo_xml_parser_free)
  193. PHP_FE(xml_parser_set_option, arginfo_xml_parser_set_option)
  194. PHP_FE(xml_parser_get_option, arginfo_xml_parser_get_option)
  195. PHP_FE_END
  196. };
  197. #ifdef LIBXML_EXPAT_COMPAT
  198. static const zend_module_dep xml_deps[] = {
  199. ZEND_MOD_REQUIRED("libxml")
  200. ZEND_MOD_END
  201. };
  202. #endif
  203. zend_module_entry xml_module_entry = {
  204. #ifdef LIBXML_EXPAT_COMPAT
  205. STANDARD_MODULE_HEADER_EX, NULL,
  206. xml_deps,
  207. #else
  208. STANDARD_MODULE_HEADER,
  209. #endif
  210. "xml", /* extension name */
  211. xml_functions, /* extension function list */
  212. PHP_MINIT(xml), /* extension-wide startup function */
  213. NULL, /* extension-wide shutdown function */
  214. NULL, /* per-request startup function */
  215. NULL, /* per-request shutdown function */
  216. PHP_MINFO(xml), /* information function */
  217. PHP_XML_VERSION,
  218. PHP_MODULE_GLOBALS(xml), /* globals descriptor */
  219. PHP_GINIT(xml), /* globals ctor */
  220. NULL, /* globals dtor */
  221. NULL, /* post deactivate */
  222. STANDARD_MODULE_PROPERTIES_EX
  223. };
  224. /* All the encoding functions are set to NULL right now, since all
  225. * the encoding is currently done internally by expat/xmltok.
  226. */
  227. const xml_encoding xml_encodings[] = {
  228. { (XML_Char *)"ISO-8859-1", xml_decode_iso_8859_1, xml_encode_iso_8859_1 },
  229. { (XML_Char *)"US-ASCII", xml_decode_us_ascii, xml_encode_us_ascii },
  230. { (XML_Char *)"UTF-8", NULL, NULL },
  231. { (XML_Char *)NULL, NULL, NULL }
  232. };
  233. static XML_Memory_Handling_Suite php_xml_mem_hdlrs;
  234. /* True globals, no need for thread safety */
  235. static int le_xml_parser;
  236. /* }}} */
  237. /* {{{ startup, shutdown and info functions */
  238. static PHP_GINIT_FUNCTION(xml)
  239. {
  240. #if defined(COMPILE_DL_XML) && defined(ZTS)
  241. ZEND_TSRMLS_CACHE_UPDATE();
  242. #endif
  243. xml_globals->default_encoding = (XML_Char*)"UTF-8";
  244. }
  245. static void *php_xml_malloc_wrapper(size_t sz)
  246. {
  247. return emalloc(sz);
  248. }
  249. static void *php_xml_realloc_wrapper(void *ptr, size_t sz)
  250. {
  251. return erealloc(ptr, sz);
  252. }
  253. static void php_xml_free_wrapper(void *ptr)
  254. {
  255. if (ptr != NULL) {
  256. efree(ptr);
  257. }
  258. }
  259. PHP_MINIT_FUNCTION(xml)
  260. {
  261. le_xml_parser = zend_register_list_destructors_ex(xml_parser_dtor, NULL, "xml", module_number);
  262. REGISTER_LONG_CONSTANT("XML_ERROR_NONE", XML_ERROR_NONE, CONST_CS|CONST_PERSISTENT);
  263. REGISTER_LONG_CONSTANT("XML_ERROR_NO_MEMORY", XML_ERROR_NO_MEMORY, CONST_CS|CONST_PERSISTENT);
  264. REGISTER_LONG_CONSTANT("XML_ERROR_SYNTAX", XML_ERROR_SYNTAX, CONST_CS|CONST_PERSISTENT);
  265. REGISTER_LONG_CONSTANT("XML_ERROR_NO_ELEMENTS", XML_ERROR_NO_ELEMENTS, CONST_CS|CONST_PERSISTENT);
  266. REGISTER_LONG_CONSTANT("XML_ERROR_INVALID_TOKEN", XML_ERROR_INVALID_TOKEN, CONST_CS|CONST_PERSISTENT);
  267. REGISTER_LONG_CONSTANT("XML_ERROR_UNCLOSED_TOKEN", XML_ERROR_UNCLOSED_TOKEN, CONST_CS|CONST_PERSISTENT);
  268. REGISTER_LONG_CONSTANT("XML_ERROR_PARTIAL_CHAR", XML_ERROR_PARTIAL_CHAR, CONST_CS|CONST_PERSISTENT);
  269. REGISTER_LONG_CONSTANT("XML_ERROR_TAG_MISMATCH", XML_ERROR_TAG_MISMATCH, CONST_CS|CONST_PERSISTENT);
  270. REGISTER_LONG_CONSTANT("XML_ERROR_DUPLICATE_ATTRIBUTE", XML_ERROR_DUPLICATE_ATTRIBUTE, CONST_CS|CONST_PERSISTENT);
  271. REGISTER_LONG_CONSTANT("XML_ERROR_JUNK_AFTER_DOC_ELEMENT", XML_ERROR_JUNK_AFTER_DOC_ELEMENT, CONST_CS|CONST_PERSISTENT);
  272. REGISTER_LONG_CONSTANT("XML_ERROR_PARAM_ENTITY_REF", XML_ERROR_PARAM_ENTITY_REF, CONST_CS|CONST_PERSISTENT);
  273. REGISTER_LONG_CONSTANT("XML_ERROR_UNDEFINED_ENTITY", XML_ERROR_UNDEFINED_ENTITY, CONST_CS|CONST_PERSISTENT);
  274. REGISTER_LONG_CONSTANT("XML_ERROR_RECURSIVE_ENTITY_REF", XML_ERROR_RECURSIVE_ENTITY_REF, CONST_CS|CONST_PERSISTENT);
  275. REGISTER_LONG_CONSTANT("XML_ERROR_ASYNC_ENTITY", XML_ERROR_ASYNC_ENTITY, CONST_CS|CONST_PERSISTENT);
  276. REGISTER_LONG_CONSTANT("XML_ERROR_BAD_CHAR_REF", XML_ERROR_BAD_CHAR_REF, CONST_CS|CONST_PERSISTENT);
  277. REGISTER_LONG_CONSTANT("XML_ERROR_BINARY_ENTITY_REF", XML_ERROR_BINARY_ENTITY_REF, CONST_CS|CONST_PERSISTENT);
  278. REGISTER_LONG_CONSTANT("XML_ERROR_ATTRIBUTE_EXTERNAL_ENTITY_REF", XML_ERROR_ATTRIBUTE_EXTERNAL_ENTITY_REF, CONST_CS|CONST_PERSISTENT);
  279. REGISTER_LONG_CONSTANT("XML_ERROR_MISPLACED_XML_PI", XML_ERROR_MISPLACED_XML_PI, CONST_CS|CONST_PERSISTENT);
  280. REGISTER_LONG_CONSTANT("XML_ERROR_UNKNOWN_ENCODING", XML_ERROR_UNKNOWN_ENCODING, CONST_CS|CONST_PERSISTENT);
  281. REGISTER_LONG_CONSTANT("XML_ERROR_INCORRECT_ENCODING", XML_ERROR_INCORRECT_ENCODING, CONST_CS|CONST_PERSISTENT);
  282. REGISTER_LONG_CONSTANT("XML_ERROR_UNCLOSED_CDATA_SECTION", XML_ERROR_UNCLOSED_CDATA_SECTION, CONST_CS|CONST_PERSISTENT);
  283. REGISTER_LONG_CONSTANT("XML_ERROR_EXTERNAL_ENTITY_HANDLING", XML_ERROR_EXTERNAL_ENTITY_HANDLING, CONST_CS|CONST_PERSISTENT);
  284. REGISTER_LONG_CONSTANT("XML_OPTION_CASE_FOLDING", PHP_XML_OPTION_CASE_FOLDING, CONST_CS|CONST_PERSISTENT);
  285. REGISTER_LONG_CONSTANT("XML_OPTION_TARGET_ENCODING", PHP_XML_OPTION_TARGET_ENCODING, CONST_CS|CONST_PERSISTENT);
  286. REGISTER_LONG_CONSTANT("XML_OPTION_SKIP_TAGSTART", PHP_XML_OPTION_SKIP_TAGSTART, CONST_CS|CONST_PERSISTENT);
  287. REGISTER_LONG_CONSTANT("XML_OPTION_SKIP_WHITE", PHP_XML_OPTION_SKIP_WHITE, CONST_CS|CONST_PERSISTENT);
  288. /* this object should not be pre-initialised at compile time,
  289. as the order of members may vary */
  290. php_xml_mem_hdlrs.malloc_fcn = php_xml_malloc_wrapper;
  291. php_xml_mem_hdlrs.realloc_fcn = php_xml_realloc_wrapper;
  292. php_xml_mem_hdlrs.free_fcn = php_xml_free_wrapper;
  293. #ifdef LIBXML_EXPAT_COMPAT
  294. REGISTER_STRING_CONSTANT("XML_SAX_IMPL", "libxml", CONST_CS|CONST_PERSISTENT);
  295. #else
  296. REGISTER_STRING_CONSTANT("XML_SAX_IMPL", "expat", CONST_CS|CONST_PERSISTENT);
  297. #endif
  298. return SUCCESS;
  299. }
  300. PHP_MINFO_FUNCTION(xml)
  301. {
  302. php_info_print_table_start();
  303. php_info_print_table_row(2, "XML Support", "active");
  304. php_info_print_table_row(2, "XML Namespace Support", "active");
  305. #if defined(LIBXML_DOTTED_VERSION) && defined(LIBXML_EXPAT_COMPAT)
  306. php_info_print_table_row(2, "libxml2 Version", LIBXML_DOTTED_VERSION);
  307. #else
  308. php_info_print_table_row(2, "EXPAT Version", XML_ExpatVersion());
  309. #endif
  310. php_info_print_table_end();
  311. }
  312. /* }}} */
  313. /* {{{ extension-internal functions */
  314. static void _xml_xmlchar_zval(const XML_Char *s, int len, const XML_Char *encoding, zval *ret)
  315. {
  316. if (s == NULL) {
  317. ZVAL_FALSE(ret);
  318. return;
  319. }
  320. if (len == 0) {
  321. len = _xml_xmlcharlen(s);
  322. }
  323. ZVAL_STR(ret, xml_utf8_decode(s, len, encoding));
  324. }
  325. /* }}} */
  326. /* {{{ xml_parser_dtor() */
  327. static void xml_parser_dtor(zend_resource *rsrc)
  328. {
  329. xml_parser *parser = (xml_parser *)rsrc->ptr;
  330. if (parser->parser) {
  331. XML_ParserFree(parser->parser);
  332. }
  333. if (parser->ltags) {
  334. int inx;
  335. for (inx = 0; ((inx < parser->level) && (inx < XML_MAXLEVEL)); inx++)
  336. efree(parser->ltags[ inx ]);
  337. efree(parser->ltags);
  338. }
  339. if (!Z_ISUNDEF(parser->startElementHandler)) {
  340. zval_ptr_dtor(&parser->startElementHandler);
  341. }
  342. if (!Z_ISUNDEF(parser->endElementHandler)) {
  343. zval_ptr_dtor(&parser->endElementHandler);
  344. }
  345. if (!Z_ISUNDEF(parser->characterDataHandler)) {
  346. zval_ptr_dtor(&parser->characterDataHandler);
  347. }
  348. if (!Z_ISUNDEF(parser->processingInstructionHandler)) {
  349. zval_ptr_dtor(&parser->processingInstructionHandler);
  350. }
  351. if (!Z_ISUNDEF(parser->defaultHandler)) {
  352. zval_ptr_dtor(&parser->defaultHandler);
  353. }
  354. if (!Z_ISUNDEF(parser->unparsedEntityDeclHandler)) {
  355. zval_ptr_dtor(&parser->unparsedEntityDeclHandler);
  356. }
  357. if (!Z_ISUNDEF(parser->notationDeclHandler)) {
  358. zval_ptr_dtor(&parser->notationDeclHandler);
  359. }
  360. if (!Z_ISUNDEF(parser->externalEntityRefHandler)) {
  361. zval_ptr_dtor(&parser->externalEntityRefHandler);
  362. }
  363. if (!Z_ISUNDEF(parser->unknownEncodingHandler)) {
  364. zval_ptr_dtor(&parser->unknownEncodingHandler);
  365. }
  366. if (!Z_ISUNDEF(parser->startNamespaceDeclHandler)) {
  367. zval_ptr_dtor(&parser->startNamespaceDeclHandler);
  368. }
  369. if (!Z_ISUNDEF(parser->endNamespaceDeclHandler)) {
  370. zval_ptr_dtor(&parser->endNamespaceDeclHandler);
  371. }
  372. if (parser->baseURI) {
  373. efree(parser->baseURI);
  374. }
  375. if (!Z_ISUNDEF(parser->object)) {
  376. zval_ptr_dtor(&parser->object);
  377. }
  378. efree(parser);
  379. }
  380. /* }}} */
  381. /* {{{ xml_set_handler() */
  382. static void xml_set_handler(zval *handler, zval *data)
  383. {
  384. /* If we have already a handler, release it */
  385. if (handler) {
  386. zval_ptr_dtor(handler);
  387. }
  388. /* IS_ARRAY might indicate that we're using array($obj, 'method') syntax */
  389. if (Z_TYPE_P(data) != IS_ARRAY && Z_TYPE_P(data) != IS_OBJECT) {
  390. convert_to_string_ex(data);
  391. if (Z_STRLEN_P(data) == 0) {
  392. ZVAL_UNDEF(handler);
  393. return;
  394. }
  395. }
  396. ZVAL_COPY(handler, data);
  397. }
  398. /* }}} */
  399. /* {{{ xml_call_handler() */
  400. static void xml_call_handler(xml_parser *parser, zval *handler, zend_function *function_ptr, int argc, zval *argv, zval *retval)
  401. {
  402. int i;
  403. ZVAL_UNDEF(retval);
  404. if (parser && handler && !EG(exception)) {
  405. int result;
  406. zend_fcall_info fci;
  407. fci.size = sizeof(fci);
  408. ZVAL_COPY_VALUE(&fci.function_name, handler);
  409. fci.object = Z_OBJ(parser->object);
  410. fci.retval = retval;
  411. fci.param_count = argc;
  412. fci.params = argv;
  413. fci.no_separation = 0;
  414. /*fci.function_handler_cache = &function_ptr;*/
  415. result = zend_call_function(&fci, NULL);
  416. if (result == FAILURE) {
  417. zval *method;
  418. zval *obj;
  419. if (Z_TYPE_P(handler) == IS_STRING) {
  420. php_error_docref(NULL, E_WARNING, "Unable to call handler %s()", Z_STRVAL_P(handler));
  421. } else if (Z_TYPE_P(handler) == IS_ARRAY &&
  422. (obj = zend_hash_index_find(Z_ARRVAL_P(handler), 0)) != NULL &&
  423. (method = zend_hash_index_find(Z_ARRVAL_P(handler), 1)) != NULL &&
  424. Z_TYPE_P(obj) == IS_OBJECT &&
  425. Z_TYPE_P(method) == IS_STRING) {
  426. php_error_docref(NULL, E_WARNING, "Unable to call handler %s::%s()", ZSTR_VAL(Z_OBJCE_P(obj)->name), Z_STRVAL_P(method));
  427. } else
  428. php_error_docref(NULL, E_WARNING, "Unable to call handler");
  429. }
  430. }
  431. for (i = 0; i < argc; i++) {
  432. zval_ptr_dtor(&argv[i]);
  433. }
  434. }
  435. /* }}} */
  436. /* {{{ xml_encode_iso_8859_1() */
  437. inline static unsigned short xml_encode_iso_8859_1(unsigned char c)
  438. {
  439. return (unsigned short)c;
  440. }
  441. /* }}} */
  442. /* {{{ xml_decode_iso_8859_1() */
  443. inline static char xml_decode_iso_8859_1(unsigned short c)
  444. {
  445. return (char)(c > 0xff ? '?' : c);
  446. }
  447. /* }}} */
  448. /* {{{ xml_encode_us_ascii() */
  449. inline static unsigned short xml_encode_us_ascii(unsigned char c)
  450. {
  451. return (unsigned short)c;
  452. }
  453. /* }}} */
  454. /* {{{ xml_decode_us_ascii() */
  455. inline static char xml_decode_us_ascii(unsigned short c)
  456. {
  457. return (char)(c > 0x7f ? '?' : c);
  458. }
  459. /* }}} */
  460. /* {{{ xml_get_encoding() */
  461. static const xml_encoding *xml_get_encoding(const XML_Char *name)
  462. {
  463. const xml_encoding *enc = &xml_encodings[0];
  464. while (enc && enc->name) {
  465. if (strcasecmp((char *)name, (char *)enc->name) == 0) {
  466. return enc;
  467. }
  468. enc++;
  469. }
  470. return NULL;
  471. }
  472. /* }}} */
  473. /* {{{ xml_utf8_encode() */
  474. PHP_XML_API zend_string *xml_utf8_encode(const char *s, size_t len, const XML_Char *encoding)
  475. {
  476. size_t pos = len;
  477. zend_string *str;
  478. unsigned int c;
  479. unsigned short (*encoder)(unsigned char) = NULL;
  480. const xml_encoding *enc = xml_get_encoding(encoding);
  481. if (enc) {
  482. encoder = enc->encoding_function;
  483. } else {
  484. /* If the target encoding was unknown, fail */
  485. return NULL;
  486. }
  487. if (encoder == NULL) {
  488. /* If no encoder function was specified, return the data as-is.
  489. */
  490. str = zend_string_init(s, len, 0);
  491. return str;
  492. }
  493. /* This is the theoretical max (will never get beyond len * 2 as long
  494. * as we are converting from single-byte characters, though) */
  495. str = zend_string_safe_alloc(len, 4, 0, 0);
  496. ZSTR_LEN(str) = 0;
  497. while (pos > 0) {
  498. c = encoder ? encoder((unsigned char)(*s)) : (unsigned short)(*s);
  499. if (c < 0x80) {
  500. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (char) c;
  501. } else if (c < 0x800) {
  502. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xc0 | (c >> 6));
  503. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0x80 | (c & 0x3f));
  504. } else if (c < 0x10000) {
  505. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xe0 | (c >> 12));
  506. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xc0 | ((c >> 6) & 0x3f));
  507. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0x80 | (c & 0x3f));
  508. } else if (c < 0x200000) {
  509. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xf0 | (c >> 18));
  510. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xe0 | ((c >> 12) & 0x3f));
  511. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0xc0 | ((c >> 6) & 0x3f));
  512. ZSTR_VAL(str)[ZSTR_LEN(str)++] = (0x80 | (c & 0x3f));
  513. }
  514. pos--;
  515. s++;
  516. }
  517. ZSTR_VAL(str)[ZSTR_LEN(str)] = '\0';
  518. str = zend_string_truncate(str, ZSTR_LEN(str), 0);
  519. return str;
  520. }
  521. /* }}} */
  522. /* {{{ xml_utf8_decode() */
  523. PHP_XML_API zend_string *xml_utf8_decode(const XML_Char *s, size_t len, const XML_Char *encoding)
  524. {
  525. size_t pos = 0;
  526. unsigned int c;
  527. char (*decoder)(unsigned short) = NULL;
  528. const xml_encoding *enc = xml_get_encoding(encoding);
  529. zend_string *str;
  530. if (enc) {
  531. decoder = enc->decoding_function;
  532. }
  533. if (decoder == NULL) {
  534. /* If the target encoding was unknown, or no decoder function
  535. * was specified, return the UTF-8-encoded data as-is.
  536. */
  537. str = zend_string_init((char *)s, len, 0);
  538. return str;
  539. }
  540. str = zend_string_alloc(len, 0);
  541. ZSTR_LEN(str) = 0;
  542. while (pos < len) {
  543. int status = FAILURE;
  544. c = php_next_utf8_char((const unsigned char*)s, (size_t) len, &pos, &status);
  545. if (status == FAILURE || c > 0xFFU) {
  546. c = '?';
  547. }
  548. ZSTR_VAL(str)[ZSTR_LEN(str)++] = decoder ? (unsigned int)decoder(c) : c;
  549. }
  550. ZSTR_VAL(str)[ZSTR_LEN(str)] = '\0';
  551. if (ZSTR_LEN(str) < len) {
  552. str = zend_string_truncate(str, ZSTR_LEN(str), 0);
  553. }
  554. return str;
  555. }
  556. /* }}} */
  557. /* {{{ _xml_xmlcharlen() */
  558. static int _xml_xmlcharlen(const XML_Char *s)
  559. {
  560. int len = 0;
  561. while (*s) {
  562. len++;
  563. s++;
  564. }
  565. return len;
  566. }
  567. /* }}} */
  568. /* {{{ _xml_zval_strdup() */
  569. PHP_XML_API char *_xml_zval_strdup(zval *val)
  570. {
  571. if (Z_TYPE_P(val) == IS_STRING) {
  572. char *buf = emalloc(Z_STRLEN_P(val) + 1);
  573. memcpy(buf, Z_STRVAL_P(val), Z_STRLEN_P(val));
  574. buf[Z_STRLEN_P(val)] = '\0';
  575. return buf;
  576. }
  577. return NULL;
  578. }
  579. /* }}} */
  580. /* {{{ _xml_add_to_info() */
  581. static void _xml_add_to_info(xml_parser *parser,char *name)
  582. {
  583. zval *element;
  584. if (Z_ISUNDEF(parser->info)) {
  585. return;
  586. }
  587. if ((element = zend_hash_str_find(Z_ARRVAL(parser->info), name, strlen(name))) == NULL) {
  588. zval values;
  589. array_init(&values);
  590. element = zend_hash_str_update(Z_ARRVAL(parser->info), name, strlen(name), &values);
  591. }
  592. add_next_index_long(element, parser->curtag);
  593. parser->curtag++;
  594. }
  595. /* }}} */
  596. /* {{{ _xml_decode_tag() */
  597. static zend_string *_xml_decode_tag(xml_parser *parser, const char *tag)
  598. {
  599. zend_string *str;
  600. str = xml_utf8_decode((const XML_Char *)tag, strlen(tag), parser->target_encoding);
  601. if (parser->case_folding) {
  602. php_strtoupper(ZSTR_VAL(str), ZSTR_LEN(str));
  603. }
  604. return str;
  605. }
  606. /* }}} */
  607. /* {{{ _xml_startElementHandler() */
  608. void _xml_startElementHandler(void *userData, const XML_Char *name, const XML_Char **attributes)
  609. {
  610. xml_parser *parser = (xml_parser *)userData;
  611. const char **attrs = (const char **) attributes;
  612. zend_string *att, *tag_name, *val;
  613. zval retval, args[3];
  614. if (parser) {
  615. parser->level++;
  616. tag_name = _xml_decode_tag(parser, (const char *)name);
  617. if (!Z_ISUNDEF(parser->startElementHandler)) {
  618. ZVAL_COPY(&args[0], &parser->index);
  619. ZVAL_STRING(&args[1], SKIP_TAGSTART(ZSTR_VAL(tag_name)));
  620. array_init(&args[2]);
  621. while (attributes && *attributes) {
  622. zval tmp;
  623. att = _xml_decode_tag(parser, (const char *)attributes[0]);
  624. val = xml_utf8_decode(attributes[1], strlen((char *)attributes[1]), parser->target_encoding);
  625. ZVAL_STR(&tmp, val);
  626. zend_symtable_update(Z_ARRVAL(args[2]), att, &tmp);
  627. attributes += 2;
  628. zend_string_release_ex(att, 0);
  629. }
  630. xml_call_handler(parser, &parser->startElementHandler, parser->startElementPtr, 3, args, &retval);
  631. zval_ptr_dtor(&retval);
  632. }
  633. if (!Z_ISUNDEF(parser->data)) {
  634. if (parser->level <= XML_MAXLEVEL) {
  635. zval tag, atr;
  636. int atcnt = 0;
  637. array_init(&tag);
  638. array_init(&atr);
  639. _xml_add_to_info(parser, ZSTR_VAL(tag_name) + parser->toffset);
  640. add_assoc_string(&tag, "tag", SKIP_TAGSTART(ZSTR_VAL(tag_name))); /* cast to avoid gcc-warning */
  641. add_assoc_string(&tag, "type", "open");
  642. add_assoc_long(&tag, "level", parser->level);
  643. parser->ltags[parser->level-1] = estrdup(ZSTR_VAL(tag_name));
  644. parser->lastwasopen = 1;
  645. attributes = (const XML_Char **) attrs;
  646. while (attributes && *attributes) {
  647. zval tmp;
  648. att = _xml_decode_tag(parser, (const char *)attributes[0]);
  649. val = xml_utf8_decode(attributes[1], strlen((char *)attributes[1]), parser->target_encoding);
  650. ZVAL_STR(&tmp, val);
  651. zend_symtable_update(Z_ARRVAL(atr), att, &tmp);
  652. atcnt++;
  653. attributes += 2;
  654. zend_string_release_ex(att, 0);
  655. }
  656. if (atcnt) {
  657. zend_hash_str_add(Z_ARRVAL(tag), "attributes", sizeof("attributes") - 1, &atr);
  658. } else {
  659. zval_ptr_dtor(&atr);
  660. }
  661. parser->ctag = zend_hash_next_index_insert(Z_ARRVAL(parser->data), &tag);
  662. } else if (parser->level == (XML_MAXLEVEL + 1)) {
  663. php_error_docref(NULL, E_WARNING, "Maximum depth exceeded - Results truncated");
  664. }
  665. }
  666. zend_string_release_ex(tag_name, 0);
  667. }
  668. }
  669. /* }}} */
  670. /* {{{ _xml_endElementHandler() */
  671. void _xml_endElementHandler(void *userData, const XML_Char *name)
  672. {
  673. xml_parser *parser = (xml_parser *)userData;
  674. if (parser) {
  675. zval retval, args[2];
  676. zend_string *tag_name = _xml_decode_tag(parser, (const char *)name);
  677. if (!Z_ISUNDEF(parser->endElementHandler)) {
  678. ZVAL_COPY(&args[0], &parser->index);
  679. ZVAL_STRING(&args[1], SKIP_TAGSTART(ZSTR_VAL(tag_name)));
  680. xml_call_handler(parser, &parser->endElementHandler, parser->endElementPtr, 2, args, &retval);
  681. zval_ptr_dtor(&retval);
  682. }
  683. if (!Z_ISUNDEF(parser->data)) {
  684. zval tag;
  685. if (parser->lastwasopen) {
  686. add_assoc_string(parser->ctag, "type", "complete");
  687. } else {
  688. array_init(&tag);
  689. _xml_add_to_info(parser, ZSTR_VAL(tag_name) + parser->toffset);
  690. add_assoc_string(&tag, "tag", SKIP_TAGSTART(ZSTR_VAL(tag_name))); /* cast to avoid gcc-warning */
  691. add_assoc_string(&tag, "type", "close");
  692. add_assoc_long(&tag, "level", parser->level);
  693. zend_hash_next_index_insert(Z_ARRVAL(parser->data), &tag);
  694. }
  695. parser->lastwasopen = 0;
  696. }
  697. zend_string_release_ex(tag_name, 0);
  698. if ((parser->ltags) && (parser->level <= XML_MAXLEVEL)) {
  699. efree(parser->ltags[parser->level-1]);
  700. }
  701. parser->level--;
  702. }
  703. }
  704. /* }}} */
  705. /* {{{ _xml_characterDataHandler() */
  706. void _xml_characterDataHandler(void *userData, const XML_Char *s, int len)
  707. {
  708. xml_parser *parser = (xml_parser *)userData;
  709. if (parser) {
  710. zval retval, args[2];
  711. if (!Z_ISUNDEF(parser->characterDataHandler)) {
  712. ZVAL_COPY(&args[0], &parser->index);
  713. _xml_xmlchar_zval(s, len, parser->target_encoding, &args[1]);
  714. xml_call_handler(parser, &parser->characterDataHandler, parser->characterDataPtr, 2, args, &retval);
  715. zval_ptr_dtor(&retval);
  716. }
  717. if (!Z_ISUNDEF(parser->data)) {
  718. size_t i;
  719. int doprint = 0;
  720. zend_string *decoded_value;
  721. decoded_value = xml_utf8_decode(s, len, parser->target_encoding);
  722. for (i = 0; i < ZSTR_LEN(decoded_value); i++) {
  723. switch (ZSTR_VAL(decoded_value)[i]) {
  724. case ' ':
  725. case '\t':
  726. case '\n':
  727. continue;
  728. default:
  729. doprint = 1;
  730. break;
  731. }
  732. if (doprint) {
  733. break;
  734. }
  735. }
  736. if (doprint || (! parser->skipwhite)) {
  737. if (parser->lastwasopen) {
  738. zval *myval;
  739. /* check if the current tag already has a value - if yes append to that! */
  740. if ((myval = zend_hash_str_find(Z_ARRVAL_P(parser->ctag), "value", sizeof("value") - 1))) {
  741. int newlen = Z_STRLEN_P(myval) + ZSTR_LEN(decoded_value);
  742. Z_STR_P(myval) = zend_string_extend(Z_STR_P(myval), newlen, 0);
  743. strncpy(Z_STRVAL_P(myval) + Z_STRLEN_P(myval) - ZSTR_LEN(decoded_value),
  744. ZSTR_VAL(decoded_value), ZSTR_LEN(decoded_value) + 1);
  745. zend_string_release_ex(decoded_value, 0);
  746. } else {
  747. add_assoc_str(parser->ctag, "value", decoded_value);
  748. }
  749. } else {
  750. zval tag;
  751. zval *curtag, *mytype, *myval;
  752. ZEND_HASH_REVERSE_FOREACH_VAL(Z_ARRVAL(parser->data), curtag) {
  753. if ((mytype = zend_hash_str_find(Z_ARRVAL_P(curtag),"type", sizeof("type") - 1))) {
  754. if (!strcmp(Z_STRVAL_P(mytype), "cdata")) {
  755. if ((myval = zend_hash_str_find(Z_ARRVAL_P(curtag), "value", sizeof("value") - 1))) {
  756. int newlen = Z_STRLEN_P(myval) + ZSTR_LEN(decoded_value);
  757. Z_STR_P(myval) = zend_string_extend(Z_STR_P(myval), newlen, 0);
  758. strncpy(Z_STRVAL_P(myval) + Z_STRLEN_P(myval) - ZSTR_LEN(decoded_value),
  759. ZSTR_VAL(decoded_value), ZSTR_LEN(decoded_value) + 1);
  760. zend_string_release_ex(decoded_value, 0);
  761. return;
  762. }
  763. }
  764. }
  765. break;
  766. } ZEND_HASH_FOREACH_END();
  767. if (parser->level <= XML_MAXLEVEL && parser->level > 0) {
  768. array_init(&tag);
  769. _xml_add_to_info(parser,SKIP_TAGSTART(parser->ltags[parser->level-1]));
  770. add_assoc_string(&tag, "tag", SKIP_TAGSTART(parser->ltags[parser->level-1]));
  771. add_assoc_str(&tag, "value", decoded_value);
  772. add_assoc_string(&tag, "type", "cdata");
  773. add_assoc_long(&tag, "level", parser->level);
  774. zend_hash_next_index_insert(Z_ARRVAL(parser->data), &tag);
  775. } else if (parser->level == (XML_MAXLEVEL + 1)) {
  776. php_error_docref(NULL, E_WARNING, "Maximum depth exceeded - Results truncated");
  777. }
  778. }
  779. } else {
  780. zend_string_release_ex(decoded_value, 0);
  781. }
  782. }
  783. }
  784. }
  785. /* }}} */
  786. /* {{{ _xml_processingInstructionHandler() */
  787. void _xml_processingInstructionHandler(void *userData, const XML_Char *target, const XML_Char *data)
  788. {
  789. xml_parser *parser = (xml_parser *)userData;
  790. if (parser && !Z_ISUNDEF(parser->processingInstructionHandler)) {
  791. zval retval, args[3];
  792. ZVAL_COPY(&args[0], &parser->index);
  793. _xml_xmlchar_zval(target, 0, parser->target_encoding, &args[1]);
  794. _xml_xmlchar_zval(data, 0, parser->target_encoding, &args[2]);
  795. xml_call_handler(parser, &parser->processingInstructionHandler, parser->processingInstructionPtr, 3, args, &retval);
  796. zval_ptr_dtor(&retval);
  797. }
  798. }
  799. /* }}} */
  800. /* {{{ _xml_defaultHandler() */
  801. void _xml_defaultHandler(void *userData, const XML_Char *s, int len)
  802. {
  803. xml_parser *parser = (xml_parser *)userData;
  804. if (parser && !Z_ISUNDEF(parser->defaultHandler)) {
  805. zval retval, args[2];
  806. ZVAL_COPY(&args[0], &parser->index);
  807. _xml_xmlchar_zval(s, len, parser->target_encoding, &args[1]);
  808. xml_call_handler(parser, &parser->defaultHandler, parser->defaultPtr, 2, args, &retval);
  809. zval_ptr_dtor(&retval);
  810. }
  811. }
  812. /* }}} */
  813. /* {{{ _xml_unparsedEntityDeclHandler() */
  814. void _xml_unparsedEntityDeclHandler(void *userData,
  815. const XML_Char *entityName,
  816. const XML_Char *base,
  817. const XML_Char *systemId,
  818. const XML_Char *publicId,
  819. const XML_Char *notationName)
  820. {
  821. xml_parser *parser = (xml_parser *)userData;
  822. if (parser && !Z_ISUNDEF(parser->unparsedEntityDeclHandler)) {
  823. zval retval, args[6];
  824. ZVAL_COPY(&args[0], &parser->index);
  825. _xml_xmlchar_zval(entityName, 0, parser->target_encoding, &args[1]);
  826. _xml_xmlchar_zval(base, 0, parser->target_encoding, &args[2]);
  827. _xml_xmlchar_zval(systemId, 0, parser->target_encoding, &args[3]);
  828. _xml_xmlchar_zval(publicId, 0, parser->target_encoding, &args[4]);
  829. _xml_xmlchar_zval(notationName, 0, parser->target_encoding, &args[5]);
  830. xml_call_handler(parser, &parser->unparsedEntityDeclHandler, parser->unparsedEntityDeclPtr, 6, args, &retval);
  831. zval_ptr_dtor(&retval);
  832. }
  833. }
  834. /* }}} */
  835. /* {{{ _xml_notationDeclHandler() */
  836. void _xml_notationDeclHandler(void *userData,
  837. const XML_Char *notationName,
  838. const XML_Char *base,
  839. const XML_Char *systemId,
  840. const XML_Char *publicId)
  841. {
  842. xml_parser *parser = (xml_parser *)userData;
  843. if (parser && !Z_ISUNDEF(parser->notationDeclHandler)) {
  844. zval retval, args[5];
  845. ZVAL_COPY(&args[0], &parser->index);
  846. _xml_xmlchar_zval(notationName, 0, parser->target_encoding, &args[1]);
  847. _xml_xmlchar_zval(base, 0, parser->target_encoding, &args[2]);
  848. _xml_xmlchar_zval(systemId, 0, parser->target_encoding, &args[3]);
  849. _xml_xmlchar_zval(publicId, 0, parser->target_encoding, &args[4]);
  850. xml_call_handler(parser, &parser->notationDeclHandler, parser->notationDeclPtr, 5, args, &retval);
  851. zval_ptr_dtor(&retval);
  852. }
  853. }
  854. /* }}} */
  855. /* {{{ _xml_externalEntityRefHandler() */
  856. int _xml_externalEntityRefHandler(XML_Parser parserPtr,
  857. const XML_Char *openEntityNames,
  858. const XML_Char *base,
  859. const XML_Char *systemId,
  860. const XML_Char *publicId)
  861. {
  862. xml_parser *parser = XML_GetUserData(parserPtr);
  863. int ret = 0; /* abort if no handler is set (should be configurable?) */
  864. if (parser && !Z_ISUNDEF(parser->externalEntityRefHandler)) {
  865. zval retval, args[5];
  866. ZVAL_COPY(&args[0], &parser->index);
  867. _xml_xmlchar_zval(openEntityNames, 0, parser->target_encoding, &args[1]);
  868. _xml_xmlchar_zval(base, 0, parser->target_encoding, &args[2]);
  869. _xml_xmlchar_zval(systemId, 0, parser->target_encoding, &args[3]);
  870. _xml_xmlchar_zval(publicId, 0, parser->target_encoding, &args[4]);
  871. xml_call_handler(parser, &parser->externalEntityRefHandler, parser->externalEntityRefPtr, 5, args, &retval);
  872. if (!Z_ISUNDEF(retval)) {
  873. convert_to_long(&retval);
  874. ret = Z_LVAL(retval);
  875. } else {
  876. ret = 0;
  877. }
  878. }
  879. return ret;
  880. }
  881. /* }}} */
  882. /* {{{ _xml_startNamespaceDeclHandler() */
  883. void _xml_startNamespaceDeclHandler(void *userData,const XML_Char *prefix, const XML_Char *uri)
  884. {
  885. xml_parser *parser = (xml_parser *)userData;
  886. if (parser && !Z_ISUNDEF(parser->startNamespaceDeclHandler)) {
  887. zval retval, args[3];
  888. ZVAL_COPY(&args[0], &parser->index);
  889. _xml_xmlchar_zval(prefix, 0, parser->target_encoding, &args[1]);
  890. _xml_xmlchar_zval(uri, 0, parser->target_encoding, &args[2]);
  891. xml_call_handler(parser, &parser->startNamespaceDeclHandler, parser->startNamespaceDeclPtr, 3, args, &retval);
  892. zval_ptr_dtor(&retval);
  893. }
  894. }
  895. /* }}} */
  896. /* {{{ _xml_endNamespaceDeclHandler() */
  897. void _xml_endNamespaceDeclHandler(void *userData, const XML_Char *prefix)
  898. {
  899. xml_parser *parser = (xml_parser *)userData;
  900. if (parser && !Z_ISUNDEF(parser->endNamespaceDeclHandler)) {
  901. zval retval, args[2];
  902. ZVAL_COPY(&args[0], &parser->index);
  903. _xml_xmlchar_zval(prefix, 0, parser->target_encoding, &args[1]);
  904. xml_call_handler(parser, &parser->endNamespaceDeclHandler, parser->endNamespaceDeclPtr, 2, args, &retval);
  905. zval_ptr_dtor(&retval);
  906. }
  907. }
  908. /* }}} */
  909. /************************* EXTENSION FUNCTIONS *************************/
  910. static void php_xml_parser_create_impl(INTERNAL_FUNCTION_PARAMETERS, int ns_support) /* {{{ */
  911. {
  912. xml_parser *parser;
  913. int auto_detect = 0;
  914. char *encoding_param = NULL;
  915. size_t encoding_param_len = 0;
  916. char *ns_param = NULL;
  917. size_t ns_param_len = 0;
  918. XML_Char *encoding;
  919. if (zend_parse_parameters(ZEND_NUM_ARGS(), (ns_support ? "|ss": "|s"), &encoding_param, &encoding_param_len, &ns_param, &ns_param_len) == FAILURE) {
  920. RETURN_FALSE;
  921. }
  922. if (encoding_param != NULL) {
  923. /* The supported encoding types are hardcoded here because
  924. * we are limited to the encodings supported by expat/xmltok.
  925. */
  926. if (encoding_param_len == 0) {
  927. encoding = XML(default_encoding);
  928. auto_detect = 1;
  929. } else if (strcasecmp(encoding_param, "ISO-8859-1") == 0) {
  930. encoding = (XML_Char*)"ISO-8859-1";
  931. } else if (strcasecmp(encoding_param, "UTF-8") == 0) {
  932. encoding = (XML_Char*)"UTF-8";
  933. } else if (strcasecmp(encoding_param, "US-ASCII") == 0) {
  934. encoding = (XML_Char*)"US-ASCII";
  935. } else {
  936. php_error_docref(NULL, E_WARNING, "unsupported source encoding \"%s\"", encoding_param);
  937. RETURN_FALSE;
  938. }
  939. } else {
  940. encoding = XML(default_encoding);
  941. }
  942. if (ns_support && ns_param == NULL){
  943. ns_param = ":";
  944. }
  945. parser = ecalloc(1, sizeof(xml_parser));
  946. parser->parser = XML_ParserCreate_MM((auto_detect ? NULL : encoding),
  947. &php_xml_mem_hdlrs, (XML_Char*)ns_param);
  948. parser->target_encoding = encoding;
  949. parser->case_folding = 1;
  950. parser->isparsing = 0;
  951. XML_SetUserData(parser->parser, parser);
  952. RETVAL_RES(zend_register_resource(parser, le_xml_parser));
  953. ZVAL_COPY_VALUE(&parser->index, return_value);
  954. }
  955. /* }}} */
  956. /* {{{ proto resource xml_parser_create([string encoding])
  957. Create an XML parser */
  958. PHP_FUNCTION(xml_parser_create)
  959. {
  960. php_xml_parser_create_impl(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  961. }
  962. /* }}} */
  963. /* {{{ proto resource xml_parser_create_ns([string encoding [, string sep]])
  964. Create an XML parser */
  965. PHP_FUNCTION(xml_parser_create_ns)
  966. {
  967. php_xml_parser_create_impl(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  968. }
  969. /* }}} */
  970. /* {{{ proto int xml_set_object(resource parser, object &obj)
  971. Set up object which should be used for callbacks */
  972. PHP_FUNCTION(xml_set_object)
  973. {
  974. xml_parser *parser;
  975. zval *pind, *mythis;
  976. if (zend_parse_parameters(ZEND_NUM_ARGS(), "ro", &pind, &mythis) == FAILURE) {
  977. return;
  978. }
  979. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  980. RETURN_FALSE;
  981. }
  982. /* please leave this commented - or ask thies@thieso.net before doing it (again) */
  983. if (!Z_ISUNDEF(parser->object)) {
  984. zval_ptr_dtor(&parser->object);
  985. }
  986. /* please leave this commented - or ask thies@thieso.net before doing it (again) */
  987. /* zval_add_ref(&parser->object); */
  988. ZVAL_COPY(&parser->object, mythis);
  989. RETVAL_TRUE;
  990. }
  991. /* }}} */
  992. /* {{{ proto int xml_set_element_handler(resource parser, string shdl, string ehdl)
  993. Set up start and end element handlers */
  994. PHP_FUNCTION(xml_set_element_handler)
  995. {
  996. xml_parser *parser;
  997. zval *pind, *shdl, *ehdl;
  998. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rzz", &pind, &shdl, &ehdl) == FAILURE) {
  999. return;
  1000. }
  1001. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1002. RETURN_FALSE;
  1003. }
  1004. xml_set_handler(&parser->startElementHandler, shdl);
  1005. xml_set_handler(&parser->endElementHandler, ehdl);
  1006. XML_SetElementHandler(parser->parser, _xml_startElementHandler, _xml_endElementHandler);
  1007. RETVAL_TRUE;
  1008. }
  1009. /* }}} */
  1010. /* {{{ proto int xml_set_character_data_handler(resource parser, string hdl)
  1011. Set up character data handler */
  1012. PHP_FUNCTION(xml_set_character_data_handler)
  1013. {
  1014. xml_parser *parser;
  1015. zval *pind, *hdl;
  1016. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1017. return;
  1018. }
  1019. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1020. RETURN_FALSE;
  1021. }
  1022. xml_set_handler(&parser->characterDataHandler, hdl);
  1023. XML_SetCharacterDataHandler(parser->parser, _xml_characterDataHandler);
  1024. RETVAL_TRUE;
  1025. }
  1026. /* }}} */
  1027. /* {{{ proto int xml_set_processing_instruction_handler(resource parser, string hdl)
  1028. Set up processing instruction (PI) handler */
  1029. PHP_FUNCTION(xml_set_processing_instruction_handler)
  1030. {
  1031. xml_parser *parser;
  1032. zval *pind, *hdl;
  1033. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1034. return;
  1035. }
  1036. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1037. RETURN_FALSE;
  1038. }
  1039. xml_set_handler(&parser->processingInstructionHandler, hdl);
  1040. XML_SetProcessingInstructionHandler(parser->parser, _xml_processingInstructionHandler);
  1041. RETVAL_TRUE;
  1042. }
  1043. /* }}} */
  1044. /* {{{ proto int xml_set_default_handler(resource parser, string hdl)
  1045. Set up default handler */
  1046. PHP_FUNCTION(xml_set_default_handler)
  1047. {
  1048. xml_parser *parser;
  1049. zval *pind, *hdl;
  1050. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1051. return;
  1052. }
  1053. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1054. RETURN_FALSE;
  1055. }
  1056. xml_set_handler(&parser->defaultHandler, hdl);
  1057. XML_SetDefaultHandler(parser->parser, _xml_defaultHandler);
  1058. RETVAL_TRUE;
  1059. }
  1060. /* }}} */
  1061. /* {{{ proto int xml_set_unparsed_entity_decl_handler(resource parser, string hdl)
  1062. Set up unparsed entity declaration handler */
  1063. PHP_FUNCTION(xml_set_unparsed_entity_decl_handler)
  1064. {
  1065. xml_parser *parser;
  1066. zval *pind, *hdl;
  1067. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1068. return;
  1069. }
  1070. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1071. RETURN_FALSE;
  1072. }
  1073. xml_set_handler(&parser->unparsedEntityDeclHandler, hdl);
  1074. XML_SetUnparsedEntityDeclHandler(parser->parser, _xml_unparsedEntityDeclHandler);
  1075. RETVAL_TRUE;
  1076. }
  1077. /* }}} */
  1078. /* {{{ proto int xml_set_notation_decl_handler(resource parser, string hdl)
  1079. Set up notation declaration handler */
  1080. PHP_FUNCTION(xml_set_notation_decl_handler)
  1081. {
  1082. xml_parser *parser;
  1083. zval *pind, *hdl;
  1084. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1085. return;
  1086. }
  1087. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1088. RETURN_FALSE;
  1089. }
  1090. xml_set_handler(&parser->notationDeclHandler, hdl);
  1091. XML_SetNotationDeclHandler(parser->parser, _xml_notationDeclHandler);
  1092. RETVAL_TRUE;
  1093. }
  1094. /* }}} */
  1095. /* {{{ proto int xml_set_external_entity_ref_handler(resource parser, string hdl)
  1096. Set up external entity reference handler */
  1097. PHP_FUNCTION(xml_set_external_entity_ref_handler)
  1098. {
  1099. xml_parser *parser;
  1100. zval *pind, *hdl;
  1101. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1102. return;
  1103. }
  1104. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1105. RETURN_FALSE;
  1106. }
  1107. xml_set_handler(&parser->externalEntityRefHandler, hdl);
  1108. XML_SetExternalEntityRefHandler(parser->parser, (void *) _xml_externalEntityRefHandler);
  1109. RETVAL_TRUE;
  1110. }
  1111. /* }}} */
  1112. /* {{{ proto int xml_set_start_namespace_decl_handler(resource parser, string hdl)
  1113. Set up character data handler */
  1114. PHP_FUNCTION(xml_set_start_namespace_decl_handler)
  1115. {
  1116. xml_parser *parser;
  1117. zval *pind, *hdl;
  1118. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1119. return;
  1120. }
  1121. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1122. RETURN_FALSE;
  1123. }
  1124. xml_set_handler(&parser->startNamespaceDeclHandler, hdl);
  1125. XML_SetStartNamespaceDeclHandler(parser->parser, _xml_startNamespaceDeclHandler);
  1126. RETVAL_TRUE;
  1127. }
  1128. /* }}} */
  1129. /* {{{ proto int xml_set_end_namespace_decl_handler(resource parser, string hdl)
  1130. Set up character data handler */
  1131. PHP_FUNCTION(xml_set_end_namespace_decl_handler)
  1132. {
  1133. xml_parser *parser;
  1134. zval *pind, *hdl;
  1135. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rz", &pind, &hdl) == FAILURE) {
  1136. return;
  1137. }
  1138. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1139. RETURN_FALSE;
  1140. }
  1141. xml_set_handler(&parser->endNamespaceDeclHandler, hdl);
  1142. XML_SetEndNamespaceDeclHandler(parser->parser, _xml_endNamespaceDeclHandler);
  1143. RETVAL_TRUE;
  1144. }
  1145. /* }}} */
  1146. /* {{{ proto int xml_parse(resource parser, string data [, bool isFinal])
  1147. Start parsing an XML document */
  1148. PHP_FUNCTION(xml_parse)
  1149. {
  1150. xml_parser *parser;
  1151. zval *pind;
  1152. char *data;
  1153. size_t data_len;
  1154. int ret;
  1155. zend_bool isFinal = 0;
  1156. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rs|b", &pind, &data, &data_len, &isFinal) == FAILURE) {
  1157. return;
  1158. }
  1159. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1160. RETURN_FALSE;
  1161. }
  1162. parser->isparsing = 1;
  1163. ret = XML_Parse(parser->parser, (XML_Char*)data, data_len, isFinal);
  1164. parser->isparsing = 0;
  1165. RETVAL_LONG(ret);
  1166. }
  1167. /* }}} */
  1168. /* {{{ proto int xml_parse_into_struct(resource parser, string data, array &values [, array &index ])
  1169. Parsing a XML document */
  1170. PHP_FUNCTION(xml_parse_into_struct)
  1171. {
  1172. xml_parser *parser;
  1173. zval *pind, *xdata, *info = NULL;
  1174. char *data;
  1175. size_t data_len;
  1176. int ret;
  1177. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rsz/|z/", &pind, &data, &data_len, &xdata, &info) == FAILURE) {
  1178. return;
  1179. }
  1180. if (info) {
  1181. zval_ptr_dtor(info);
  1182. array_init(info);
  1183. }
  1184. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1185. RETURN_FALSE;
  1186. }
  1187. zval_ptr_dtor(xdata);
  1188. array_init(xdata);
  1189. ZVAL_COPY_VALUE(&parser->data, xdata);
  1190. if (info) {
  1191. ZVAL_COPY_VALUE(&parser->info, info);
  1192. }
  1193. parser->level = 0;
  1194. parser->ltags = safe_emalloc(XML_MAXLEVEL, sizeof(char *), 0);
  1195. XML_SetElementHandler(parser->parser, _xml_startElementHandler, _xml_endElementHandler);
  1196. XML_SetCharacterDataHandler(parser->parser, _xml_characterDataHandler);
  1197. parser->isparsing = 1;
  1198. ret = XML_Parse(parser->parser, (XML_Char*)data, data_len, 1);
  1199. parser->isparsing = 0;
  1200. RETVAL_LONG(ret);
  1201. }
  1202. /* }}} */
  1203. /* {{{ proto int xml_get_error_code(resource parser)
  1204. Get XML parser error code */
  1205. PHP_FUNCTION(xml_get_error_code)
  1206. {
  1207. xml_parser *parser;
  1208. zval *pind;
  1209. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &pind) == FAILURE) {
  1210. return;
  1211. }
  1212. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1213. RETURN_FALSE;
  1214. }
  1215. RETURN_LONG((zend_long)XML_GetErrorCode(parser->parser));
  1216. }
  1217. /* }}} */
  1218. /* {{{ proto string xml_error_string(int code)
  1219. Get XML parser error string */
  1220. PHP_FUNCTION(xml_error_string)
  1221. {
  1222. zend_long code;
  1223. char *str;
  1224. if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &code) == FAILURE) {
  1225. return;
  1226. }
  1227. str = (char *)XML_ErrorString((int)code);
  1228. if (str) {
  1229. RETVAL_STRING(str);
  1230. }
  1231. }
  1232. /* }}} */
  1233. /* {{{ proto int xml_get_current_line_number(resource parser)
  1234. Get current line number for an XML parser */
  1235. PHP_FUNCTION(xml_get_current_line_number)
  1236. {
  1237. xml_parser *parser;
  1238. zval *pind;
  1239. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &pind) == FAILURE) {
  1240. return;
  1241. }
  1242. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1243. RETURN_FALSE;
  1244. }
  1245. RETVAL_LONG(XML_GetCurrentLineNumber(parser->parser));
  1246. }
  1247. /* }}} */
  1248. /* {{{ proto int xml_get_current_column_number(resource parser)
  1249. Get current column number for an XML parser */
  1250. PHP_FUNCTION(xml_get_current_column_number)
  1251. {
  1252. xml_parser *parser;
  1253. zval *pind;
  1254. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &pind) == FAILURE) {
  1255. return;
  1256. }
  1257. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1258. RETURN_FALSE;
  1259. }
  1260. RETVAL_LONG(XML_GetCurrentColumnNumber(parser->parser));
  1261. }
  1262. /* }}} */
  1263. /* {{{ proto int xml_get_current_byte_index(resource parser)
  1264. Get current byte index for an XML parser */
  1265. PHP_FUNCTION(xml_get_current_byte_index)
  1266. {
  1267. xml_parser *parser;
  1268. zval *pind;
  1269. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &pind) == FAILURE) {
  1270. return;
  1271. }
  1272. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1273. RETURN_FALSE;
  1274. }
  1275. RETVAL_LONG(XML_GetCurrentByteIndex(parser->parser));
  1276. }
  1277. /* }}} */
  1278. /* {{{ proto int xml_parser_free(resource parser)
  1279. Free an XML parser */
  1280. PHP_FUNCTION(xml_parser_free)
  1281. {
  1282. zval *pind;
  1283. xml_parser *parser;
  1284. if (zend_parse_parameters(ZEND_NUM_ARGS(), "r", &pind) == FAILURE) {
  1285. return;
  1286. }
  1287. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1288. RETURN_FALSE;
  1289. }
  1290. if (parser->isparsing == 1) {
  1291. php_error_docref(NULL, E_WARNING, "Parser cannot be freed while it is parsing.");
  1292. RETURN_FALSE;
  1293. }
  1294. if (zend_list_close(Z_RES(parser->index)) == FAILURE) {
  1295. RETURN_FALSE;
  1296. }
  1297. RETURN_TRUE;
  1298. }
  1299. /* }}} */
  1300. /* {{{ proto int xml_parser_set_option(resource parser, int option, mixed value)
  1301. Set options in an XML parser */
  1302. PHP_FUNCTION(xml_parser_set_option)
  1303. {
  1304. xml_parser *parser;
  1305. zval *pind, *val;
  1306. zend_long opt;
  1307. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rlz", &pind, &opt, &val) == FAILURE) {
  1308. return;
  1309. }
  1310. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1311. RETURN_FALSE;
  1312. }
  1313. switch (opt) {
  1314. case PHP_XML_OPTION_CASE_FOLDING:
  1315. parser->case_folding = zval_get_long(val);
  1316. break;
  1317. case PHP_XML_OPTION_SKIP_TAGSTART:
  1318. parser->toffset = zval_get_long(val);
  1319. if (parser->toffset < 0) {
  1320. php_error_docref(NULL, E_NOTICE, "tagstart ignored, because it is out of range");
  1321. parser->toffset = 0;
  1322. }
  1323. break;
  1324. case PHP_XML_OPTION_SKIP_WHITE:
  1325. parser->skipwhite = zval_get_long(val);
  1326. break;
  1327. case PHP_XML_OPTION_TARGET_ENCODING: {
  1328. const xml_encoding *enc;
  1329. convert_to_string_ex(val);
  1330. enc = xml_get_encoding((XML_Char*)Z_STRVAL_P(val));
  1331. if (enc == NULL) {
  1332. php_error_docref(NULL, E_WARNING, "Unsupported target encoding \"%s\"", Z_STRVAL_P(val));
  1333. RETURN_FALSE;
  1334. }
  1335. parser->target_encoding = enc->name;
  1336. break;
  1337. }
  1338. default:
  1339. php_error_docref(NULL, E_WARNING, "Unknown option");
  1340. RETURN_FALSE;
  1341. break;
  1342. }
  1343. RETVAL_TRUE;
  1344. }
  1345. /* }}} */
  1346. /* {{{ proto int xml_parser_get_option(resource parser, int option)
  1347. Get options from an XML parser */
  1348. PHP_FUNCTION(xml_parser_get_option)
  1349. {
  1350. xml_parser *parser;
  1351. zval *pind;
  1352. zend_long opt;
  1353. if (zend_parse_parameters(ZEND_NUM_ARGS(), "rl", &pind, &opt) == FAILURE) {
  1354. return;
  1355. }
  1356. if ((parser = (xml_parser *)zend_fetch_resource(Z_RES_P(pind), "XML Parser", le_xml_parser)) == NULL) {
  1357. RETURN_FALSE;
  1358. }
  1359. switch (opt) {
  1360. case PHP_XML_OPTION_CASE_FOLDING:
  1361. RETURN_LONG(parser->case_folding);
  1362. break;
  1363. case PHP_XML_OPTION_SKIP_TAGSTART:
  1364. RETURN_LONG(parser->toffset);
  1365. break;
  1366. case PHP_XML_OPTION_SKIP_WHITE:
  1367. RETURN_LONG(parser->skipwhite);
  1368. break;
  1369. case PHP_XML_OPTION_TARGET_ENCODING:
  1370. RETURN_STRING((char *)parser->target_encoding);
  1371. break;
  1372. default:
  1373. php_error_docref(NULL, E_WARNING, "Unknown option");
  1374. RETURN_FALSE;
  1375. break;
  1376. }
  1377. RETVAL_FALSE; /* never reached */
  1378. }
  1379. /* }}} */
  1380. #endif
  1381. /*
  1382. * Local variables:
  1383. * tab-width: 4
  1384. * c-basic-offset: 4
  1385. * End:
  1386. * vim600: sw=4 ts=4 fdm=marker
  1387. * vim<600: sw=4 ts=4
  1388. */