compat.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792
  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: Sterling Hughes <sterling@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #include "php.h"
  19. #if defined(HAVE_LIBXML) && (defined(HAVE_XML) || defined(HAVE_XMLRPC)) && !defined(HAVE_LIBEXPAT)
  20. #include "expat_compat.h"
  21. typedef struct _php_xml_ns {
  22. xmlNsPtr nsptr;
  23. int ref_count;
  24. void *next;
  25. void *prev;
  26. } php_xml_ns;
  27. #ifdef LIBXML_EXPAT_COMPAT
  28. #define IS_NS_DECL(__ns) \
  29. ((__ns) != NULL && strlen(__ns) == 5 && *(__ns) == 'x' && *((__ns)+1) == 'm' && \
  30. *((__ns)+2) == 'l' && *((__ns)+3) == 'n' && *((__ns)+4) == 's')
  31. static void
  32. _qualify_namespace(XML_Parser parser, const xmlChar *name, const xmlChar *URI, xmlChar **qualified)
  33. {
  34. if (URI) {
  35. /* Use libxml functions otherwise its memory deallocation is screwed up */
  36. *qualified = xmlStrdup(URI);
  37. *qualified = xmlStrncat(*qualified, parser->_ns_separator, 1);
  38. *qualified = xmlStrncat(*qualified, name, xmlStrlen(name));
  39. } else {
  40. *qualified = xmlStrdup(name);
  41. }
  42. }
  43. static void
  44. _start_element_handler(void *user, const xmlChar *name, const xmlChar **attributes)
  45. {
  46. XML_Parser parser = (XML_Parser) user;
  47. xmlChar *qualified_name = NULL;
  48. if (parser->h_start_element == NULL) {
  49. if (parser->h_default) {
  50. int attno = 0;
  51. qualified_name = xmlStrncatNew((xmlChar *)"<", name, xmlStrlen(name));
  52. if (attributes) {
  53. while (attributes[attno] != NULL) {
  54. int att_len;
  55. char *att_string, *att_name, *att_value;
  56. att_name = (char *)attributes[attno++];
  57. att_value = (char *)attributes[attno++];
  58. att_len = spprintf(&att_string, 0, " %s=\"%s\"", att_name, att_value);
  59. qualified_name = xmlStrncat(qualified_name, (xmlChar *)att_string, att_len);
  60. efree(att_string);
  61. }
  62. }
  63. qualified_name = xmlStrncat(qualified_name, (xmlChar *)">", 1);
  64. parser->h_default(parser->user, (const XML_Char *) qualified_name, xmlStrlen(qualified_name));
  65. xmlFree(qualified_name);
  66. }
  67. return;
  68. }
  69. qualified_name = xmlStrdup(name);
  70. parser->h_start_element(parser->user, (const XML_Char *) qualified_name, (const XML_Char **) attributes);
  71. xmlFree(qualified_name);
  72. }
  73. static void
  74. _start_element_handler_ns(void *user, const xmlChar *name, const xmlChar *prefix, const xmlChar *URI, int nb_namespaces, const xmlChar ** namespaces, int nb_attributes, int nb_defaulted, const xmlChar ** attributes)
  75. {
  76. XML_Parser parser = (XML_Parser) user;
  77. xmlChar *qualified_name = NULL;
  78. xmlChar **attrs = NULL;
  79. int i;
  80. int z = 0;
  81. int y = 0;
  82. if (nb_namespaces > 0 && parser->h_start_ns != NULL) {
  83. for (i = 0; i < nb_namespaces; i += 1) {
  84. parser->h_start_ns(parser->user, (const XML_Char *) namespaces[y], (const XML_Char *) namespaces[y+1]);
  85. y += 2;
  86. }
  87. y = 0;
  88. }
  89. if (parser->h_start_element == NULL) {
  90. if (parser->h_default) {
  91. if (prefix) {
  92. qualified_name = xmlStrncatNew((xmlChar *)"<", prefix, xmlStrlen(prefix));
  93. qualified_name = xmlStrncat(qualified_name, (xmlChar *)":", 1);
  94. qualified_name = xmlStrncat(qualified_name, name, xmlStrlen(name));
  95. } else {
  96. qualified_name = xmlStrncatNew((xmlChar *)"<", name, xmlStrlen(name));
  97. }
  98. if (namespaces) {
  99. int i, j;
  100. for (i = 0,j = 0;j < nb_namespaces;j++) {
  101. int ns_len;
  102. char *ns_string, *ns_prefix, *ns_url;
  103. ns_prefix = (char *) namespaces[i++];
  104. ns_url = (char *) namespaces[i++];
  105. if (ns_prefix) {
  106. ns_len = spprintf(&ns_string, 0, " xmlns:%s=\"%s\"", ns_prefix, ns_url);
  107. } else {
  108. ns_len = spprintf(&ns_string, 0, " xmlns=\"%s\"", ns_url);
  109. }
  110. qualified_name = xmlStrncat(qualified_name, (xmlChar *)ns_string, ns_len);
  111. efree(ns_string);
  112. }
  113. }
  114. if (attributes) {
  115. for (i = 0; i < nb_attributes; i += 1) {
  116. int att_len;
  117. char *att_string, *att_name, *att_value, *att_prefix, *att_valueend;
  118. att_name = (char *) attributes[y++];
  119. att_prefix = (char *)attributes[y++];
  120. y++;
  121. att_value = (char *)attributes[y++];
  122. att_valueend = (char *)attributes[y++];
  123. if (att_prefix) {
  124. att_len = spprintf(&att_string, 0, " %s:%s=\"", att_prefix, att_name);
  125. } else {
  126. att_len = spprintf(&att_string, 0, " %s=\"", att_name);
  127. }
  128. qualified_name = xmlStrncat(qualified_name, (xmlChar *)att_string, att_len);
  129. qualified_name = xmlStrncat(qualified_name, (xmlChar *)att_value, att_valueend - att_value);
  130. qualified_name = xmlStrncat(qualified_name, (xmlChar *)"\"", 1);
  131. efree(att_string);
  132. }
  133. }
  134. qualified_name = xmlStrncat(qualified_name, (xmlChar *)">", 1);
  135. parser->h_default(parser->user, (const XML_Char *) qualified_name, xmlStrlen(qualified_name));
  136. xmlFree(qualified_name);
  137. }
  138. return;
  139. }
  140. _qualify_namespace(parser, name, URI, &qualified_name);
  141. if (attributes != NULL) {
  142. xmlChar *qualified_name_attr = NULL;
  143. attrs = safe_emalloc((nb_attributes * 2) + 1, sizeof(int *), 0);
  144. for (i = 0; i < nb_attributes; i += 1) {
  145. if (attributes[y+1] != NULL) {
  146. _qualify_namespace(parser, attributes[y] , attributes[y + 2], &qualified_name_attr);
  147. } else {
  148. qualified_name_attr = xmlStrdup(attributes[y]);
  149. }
  150. attrs[z] = qualified_name_attr;
  151. attrs[z + 1] = xmlStrndup(attributes[y + 3] , (int) (attributes[y + 4] - attributes[y + 3]));
  152. z += 2;
  153. y += 5;
  154. }
  155. attrs[z] = NULL;
  156. }
  157. parser->h_start_element(parser->user, (const XML_Char *) qualified_name, (const XML_Char **) attrs);
  158. if (attrs) {
  159. for (i = 0; i < z; i++) {
  160. xmlFree(attrs[i]);
  161. }
  162. efree(attrs);
  163. }
  164. xmlFree(qualified_name);
  165. }
  166. static void
  167. _end_element_handler(void *user, const xmlChar *name)
  168. {
  169. xmlChar *qualified_name;
  170. XML_Parser parser = (XML_Parser) user;
  171. if (parser->h_end_element == NULL) {
  172. if (parser->h_default) {
  173. char *end_element;
  174. spprintf(&end_element, 0, "</%s>", (char *)name);
  175. parser->h_default(parser->user, (const XML_Char *) end_element, strlen(end_element));
  176. efree(end_element);
  177. }
  178. return;
  179. }
  180. qualified_name = xmlStrdup(name);
  181. parser->h_end_element(parser->user, (const XML_Char *) qualified_name);
  182. xmlFree(qualified_name);
  183. }
  184. static void
  185. _end_element_handler_ns(void *user, const xmlChar *name, const xmlChar * prefix, const xmlChar *URI)
  186. {
  187. xmlChar *qualified_name;
  188. XML_Parser parser = (XML_Parser) user;
  189. if (parser->h_end_element == NULL) {
  190. if (parser->h_default) {
  191. char *end_element;
  192. int end_element_len;
  193. if (prefix) {
  194. end_element_len = spprintf(&end_element, 0, "</%s:%s>", (char *) prefix, (char *)name);
  195. } else {
  196. end_element_len = spprintf(&end_element, 0, "</%s>", (char *)name);
  197. }
  198. parser->h_default(parser->user, (const XML_Char *) end_element, end_element_len);
  199. efree(end_element);
  200. }
  201. return;
  202. }
  203. _qualify_namespace(parser, name, URI, &qualified_name);
  204. parser->h_end_element(parser->user, (const XML_Char *) qualified_name);
  205. xmlFree(qualified_name);
  206. }
  207. static void
  208. _cdata_handler(void *user, const xmlChar *cdata, int cdata_len)
  209. {
  210. XML_Parser parser = (XML_Parser) user;
  211. if (parser->h_cdata == NULL) {
  212. if (parser->h_default) {
  213. parser->h_default(parser->user, (const XML_Char *) cdata, cdata_len);
  214. }
  215. return;
  216. }
  217. parser->h_cdata(parser->user, (const XML_Char *) cdata, cdata_len);
  218. }
  219. static void
  220. _pi_handler(void *user, const xmlChar *target, const xmlChar *data)
  221. {
  222. XML_Parser parser = (XML_Parser) user;
  223. if (parser->h_pi == NULL) {
  224. if (parser->h_default) {
  225. char *full_pi;
  226. spprintf(&full_pi, 0, "<?%s %s?>", (char *)target, (char *)data);
  227. parser->h_default(parser->user, (const XML_Char *) full_pi, strlen(full_pi));
  228. efree(full_pi);
  229. }
  230. return;
  231. }
  232. parser->h_pi(parser->user, (const XML_Char *) target, (const XML_Char *) data);
  233. }
  234. static void
  235. _unparsed_entity_decl_handler(void *user,
  236. const xmlChar *name,
  237. const xmlChar *pub_id,
  238. const xmlChar *sys_id,
  239. const xmlChar *notation)
  240. {
  241. XML_Parser parser = (XML_Parser) user;
  242. if (parser->h_unparsed_entity_decl == NULL) {
  243. return;
  244. }
  245. parser->h_unparsed_entity_decl(parser->user, name, NULL, sys_id, pub_id, notation);
  246. }
  247. static void
  248. _notation_decl_handler(void *user, const xmlChar *notation, const xmlChar *pub_id, const xmlChar *sys_id)
  249. {
  250. XML_Parser parser = (XML_Parser) user;
  251. if (parser->h_notation_decl == NULL) {
  252. return;
  253. }
  254. parser->h_notation_decl(parser->user, notation, NULL, sys_id, pub_id);
  255. }
  256. static void
  257. _build_comment(const xmlChar *data, int data_len, xmlChar **comment, int *comment_len)
  258. {
  259. *comment_len = data_len + 7;
  260. *comment = xmlMalloc(*comment_len + 1);
  261. memcpy(*comment, "<!--", 4);
  262. memcpy(*comment + 4, data, data_len);
  263. memcpy(*comment + 4 + data_len, "-->", 3);
  264. (*comment)[*comment_len] = '\0';
  265. }
  266. static void
  267. _comment_handler(void *user, const xmlChar *comment)
  268. {
  269. XML_Parser parser = (XML_Parser) user;
  270. if (parser->h_default) {
  271. xmlChar *d_comment;
  272. int d_comment_len;
  273. _build_comment(comment, xmlStrlen(comment), &d_comment, &d_comment_len);
  274. parser->h_default(parser->user, d_comment, d_comment_len);
  275. xmlFree(d_comment);
  276. }
  277. }
  278. static void
  279. _build_entity(const xmlChar *name, int len, xmlChar **entity, int *entity_len)
  280. {
  281. *entity_len = len + 2;
  282. *entity = xmlMalloc(*entity_len + 1);
  283. (*entity)[0] = '&';
  284. memcpy(*entity+1, name, len);
  285. (*entity)[len+1] = ';';
  286. (*entity)[*entity_len] = '\0';
  287. }
  288. static void
  289. _external_entity_ref_handler(void *user, const xmlChar *names, int type, const xmlChar *sys_id, const xmlChar *pub_id, xmlChar *content)
  290. {
  291. XML_Parser parser = (XML_Parser) user;
  292. if (parser->h_external_entity_ref == NULL) {
  293. return;
  294. }
  295. if (!parser->h_external_entity_ref(parser, names, (XML_Char *) "", sys_id, pub_id)) {
  296. xmlStopParser(parser->parser);
  297. parser->parser->errNo = XML_ERROR_EXTERNAL_ENTITY_HANDLING;
  298. };
  299. }
  300. static xmlEntityPtr
  301. _get_entity(void *user, const xmlChar *name)
  302. {
  303. XML_Parser parser = (XML_Parser) user;
  304. xmlEntityPtr ret = NULL;
  305. if (parser->parser->inSubset == 0) {
  306. ret = xmlGetPredefinedEntity(name);
  307. if (ret == NULL)
  308. ret = xmlGetDocEntity(parser->parser->myDoc, name);
  309. if (ret == NULL || (parser->parser->instate != XML_PARSER_ENTITY_VALUE && parser->parser->instate != XML_PARSER_ATTRIBUTE_VALUE)) {
  310. if (ret == NULL || ret->etype == XML_INTERNAL_GENERAL_ENTITY || ret->etype == XML_INTERNAL_PARAMETER_ENTITY || ret->etype == XML_INTERNAL_PREDEFINED_ENTITY) {
  311. /* Predefined entities will expand unless no cdata handler is present */
  312. if (parser->h_default && ! (ret && ret->etype == XML_INTERNAL_PREDEFINED_ENTITY && parser->h_cdata)) {
  313. xmlChar *entity;
  314. int len;
  315. _build_entity(name, xmlStrlen(name), &entity, &len);
  316. parser->h_default(parser->user, (const xmlChar *) entity, len);
  317. xmlFree(entity);
  318. } else {
  319. /* expat will not expand internal entities if default handler is present otherwise
  320. it will expand and pass them to cdata handler */
  321. if (parser->h_cdata && ret) {
  322. parser->h_cdata(parser->user, ret->content, xmlStrlen(ret->content));
  323. }
  324. }
  325. } else {
  326. if (ret->etype == XML_EXTERNAL_GENERAL_PARSED_ENTITY) {
  327. _external_entity_ref_handler(user, ret->name, ret->etype, ret->SystemID, ret->ExternalID, NULL);
  328. }
  329. }
  330. }
  331. }
  332. return ret;
  333. }
  334. static const xmlSAXHandler
  335. php_xml_compat_handlers = {
  336. NULL, /* internalSubset */
  337. NULL, /* isStandalone */
  338. NULL, /* hasInternalSubset */
  339. NULL, /* hasExternalSubset */
  340. NULL, /* resolveEntity */
  341. _get_entity, /* getEntity */
  342. NULL, /* entityDecl */
  343. _notation_decl_handler,
  344. NULL, /* attributeDecl */
  345. NULL, /* elementDecl */
  346. _unparsed_entity_decl_handler, /* unparsedEntity */
  347. NULL, /* setDocumentLocator */
  348. NULL, /* startDocument */
  349. NULL, /* endDocument */
  350. _start_element_handler, /* startElement */
  351. _end_element_handler, /* endElement */
  352. NULL, /* reference */
  353. _cdata_handler,
  354. NULL, /* ignorableWhitespace */
  355. _pi_handler,
  356. _comment_handler, /* comment */
  357. NULL, /* warning */
  358. NULL, /* error */
  359. NULL, /* fatalError */
  360. NULL, /* getParameterEntity */
  361. _cdata_handler, /* cdataBlock */
  362. NULL, /* externalSubset */
  363. XML_SAX2_MAGIC,
  364. NULL,
  365. _start_element_handler_ns,
  366. _end_element_handler_ns,
  367. NULL
  368. };
  369. PHP_XML_API XML_Parser
  370. XML_ParserCreate(const XML_Char *encoding)
  371. {
  372. return XML_ParserCreate_MM(encoding, NULL, NULL);
  373. }
  374. PHP_XML_API XML_Parser
  375. XML_ParserCreateNS(const XML_Char *encoding, const XML_Char sep)
  376. {
  377. XML_Char tmp[2];
  378. tmp[0] = sep;
  379. tmp[1] = '\0';
  380. return XML_ParserCreate_MM(encoding, NULL, tmp);
  381. }
  382. PHP_XML_API XML_Parser
  383. XML_ParserCreate_MM(const XML_Char *encoding, const XML_Memory_Handling_Suite *memsuite, const XML_Char *sep)
  384. {
  385. XML_Parser parser;
  386. parser = (XML_Parser) emalloc(sizeof(struct _XML_Parser));
  387. memset(parser, 0, sizeof(struct _XML_Parser));
  388. parser->use_namespace = 0;
  389. parser->_ns_separator = NULL;
  390. parser->parser = xmlCreatePushParserCtxt((xmlSAXHandlerPtr) &php_xml_compat_handlers, (void *) parser, NULL, 0, NULL);
  391. if (parser->parser == NULL) {
  392. efree(parser);
  393. return NULL;
  394. }
  395. #if LIBXML_VERSION <= 20617
  396. /* for older versions of libxml2, allow correct detection of
  397. * charset in documents with a BOM: */
  398. parser->parser->charset = XML_CHAR_ENCODING_NONE;
  399. #endif
  400. #if LIBXML_VERSION >= 20703
  401. xmlCtxtUseOptions(parser->parser, XML_PARSE_OLDSAX);
  402. #endif
  403. parser->parser->replaceEntities = 1;
  404. parser->parser->wellFormed = 0;
  405. if (sep != NULL) {
  406. parser->use_namespace = 1;
  407. parser->parser->sax2 = 1;
  408. parser->_ns_separator = xmlStrdup(sep);
  409. } else {
  410. /* Reset flag as XML_SAX2_MAGIC is needed for xmlCreatePushParserCtxt
  411. so must be set in the handlers */
  412. parser->parser->sax->initialized = 1;
  413. }
  414. return parser;
  415. }
  416. PHP_XML_API void
  417. XML_SetUserData(XML_Parser parser, void *user)
  418. {
  419. parser->user = user;
  420. }
  421. PHP_XML_API void *
  422. XML_GetUserData(XML_Parser parser)
  423. {
  424. return parser->user;
  425. }
  426. PHP_XML_API void
  427. XML_SetElementHandler(XML_Parser parser, XML_StartElementHandler start, XML_EndElementHandler end)
  428. {
  429. parser->h_start_element = start;
  430. parser->h_end_element = end;
  431. }
  432. PHP_XML_API void
  433. XML_SetCharacterDataHandler(XML_Parser parser, XML_CharacterDataHandler cdata)
  434. {
  435. parser->h_cdata = cdata;
  436. }
  437. PHP_XML_API void
  438. XML_SetProcessingInstructionHandler(XML_Parser parser, XML_ProcessingInstructionHandler pi)
  439. {
  440. parser->h_pi = pi;
  441. }
  442. PHP_XML_API void
  443. XML_SetCommentHandler(XML_Parser parser, XML_CommentHandler comment)
  444. {
  445. parser->h_comment = comment;
  446. }
  447. PHP_XML_API void
  448. XML_SetDefaultHandler(XML_Parser parser, XML_DefaultHandler d)
  449. {
  450. parser->h_default = d;
  451. }
  452. PHP_XML_API void
  453. XML_SetUnparsedEntityDeclHandler(XML_Parser parser, XML_UnparsedEntityDeclHandler unparsed_decl)
  454. {
  455. parser->h_unparsed_entity_decl = unparsed_decl;
  456. }
  457. PHP_XML_API void
  458. XML_SetNotationDeclHandler(XML_Parser parser, XML_NotationDeclHandler notation_decl)
  459. {
  460. parser->h_notation_decl = notation_decl;
  461. }
  462. PHP_XML_API void
  463. XML_SetExternalEntityRefHandler(XML_Parser parser, XML_ExternalEntityRefHandler ext_entity)
  464. {
  465. parser->h_external_entity_ref = ext_entity;
  466. }
  467. PHP_XML_API void
  468. XML_SetStartNamespaceDeclHandler(XML_Parser parser, XML_StartNamespaceDeclHandler start_ns)
  469. {
  470. parser->h_start_ns = start_ns;
  471. }
  472. PHP_XML_API void
  473. XML_SetEndNamespaceDeclHandler(XML_Parser parser, XML_EndNamespaceDeclHandler end_ns)
  474. {
  475. parser->h_end_ns = end_ns;
  476. }
  477. PHP_XML_API int
  478. XML_Parse(XML_Parser parser, const XML_Char *data, int data_len, int is_final)
  479. {
  480. int error;
  481. /* The following is a hack to keep BC with PHP 4 while avoiding
  482. the inifite loop in libxml <= 2.6.17 which occurs when no encoding
  483. has been defined and none can be detected */
  484. #if LIBXML_VERSION <= 20617
  485. if (parser->parser->charset == XML_CHAR_ENCODING_NONE) {
  486. if (data_len >= 4 || (parser->parser->input->buf->buffer->use + data_len >= 4)) {
  487. xmlChar start[4];
  488. int char_count;
  489. char_count = parser->parser->input->buf->buffer->use;
  490. if (char_count > 4) {
  491. char_count = 4;
  492. }
  493. memcpy(start, parser->parser->input->buf->buffer->content, (size_t)char_count);
  494. memcpy(start + char_count, data, (size_t)(4 - char_count));
  495. if (xmlDetectCharEncoding(&start[0], 4) == XML_CHAR_ENCODING_NONE) {
  496. parser->parser->charset = XML_CHAR_ENCODING_UTF8;
  497. }
  498. }
  499. }
  500. #endif
  501. if (parser->parser->lastError.level >= XML_ERR_WARNING) {
  502. return 0;
  503. }
  504. error = xmlParseChunk(parser->parser, (char *) data, data_len, is_final);
  505. if (error) {
  506. return 0;
  507. } else {
  508. return 1;
  509. }
  510. }
  511. PHP_XML_API int
  512. XML_GetErrorCode(XML_Parser parser)
  513. {
  514. return parser->parser->errNo;
  515. }
  516. static const XML_Char *const error_mapping[] = {
  517. (const XML_Char *)"No error",
  518. (const XML_Char *)"No memory",
  519. (const XML_Char *)"Invalid document start",
  520. (const XML_Char *)"Empty document",
  521. (const XML_Char *)"Not well-formed (invalid token)",
  522. (const XML_Char *)"Invalid document end",
  523. (const XML_Char *)"Invalid hexadecimal character reference",
  524. (const XML_Char *)"Invalid decimal character reference",
  525. (const XML_Char *)"Invalid character reference",
  526. (const XML_Char *)"Invalid character",
  527. (const XML_Char *)"XML_ERR_CHARREF_AT_EOF",
  528. (const XML_Char *)"XML_ERR_CHARREF_IN_PROLOG",
  529. (const XML_Char *)"XML_ERR_CHARREF_IN_EPILOG",
  530. (const XML_Char *)"XML_ERR_CHARREF_IN_DTD",
  531. (const XML_Char *)"XML_ERR_ENTITYREF_AT_EOF",
  532. (const XML_Char *)"XML_ERR_ENTITYREF_IN_PROLOG",
  533. (const XML_Char *)"XML_ERR_ENTITYREF_IN_EPILOG",
  534. (const XML_Char *)"XML_ERR_ENTITYREF_IN_DTD",
  535. (const XML_Char *)"PEReference at end of document",
  536. (const XML_Char *)"PEReference in prolog",
  537. (const XML_Char *)"PEReference in epilog",
  538. (const XML_Char *)"PEReference: forbidden within markup decl in internal subset",
  539. (const XML_Char *)"XML_ERR_ENTITYREF_NO_NAME",
  540. (const XML_Char *)"EntityRef: expecting ';'",
  541. (const XML_Char *)"PEReference: no name",
  542. (const XML_Char *)"PEReference: expecting ';'",
  543. (const XML_Char *)"Undeclared entity error",
  544. (const XML_Char *)"Undeclared entity warning",
  545. (const XML_Char *)"Unparsed Entity",
  546. (const XML_Char *)"XML_ERR_ENTITY_IS_EXTERNAL",
  547. (const XML_Char *)"XML_ERR_ENTITY_IS_PARAMETER",
  548. (const XML_Char *)"Unknown encoding",
  549. (const XML_Char *)"Unsupported encoding",
  550. (const XML_Char *)"String not started expecting ' or \"",
  551. (const XML_Char *)"String not closed expecting \" or '",
  552. (const XML_Char *)"Namespace declaration error",
  553. (const XML_Char *)"EntityValue: \" or ' expected",
  554. (const XML_Char *)"EntityValue: \" or ' expected",
  555. (const XML_Char *)"< in attribute",
  556. (const XML_Char *)"Attribute not started",
  557. (const XML_Char *)"Attribute not finished",
  558. (const XML_Char *)"Attribute without value",
  559. (const XML_Char *)"Attribute redefined",
  560. (const XML_Char *)"SystemLiteral \" or ' expected",
  561. (const XML_Char *)"SystemLiteral \" or ' expected",
  562. /* (const XML_Char *)"XML_ERR_COMMENT_NOT_STARTED", <= eliminated on purpose */
  563. (const XML_Char *)"Comment not finished",
  564. (const XML_Char *)"Processing Instruction not started",
  565. (const XML_Char *)"Processing Instruction not finished",
  566. (const XML_Char *)"NOTATION: Name expected here",
  567. (const XML_Char *)"'>' required to close NOTATION declaration",
  568. (const XML_Char *)"'(' required to start ATTLIST enumeration",
  569. (const XML_Char *)"'(' required to start ATTLIST enumeration",
  570. (const XML_Char *)"MixedContentDecl : '|' or ')*' expected",
  571. (const XML_Char *)"XML_ERR_MIXED_NOT_FINISHED",
  572. (const XML_Char *)"ELEMENT in DTD not started",
  573. (const XML_Char *)"ELEMENT in DTD not finished",
  574. (const XML_Char *)"XML declaration not started",
  575. (const XML_Char *)"XML declaration not finished",
  576. (const XML_Char *)"XML_ERR_CONDSEC_NOT_STARTED",
  577. (const XML_Char *)"XML conditional section not closed",
  578. (const XML_Char *)"Content error in the external subset",
  579. (const XML_Char *)"DOCTYPE not finished",
  580. (const XML_Char *)"Sequence ']]>' not allowed in content",
  581. (const XML_Char *)"CDATA not finished",
  582. (const XML_Char *)"Reserved XML Name",
  583. (const XML_Char *)"Space required",
  584. (const XML_Char *)"XML_ERR_SEPARATOR_REQUIRED",
  585. (const XML_Char *)"NmToken expected in ATTLIST enumeration",
  586. (const XML_Char *)"XML_ERR_NAME_REQUIRED",
  587. (const XML_Char *)"MixedContentDecl : '#PCDATA' expected",
  588. (const XML_Char *)"SYSTEM or PUBLIC, the URI is missing",
  589. (const XML_Char *)"PUBLIC, the Public Identifier is missing",
  590. (const XML_Char *)"< required",
  591. (const XML_Char *)"> required",
  592. (const XML_Char *)"</ required",
  593. (const XML_Char *)"= required",
  594. (const XML_Char *)"Mismatched tag",
  595. (const XML_Char *)"Tag not finished",
  596. (const XML_Char *)"standalone accepts only 'yes' or 'no'",
  597. (const XML_Char *)"Invalid XML encoding name",
  598. (const XML_Char *)"Comment must not contain '--' (double-hyphen)",
  599. (const XML_Char *)"Invalid encoding",
  600. (const XML_Char *)"external parsed entities cannot be standalone",
  601. (const XML_Char *)"XML conditional section '[' expected",
  602. (const XML_Char *)"Entity value required",
  603. (const XML_Char *)"chunk is not well balanced",
  604. (const XML_Char *)"extra content at the end of well balanced chunk",
  605. (const XML_Char *)"XML_ERR_ENTITY_CHAR_ERROR",
  606. (const XML_Char *)"PEReferences forbidden in internal subset",
  607. (const XML_Char *)"Detected an entity reference loop",
  608. (const XML_Char *)"XML_ERR_ENTITY_BOUNDARY",
  609. (const XML_Char *)"Invalid URI",
  610. (const XML_Char *)"Fragment not allowed",
  611. (const XML_Char *)"XML_WAR_CATALOG_PI",
  612. (const XML_Char *)"XML_ERR_NO_DTD",
  613. (const XML_Char *)"conditional section INCLUDE or IGNORE keyword expected", /* 95 */
  614. (const XML_Char *)"Version in XML Declaration missing", /* 96 */
  615. (const XML_Char *)"XML_WAR_UNKNOWN_VERSION", /* 97 */
  616. (const XML_Char *)"XML_WAR_LANG_VALUE", /* 98 */
  617. (const XML_Char *)"XML_WAR_NS_URI", /* 99 */
  618. (const XML_Char *)"XML_WAR_NS_URI_RELATIVE", /* 100 */
  619. (const XML_Char *)"Missing encoding in text declaration" /* 101 */
  620. };
  621. PHP_XML_API const XML_Char *
  622. XML_ErrorString(int code)
  623. {
  624. if (code < 0 || code >= (int)(sizeof(error_mapping) / sizeof(error_mapping[0]))) {
  625. return (const XML_Char *) "Unknown";
  626. }
  627. return error_mapping[code];
  628. }
  629. PHP_XML_API int
  630. XML_GetCurrentLineNumber(XML_Parser parser)
  631. {
  632. return parser->parser->input->line;
  633. }
  634. PHP_XML_API int
  635. XML_GetCurrentColumnNumber(XML_Parser parser)
  636. {
  637. return parser->parser->input->col;
  638. }
  639. PHP_XML_API int
  640. XML_GetCurrentByteIndex(XML_Parser parser)
  641. {
  642. return parser->parser->input->consumed +
  643. (parser->parser->input->cur - parser->parser->input->base);
  644. }
  645. PHP_XML_API int
  646. XML_GetCurrentByteCount(XML_Parser parser)
  647. {
  648. /* WARNING: this is identical to ByteIndex; it should probably
  649. * be different */
  650. return parser->parser->input->consumed +
  651. (parser->parser->input->cur - parser->parser->input->base);
  652. }
  653. PHP_XML_API const XML_Char *XML_ExpatVersion(void)
  654. {
  655. return (const XML_Char *) "1.0";
  656. }
  657. PHP_XML_API void
  658. XML_ParserFree(XML_Parser parser)
  659. {
  660. if (parser->use_namespace) {
  661. if (parser->_ns_separator) {
  662. xmlFree(parser->_ns_separator);
  663. }
  664. }
  665. if (parser->parser->myDoc) {
  666. xmlFreeDoc(parser->parser->myDoc);
  667. parser->parser->myDoc = NULL;
  668. }
  669. xmlFreeParserCtxt(parser->parser);
  670. efree(parser);
  671. }
  672. #endif /* LIBXML_EXPAT_COMPAT */
  673. #endif
  674. /**
  675. * Local Variables:
  676. * tab-width: 4
  677. * c-basic-offset: 4
  678. * indent-tabs-mode: t
  679. * End:
  680. * vim600: fdm=marker
  681. * vim: ts=4 noet sw=4
  682. */