php_dom.c 64 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 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: Christian Stocker <chregu@php.net> |
  16. | Rob Richards <rrichards@php.net> |
  17. | Marcus Borger <helly@php.net> |
  18. +----------------------------------------------------------------------+
  19. */
  20. /* $Id$ */
  21. #ifdef HAVE_CONFIG_H
  22. #include "config.h"
  23. #endif
  24. #include "php.h"
  25. #if HAVE_LIBXML && HAVE_DOM
  26. #include "ext/standard/php_rand.h"
  27. #include "php_dom.h"
  28. #include "dom_properties.h"
  29. #include "zend_interfaces.h"
  30. #include "ext/standard/info.h"
  31. #define PHP_XPATH 1
  32. #define PHP_XPTR 2
  33. /* {{{ class entries */
  34. zend_class_entry *dom_node_class_entry;
  35. zend_class_entry *dom_domexception_class_entry;
  36. zend_class_entry *dom_domstringlist_class_entry;
  37. zend_class_entry *dom_namelist_class_entry;
  38. zend_class_entry *dom_domimplementationlist_class_entry;
  39. zend_class_entry *dom_domimplementationsource_class_entry;
  40. zend_class_entry *dom_domimplementation_class_entry;
  41. zend_class_entry *dom_documentfragment_class_entry;
  42. zend_class_entry *dom_document_class_entry;
  43. zend_class_entry *dom_nodelist_class_entry;
  44. zend_class_entry *dom_namednodemap_class_entry;
  45. zend_class_entry *dom_characterdata_class_entry;
  46. zend_class_entry *dom_attr_class_entry;
  47. zend_class_entry *dom_element_class_entry;
  48. zend_class_entry *dom_text_class_entry;
  49. zend_class_entry *dom_comment_class_entry;
  50. zend_class_entry *dom_typeinfo_class_entry;
  51. zend_class_entry *dom_userdatahandler_class_entry;
  52. zend_class_entry *dom_domerror_class_entry;
  53. zend_class_entry *dom_domerrorhandler_class_entry;
  54. zend_class_entry *dom_domlocator_class_entry;
  55. zend_class_entry *dom_domconfiguration_class_entry;
  56. zend_class_entry *dom_cdatasection_class_entry;
  57. zend_class_entry *dom_documenttype_class_entry;
  58. zend_class_entry *dom_notation_class_entry;
  59. zend_class_entry *dom_entity_class_entry;
  60. zend_class_entry *dom_entityreference_class_entry;
  61. zend_class_entry *dom_processinginstruction_class_entry;
  62. zend_class_entry *dom_string_extend_class_entry;
  63. #if defined(LIBXML_XPATH_ENABLED)
  64. zend_class_entry *dom_xpath_class_entry;
  65. #endif
  66. zend_class_entry *dom_namespace_node_class_entry;
  67. /* }}} */
  68. zend_object_handlers dom_object_handlers;
  69. zend_object_handlers dom_nnodemap_object_handlers;
  70. static HashTable classes;
  71. /* {{{ prop handler tables */
  72. static HashTable dom_domstringlist_prop_handlers;
  73. static HashTable dom_namelist_prop_handlers;
  74. static HashTable dom_domimplementationlist_prop_handlers;
  75. static HashTable dom_document_prop_handlers;
  76. static HashTable dom_node_prop_handlers;
  77. static HashTable dom_nodelist_prop_handlers;
  78. static HashTable dom_namednodemap_prop_handlers;
  79. static HashTable dom_characterdata_prop_handlers;
  80. static HashTable dom_attr_prop_handlers;
  81. static HashTable dom_element_prop_handlers;
  82. static HashTable dom_text_prop_handlers;
  83. static HashTable dom_typeinfo_prop_handlers;
  84. static HashTable dom_domerror_prop_handlers;
  85. static HashTable dom_domlocator_prop_handlers;
  86. static HashTable dom_documenttype_prop_handlers;
  87. static HashTable dom_notation_prop_handlers;
  88. static HashTable dom_entity_prop_handlers;
  89. static HashTable dom_processinginstruction_prop_handlers;
  90. static HashTable dom_namespace_node_prop_handlers;
  91. #if defined(LIBXML_XPATH_ENABLED)
  92. static HashTable dom_xpath_prop_handlers;
  93. #endif
  94. /* }}} */
  95. typedef int (*dom_read_t)(dom_object *obj, zval **retval TSRMLS_DC);
  96. typedef int (*dom_write_t)(dom_object *obj, zval *newval TSRMLS_DC);
  97. typedef struct _dom_prop_handler {
  98. dom_read_t read_func;
  99. dom_write_t write_func;
  100. } dom_prop_handler;
  101. /* {{{ int dom_node_is_read_only(xmlNodePtr node) */
  102. int dom_node_is_read_only(xmlNodePtr node) {
  103. switch (node->type) {
  104. case XML_ENTITY_REF_NODE:
  105. case XML_ENTITY_NODE:
  106. case XML_DOCUMENT_TYPE_NODE:
  107. case XML_NOTATION_NODE:
  108. case XML_DTD_NODE:
  109. case XML_ELEMENT_DECL:
  110. case XML_ATTRIBUTE_DECL:
  111. case XML_ENTITY_DECL:
  112. case XML_NAMESPACE_DECL:
  113. return SUCCESS;
  114. break;
  115. default:
  116. if (node->doc == NULL) {
  117. return SUCCESS;
  118. } else {
  119. return FAILURE;
  120. }
  121. }
  122. }
  123. /* }}} end dom_node_is_read_only */
  124. /* {{{ int dom_node_children_valid(xmlNodePtr node) */
  125. int dom_node_children_valid(xmlNodePtr node) {
  126. switch (node->type) {
  127. case XML_DOCUMENT_TYPE_NODE:
  128. case XML_DTD_NODE:
  129. case XML_PI_NODE:
  130. case XML_COMMENT_NODE:
  131. case XML_TEXT_NODE:
  132. case XML_CDATA_SECTION_NODE:
  133. case XML_NOTATION_NODE:
  134. return FAILURE;
  135. break;
  136. default:
  137. return SUCCESS;
  138. }
  139. }
  140. /* }}} end dom_node_children_valid */
  141. /* {{{ dom_get_doc_props() */
  142. dom_doc_propsptr dom_get_doc_props(php_libxml_ref_obj *document)
  143. {
  144. dom_doc_propsptr doc_props;
  145. if (document && document->doc_props) {
  146. return document->doc_props;
  147. } else {
  148. doc_props = emalloc(sizeof(libxml_doc_props));
  149. doc_props->formatoutput = 0;
  150. doc_props->validateonparse = 0;
  151. doc_props->resolveexternals = 0;
  152. doc_props->preservewhitespace = 1;
  153. doc_props->substituteentities = 0;
  154. doc_props->stricterror = 1;
  155. doc_props->recover = 0;
  156. doc_props->classmap = NULL;
  157. if (document) {
  158. document->doc_props = doc_props;
  159. }
  160. return doc_props;
  161. }
  162. }
  163. static void dom_copy_doc_props(php_libxml_ref_obj *source_doc, php_libxml_ref_obj *dest_doc)
  164. {
  165. dom_doc_propsptr source, dest;
  166. if (source_doc && dest_doc) {
  167. source = dom_get_doc_props(source_doc);
  168. dest = dom_get_doc_props(dest_doc);
  169. dest->formatoutput = source->formatoutput;
  170. dest->validateonparse = source->validateonparse;
  171. dest->resolveexternals = source->resolveexternals;
  172. dest->preservewhitespace = source->preservewhitespace;
  173. dest->substituteentities = source->substituteentities;
  174. dest->stricterror = source->stricterror;
  175. dest->recover = source->recover;
  176. if (source->classmap) {
  177. ALLOC_HASHTABLE(dest->classmap);
  178. zend_hash_init(dest->classmap, 0, NULL, NULL, 0);
  179. zend_hash_copy(dest->classmap, source->classmap, NULL, NULL, sizeof(zend_class_entry *));
  180. }
  181. }
  182. }
  183. int dom_set_doc_classmap(php_libxml_ref_obj *document, zend_class_entry *basece, zend_class_entry *ce TSRMLS_DC)
  184. {
  185. dom_doc_propsptr doc_props;
  186. if (document) {
  187. doc_props = dom_get_doc_props(document);
  188. if (doc_props->classmap == NULL) {
  189. if (ce == NULL) {
  190. return SUCCESS;
  191. }
  192. ALLOC_HASHTABLE(doc_props->classmap);
  193. zend_hash_init(doc_props->classmap, 0, NULL, NULL, 0);
  194. }
  195. if (ce) {
  196. return zend_hash_update(doc_props->classmap, basece->name, basece->name_length + 1, &ce, sizeof(zend_class_entry *), NULL);
  197. } else {
  198. zend_hash_del(doc_props->classmap, basece->name, basece->name_length + 1);
  199. }
  200. }
  201. return SUCCESS;
  202. }
  203. zend_class_entry *dom_get_doc_classmap(php_libxml_ref_obj *document, zend_class_entry *basece TSRMLS_DC)
  204. {
  205. dom_doc_propsptr doc_props;
  206. zend_class_entry **ce = NULL;
  207. if (document) {
  208. doc_props = dom_get_doc_props(document);
  209. if (doc_props->classmap) {
  210. if (zend_hash_find(doc_props->classmap, basece->name, basece->name_length + 1, (void**) &ce) == SUCCESS) {
  211. return *ce;
  212. }
  213. }
  214. }
  215. return basece;
  216. }
  217. /* }}} */
  218. /* {{{ dom_get_strict_error() */
  219. int dom_get_strict_error(php_libxml_ref_obj *document) {
  220. int stricterror;
  221. dom_doc_propsptr doc_props;
  222. doc_props = dom_get_doc_props(document);
  223. stricterror = doc_props->stricterror;
  224. if (document == NULL) {
  225. efree(doc_props);
  226. }
  227. return stricterror;
  228. }
  229. /* }}} */
  230. /* {{{ xmlNodePtr dom_object_get_node(dom_object *obj) */
  231. PHP_DOM_EXPORT xmlNodePtr dom_object_get_node(dom_object *obj)
  232. {
  233. if (obj && obj->ptr != NULL) {
  234. return ((php_libxml_node_ptr *)obj->ptr)->node;
  235. } else {
  236. return NULL;
  237. }
  238. }
  239. /* }}} end dom_object_get_node */
  240. /* {{{ dom_object *php_dom_object_get_data(xmlNodePtr obj) */
  241. PHP_DOM_EXPORT dom_object *php_dom_object_get_data(xmlNodePtr obj)
  242. {
  243. if (obj && obj->_private != NULL) {
  244. return (dom_object *) ((php_libxml_node_ptr *) obj->_private)->_private;
  245. } else {
  246. return NULL;
  247. }
  248. }
  249. /* }}} end php_dom_object_get_data */
  250. /* {{{ dom_read_na */
  251. static int dom_read_na(dom_object *obj, zval **retval TSRMLS_DC)
  252. {
  253. *retval = NULL;
  254. php_error_docref(NULL TSRMLS_CC, E_ERROR, "Cannot read property");
  255. return FAILURE;
  256. }
  257. /* }}} */
  258. /* {{{ dom_write_na */
  259. static int dom_write_na(dom_object *obj, zval *newval TSRMLS_DC)
  260. {
  261. php_error_docref(NULL TSRMLS_CC, E_ERROR, "Cannot write property");
  262. return FAILURE;
  263. }
  264. /* }}} */
  265. /* {{{ dom_register_prop_handler */
  266. static void dom_register_prop_handler(HashTable *prop_handler, char *name, dom_read_t read_func, dom_write_t write_func TSRMLS_DC)
  267. {
  268. dom_prop_handler hnd;
  269. hnd.read_func = read_func ? read_func : dom_read_na;
  270. hnd.write_func = write_func ? write_func : dom_write_na;
  271. zend_hash_add(prop_handler, name, strlen(name)+1, &hnd, sizeof(dom_prop_handler), NULL);
  272. }
  273. /* }}} */
  274. static zval **dom_get_property_ptr_ptr(zval *object, zval *member, int type, const zend_literal *key TSRMLS_DC) /* {{{ */
  275. {
  276. dom_object *obj;
  277. zval tmp_member;
  278. zval **retval = NULL;
  279. dom_prop_handler *hnd;
  280. zend_object_handlers *std_hnd;
  281. int ret = FAILURE;
  282. if (member->type != IS_STRING) {
  283. tmp_member = *member;
  284. zval_copy_ctor(&tmp_member);
  285. convert_to_string(&tmp_member);
  286. member = &tmp_member;
  287. }
  288. obj = (dom_object *)zend_objects_get_address(object TSRMLS_CC);
  289. if (obj->prop_handler != NULL) {
  290. ret = zend_hash_find(obj->prop_handler, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, (void **) &hnd);
  291. }
  292. if (ret == FAILURE) {
  293. std_hnd = zend_get_std_object_handlers();
  294. retval = std_hnd->get_property_ptr_ptr(object, member, type, key TSRMLS_CC);
  295. }
  296. if (member == &tmp_member) {
  297. zval_dtor(member);
  298. }
  299. return retval;
  300. }
  301. /* }}} */
  302. /* {{{ dom_read_property */
  303. zval *dom_read_property(zval *object, zval *member, int type, const zend_literal *key TSRMLS_DC)
  304. {
  305. dom_object *obj;
  306. zval tmp_member;
  307. zval *retval;
  308. dom_prop_handler *hnd;
  309. zend_object_handlers *std_hnd;
  310. int ret;
  311. if (member->type != IS_STRING) {
  312. tmp_member = *member;
  313. zval_copy_ctor(&tmp_member);
  314. convert_to_string(&tmp_member);
  315. member = &tmp_member;
  316. }
  317. ret = FAILURE;
  318. obj = (dom_object *)zend_objects_get_address(object TSRMLS_CC);
  319. if (obj->prop_handler != NULL) {
  320. ret = zend_hash_find(obj->prop_handler, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, (void **) &hnd);
  321. } else if (instanceof_function(obj->std.ce, dom_node_class_entry TSRMLS_CC)) {
  322. php_error(E_WARNING, "Couldn't fetch %s. Node no longer exists", obj->std.ce->name);
  323. }
  324. if (ret == SUCCESS) {
  325. ret = hnd->read_func(obj, &retval TSRMLS_CC);
  326. if (ret == SUCCESS) {
  327. /* ensure we're creating a temporary variable */
  328. Z_SET_REFCOUNT_P(retval, 0);
  329. Z_UNSET_ISREF_P(retval);
  330. } else {
  331. retval = EG(uninitialized_zval_ptr);
  332. }
  333. } else {
  334. std_hnd = zend_get_std_object_handlers();
  335. retval = std_hnd->read_property(object, member, type, key TSRMLS_CC);
  336. }
  337. if (member == &tmp_member) {
  338. zval_dtor(member);
  339. }
  340. return retval;
  341. }
  342. /* }}} */
  343. /* {{{ dom_write_property */
  344. void dom_write_property(zval *object, zval *member, zval *value, const zend_literal *key TSRMLS_DC)
  345. {
  346. dom_object *obj;
  347. zval tmp_member;
  348. dom_prop_handler *hnd;
  349. zend_object_handlers *std_hnd;
  350. int ret;
  351. if (member->type != IS_STRING) {
  352. tmp_member = *member;
  353. zval_copy_ctor(&tmp_member);
  354. convert_to_string(&tmp_member);
  355. member = &tmp_member;
  356. }
  357. ret = FAILURE;
  358. obj = (dom_object *)zend_objects_get_address(object TSRMLS_CC);
  359. if (obj->prop_handler != NULL) {
  360. ret = zend_hash_find((HashTable *)obj->prop_handler, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, (void **) &hnd);
  361. }
  362. if (ret == SUCCESS) {
  363. hnd->write_func(obj, value TSRMLS_CC);
  364. } else {
  365. std_hnd = zend_get_std_object_handlers();
  366. std_hnd->write_property(object, member, value, key TSRMLS_CC);
  367. }
  368. if (member == &tmp_member) {
  369. zval_dtor(member);
  370. }
  371. }
  372. /* }}} */
  373. /* {{{ dom_property_exists */
  374. static int dom_property_exists(zval *object, zval *member, int check_empty, const zend_literal *key TSRMLS_DC)
  375. {
  376. dom_object *obj;
  377. zval tmp_member;
  378. dom_prop_handler *hnd;
  379. zend_object_handlers *std_hnd;
  380. int ret, retval=0;
  381. if (member->type != IS_STRING) {
  382. tmp_member = *member;
  383. zval_copy_ctor(&tmp_member);
  384. convert_to_string(&tmp_member);
  385. member = &tmp_member;
  386. }
  387. ret = FAILURE;
  388. obj = (dom_object *)zend_objects_get_address(object TSRMLS_CC);
  389. if (obj->prop_handler != NULL) {
  390. ret = zend_hash_find((HashTable *)obj->prop_handler, Z_STRVAL_P(member), Z_STRLEN_P(member)+1, (void **) &hnd);
  391. }
  392. if (ret == SUCCESS) {
  393. zval *tmp;
  394. if (check_empty == 2) {
  395. retval = 1;
  396. } else if (hnd->read_func(obj, &tmp TSRMLS_CC) == SUCCESS) {
  397. Z_SET_REFCOUNT_P(tmp, 1);
  398. Z_UNSET_ISREF_P(tmp);
  399. if (check_empty == 1) {
  400. retval = zend_is_true(tmp);
  401. } else if (check_empty == 0) {
  402. retval = (Z_TYPE_P(tmp) != IS_NULL);
  403. }
  404. zval_ptr_dtor(&tmp);
  405. }
  406. } else {
  407. std_hnd = zend_get_std_object_handlers();
  408. retval = std_hnd->has_property(object, member, check_empty, key TSRMLS_CC);
  409. }
  410. if (member == &tmp_member) {
  411. zval_dtor(member);
  412. }
  413. return retval;
  414. }
  415. /* }}} */
  416. static HashTable* dom_get_debug_info_helper(zval *object, int *is_temp TSRMLS_DC) /* {{{ */
  417. {
  418. dom_object *obj = zend_object_store_get_object(object TSRMLS_CC);
  419. HashTable *debug_info,
  420. *prop_handlers = obj->prop_handler,
  421. *std_props;
  422. HashPosition pos;
  423. dom_prop_handler *entry;
  424. zval *object_value,
  425. *null_value;
  426. *is_temp = 1;
  427. ALLOC_HASHTABLE(debug_info);
  428. ZEND_INIT_SYMTABLE_EX(debug_info, 32, 0);
  429. std_props = zend_std_get_properties(object TSRMLS_CC);
  430. zend_hash_copy(debug_info, std_props, (copy_ctor_func_t)zval_add_ref,
  431. NULL, sizeof(zval*));
  432. if (!prop_handlers) {
  433. return debug_info;
  434. }
  435. ALLOC_INIT_ZVAL(object_value);
  436. ZVAL_STRING(object_value, "(object value omitted)", 1);
  437. ALLOC_INIT_ZVAL(null_value);
  438. ZVAL_NULL(null_value);
  439. for (zend_hash_internal_pointer_reset_ex(prop_handlers, &pos);
  440. zend_hash_get_current_data_ex(prop_handlers, (void **)&entry, &pos)
  441. == SUCCESS;
  442. zend_hash_move_forward_ex(prop_handlers, &pos)) {
  443. zval *value;
  444. char *string_key = NULL;
  445. uint string_length = 0;
  446. ulong num_key;
  447. if (entry->read_func(obj, &value TSRMLS_CC) == FAILURE) {
  448. continue;
  449. }
  450. if (zend_hash_get_current_key_ex(prop_handlers, &string_key,
  451. &string_length, &num_key, 0, &pos) != HASH_KEY_IS_STRING) {
  452. continue;
  453. }
  454. if (value == EG(uninitialized_zval_ptr)) {
  455. value = null_value;
  456. } else if (Z_TYPE_P(value) == IS_OBJECT) {
  457. /* these are zvalues create on demand, with refcount and is_ref
  458. * status left in an uninitialized stated */
  459. zval_dtor(value);
  460. efree(value);
  461. value = object_value;
  462. } else {
  463. /* see comment above */
  464. Z_SET_REFCOUNT_P(value, 0);
  465. Z_UNSET_ISREF_P(value);
  466. }
  467. zval_add_ref(&value);
  468. zend_hash_add(debug_info, string_key, string_length,
  469. &value, sizeof(zval *), NULL);
  470. }
  471. zval_ptr_dtor(&null_value);
  472. zval_ptr_dtor(&object_value);
  473. return debug_info;
  474. }
  475. /* }}} */
  476. static HashTable* dom_get_debug_info(zval *object, int *is_temp TSRMLS_DC) /* {{{ */
  477. {
  478. return dom_get_debug_info_helper(object, is_temp TSRMLS_CC);
  479. }
  480. /* }}} */
  481. void *php_dom_export_node(zval *object TSRMLS_DC) /* {{{ */
  482. {
  483. php_libxml_node_object *intern;
  484. xmlNodePtr nodep = NULL;
  485. intern = (php_libxml_node_object *)zend_object_store_get_object(object TSRMLS_CC);
  486. if (intern && intern->node) {
  487. nodep = intern->node->node;
  488. }
  489. return nodep;
  490. }
  491. /* }}} */
  492. /* {{{ proto somNode dom_import_simplexml(sxeobject node)
  493. Get a simplexml_element object from dom to allow for processing */
  494. PHP_FUNCTION(dom_import_simplexml)
  495. {
  496. zval *node;
  497. xmlNodePtr nodep = NULL;
  498. php_libxml_node_object *nodeobj;
  499. int ret;
  500. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "o", &node) == FAILURE) {
  501. return;
  502. }
  503. nodeobj = (php_libxml_node_object *)zend_object_store_get_object(node TSRMLS_CC);
  504. nodep = php_libxml_import_node(node TSRMLS_CC);
  505. if (nodep && nodeobj && (nodep->type == XML_ELEMENT_NODE || nodep->type == XML_ATTRIBUTE_NODE)) {
  506. DOM_RET_OBJ((xmlNodePtr) nodep, &ret, (dom_object *)nodeobj);
  507. } else {
  508. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Nodetype to import");
  509. RETURN_NULL();
  510. }
  511. }
  512. /* }}} */
  513. zend_object_value dom_objects_store_clone_obj(zval *zobject TSRMLS_DC) /* {{{ */
  514. {
  515. zend_object_value retval;
  516. void *new_object;
  517. dom_object *intern;
  518. dom_object *old_object;
  519. struct _store_object *obj;
  520. zend_object_handle handle = Z_OBJ_HANDLE_P(zobject);
  521. obj = &EG(objects_store).object_buckets[handle].bucket.obj;
  522. if (obj->clone == NULL) {
  523. php_error(E_ERROR, "Trying to clone an uncloneable object of class %s", Z_OBJCE_P(zobject)->name);
  524. }
  525. obj->clone(obj->object, &new_object TSRMLS_CC);
  526. retval.handle = zend_objects_store_put(new_object, obj->dtor, obj->free_storage, obj->clone TSRMLS_CC);
  527. intern = (dom_object *) new_object;
  528. intern->handle = retval.handle;
  529. retval.handlers = Z_OBJ_HT_P(zobject);
  530. old_object = (dom_object *) obj->object;
  531. zend_objects_clone_members(&intern->std, retval, &old_object->std, intern->handle TSRMLS_CC);
  532. return retval;
  533. }
  534. /* }}} */
  535. /* {{{ arginfo */
  536. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_import_simplexml, 0, 0, 1)
  537. ZEND_ARG_INFO(0, node)
  538. ZEND_END_ARG_INFO()
  539. /* }}} */
  540. static const zend_function_entry dom_functions[] = {
  541. PHP_FE(dom_import_simplexml, arginfo_dom_import_simplexml)
  542. PHP_FE_END
  543. };
  544. static zend_object_handlers* dom_get_obj_handlers(TSRMLS_D) {
  545. return &dom_object_handlers;
  546. }
  547. static const zend_module_dep dom_deps[] = {
  548. ZEND_MOD_REQUIRED("libxml")
  549. ZEND_MOD_CONFLICTS("domxml")
  550. ZEND_MOD_END
  551. };
  552. zend_module_entry dom_module_entry = { /* {{{ */
  553. STANDARD_MODULE_HEADER_EX, NULL,
  554. dom_deps,
  555. "dom",
  556. dom_functions,
  557. PHP_MINIT(dom),
  558. PHP_MSHUTDOWN(dom),
  559. NULL,
  560. NULL,
  561. PHP_MINFO(dom),
  562. DOM_API_VERSION, /* Extension versionnumber */
  563. STANDARD_MODULE_PROPERTIES
  564. };
  565. /* }}} */
  566. #ifdef COMPILE_DL_DOM
  567. ZEND_GET_MODULE(dom)
  568. #endif
  569. /* {{{ PHP_MINIT_FUNCTION(dom) */
  570. PHP_MINIT_FUNCTION(dom)
  571. {
  572. zend_class_entry ce;
  573. memcpy(&dom_object_handlers, zend_get_std_object_handlers(), sizeof(zend_object_handlers));
  574. dom_object_handlers.read_property = dom_read_property;
  575. dom_object_handlers.write_property = dom_write_property;
  576. dom_object_handlers.get_property_ptr_ptr = dom_get_property_ptr_ptr;
  577. dom_object_handlers.clone_obj = dom_objects_store_clone_obj;
  578. dom_object_handlers.has_property = dom_property_exists;
  579. dom_object_handlers.get_debug_info = dom_get_debug_info;
  580. memcpy(&dom_nnodemap_object_handlers, &dom_object_handlers, sizeof(zend_object_handlers));
  581. dom_nnodemap_object_handlers.read_dimension = dom_nodelist_read_dimension;
  582. dom_nnodemap_object_handlers.has_dimension = dom_nodelist_has_dimension;
  583. zend_hash_init(&classes, 0, NULL, NULL, 1);
  584. INIT_CLASS_ENTRY(ce, "DOMException", php_dom_domexception_class_functions);
  585. dom_domexception_class_entry = zend_register_internal_class_ex(&ce, zend_exception_get_default(TSRMLS_C), NULL TSRMLS_CC);
  586. dom_domexception_class_entry->ce_flags |= ZEND_ACC_FINAL;
  587. zend_declare_property_long(dom_domexception_class_entry, "code", sizeof("code")-1, 0, ZEND_ACC_PUBLIC TSRMLS_CC);
  588. REGISTER_DOM_CLASS(ce, "DOMStringList", NULL, php_dom_domstringlist_class_functions, dom_domstringlist_class_entry);
  589. zend_hash_init(&dom_domstringlist_prop_handlers, 0, NULL, NULL, 1);
  590. dom_register_prop_handler(&dom_domstringlist_prop_handlers, "length", dom_domstringlist_length_read, NULL TSRMLS_CC);
  591. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_domstringlist_prop_handlers, sizeof(dom_domstringlist_prop_handlers), NULL);
  592. REGISTER_DOM_CLASS(ce, "DOMNameList", NULL, php_dom_namelist_class_functions, dom_namelist_class_entry);
  593. zend_hash_init(&dom_namelist_prop_handlers, 0, NULL, NULL, 1);
  594. dom_register_prop_handler(&dom_namelist_prop_handlers, "length", dom_namelist_length_read, NULL TSRMLS_CC);
  595. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_namelist_prop_handlers, sizeof(dom_namelist_prop_handlers), NULL);
  596. REGISTER_DOM_CLASS(ce, "DOMImplementationList", NULL, php_dom_domimplementationlist_class_functions, dom_domimplementationlist_class_entry);
  597. zend_hash_init(&dom_domimplementationlist_prop_handlers, 0, NULL, NULL, 1);
  598. dom_register_prop_handler(&dom_domimplementationlist_prop_handlers, "length", dom_domimplementationlist_length_read, NULL TSRMLS_CC);
  599. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_domimplementationlist_prop_handlers, sizeof(dom_domimplementationlist_prop_handlers), NULL);
  600. REGISTER_DOM_CLASS(ce, "DOMImplementationSource", NULL, php_dom_domimplementationsource_class_functions, dom_domimplementationsource_class_entry);
  601. REGISTER_DOM_CLASS(ce, "DOMImplementation", NULL, php_dom_domimplementation_class_functions, dom_domimplementation_class_entry);
  602. REGISTER_DOM_CLASS(ce, "DOMNode", NULL, php_dom_node_class_functions, dom_node_class_entry);
  603. zend_hash_init(&dom_node_prop_handlers, 0, NULL, NULL, 1);
  604. dom_register_prop_handler(&dom_node_prop_handlers, "nodeName", dom_node_node_name_read, NULL TSRMLS_CC);
  605. dom_register_prop_handler(&dom_node_prop_handlers, "nodeValue", dom_node_node_value_read, dom_node_node_value_write TSRMLS_CC);
  606. dom_register_prop_handler(&dom_node_prop_handlers, "nodeType", dom_node_node_type_read, NULL TSRMLS_CC);
  607. dom_register_prop_handler(&dom_node_prop_handlers, "parentNode", dom_node_parent_node_read, NULL TSRMLS_CC);
  608. dom_register_prop_handler(&dom_node_prop_handlers, "childNodes", dom_node_child_nodes_read, NULL TSRMLS_CC);
  609. dom_register_prop_handler(&dom_node_prop_handlers, "firstChild", dom_node_first_child_read, NULL TSRMLS_CC);
  610. dom_register_prop_handler(&dom_node_prop_handlers, "lastChild", dom_node_last_child_read, NULL TSRMLS_CC);
  611. dom_register_prop_handler(&dom_node_prop_handlers, "previousSibling", dom_node_previous_sibling_read, NULL TSRMLS_CC);
  612. dom_register_prop_handler(&dom_node_prop_handlers, "nextSibling", dom_node_next_sibling_read, NULL TSRMLS_CC);
  613. dom_register_prop_handler(&dom_node_prop_handlers, "attributes", dom_node_attributes_read, NULL TSRMLS_CC);
  614. dom_register_prop_handler(&dom_node_prop_handlers, "ownerDocument", dom_node_owner_document_read, NULL TSRMLS_CC);
  615. dom_register_prop_handler(&dom_node_prop_handlers, "namespaceURI", dom_node_namespace_uri_read, NULL TSRMLS_CC);
  616. dom_register_prop_handler(&dom_node_prop_handlers, "prefix", dom_node_prefix_read, dom_node_prefix_write TSRMLS_CC);
  617. dom_register_prop_handler(&dom_node_prop_handlers, "localName", dom_node_local_name_read, NULL TSRMLS_CC);
  618. dom_register_prop_handler(&dom_node_prop_handlers, "baseURI", dom_node_base_uri_read, NULL TSRMLS_CC);
  619. dom_register_prop_handler(&dom_node_prop_handlers, "textContent", dom_node_text_content_read, dom_node_text_content_write TSRMLS_CC);
  620. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_node_prop_handlers, sizeof(dom_node_prop_handlers), NULL);
  621. REGISTER_DOM_CLASS(ce, "DOMNameSpaceNode", NULL, NULL, dom_namespace_node_class_entry);
  622. zend_hash_init(&dom_namespace_node_prop_handlers, 0, NULL, NULL, 1);
  623. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "nodeName", dom_node_node_name_read, NULL TSRMLS_CC);
  624. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "nodeValue", dom_node_node_value_read, NULL TSRMLS_CC);
  625. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "nodeType", dom_node_node_type_read, NULL TSRMLS_CC);
  626. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "prefix", dom_node_prefix_read, NULL TSRMLS_CC);
  627. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "localName", dom_node_local_name_read, NULL TSRMLS_CC);
  628. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "namespaceURI", dom_node_namespace_uri_read, NULL TSRMLS_CC);
  629. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "ownerDocument", dom_node_owner_document_read, NULL TSRMLS_CC);
  630. dom_register_prop_handler(&dom_namespace_node_prop_handlers, "parentNode", dom_node_parent_node_read, NULL TSRMLS_CC);
  631. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_namespace_node_prop_handlers, sizeof(dom_namespace_node_prop_handlers), NULL);
  632. REGISTER_DOM_CLASS(ce, "DOMDocumentFragment", dom_node_class_entry, php_dom_documentfragment_class_functions, dom_documentfragment_class_entry);
  633. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_node_prop_handlers, sizeof(dom_node_prop_handlers), NULL);
  634. REGISTER_DOM_CLASS(ce, "DOMDocument", dom_node_class_entry, php_dom_document_class_functions, dom_document_class_entry);
  635. zend_hash_init(&dom_document_prop_handlers, 0, NULL, NULL, 1);
  636. dom_register_prop_handler(&dom_document_prop_handlers, "doctype", dom_document_doctype_read, NULL TSRMLS_CC);
  637. dom_register_prop_handler(&dom_document_prop_handlers, "implementation", dom_document_implementation_read, NULL TSRMLS_CC);
  638. dom_register_prop_handler(&dom_document_prop_handlers, "documentElement", dom_document_document_element_read, NULL TSRMLS_CC);
  639. dom_register_prop_handler(&dom_document_prop_handlers, "actualEncoding", dom_document_encoding_read, NULL TSRMLS_CC);
  640. dom_register_prop_handler(&dom_document_prop_handlers, "encoding", dom_document_encoding_read, dom_document_encoding_write TSRMLS_CC);
  641. dom_register_prop_handler(&dom_document_prop_handlers, "xmlEncoding", dom_document_encoding_read, NULL TSRMLS_CC);
  642. dom_register_prop_handler(&dom_document_prop_handlers, "standalone", dom_document_standalone_read, dom_document_standalone_write TSRMLS_CC);
  643. dom_register_prop_handler(&dom_document_prop_handlers, "xmlStandalone", dom_document_standalone_read, dom_document_standalone_write TSRMLS_CC);
  644. dom_register_prop_handler(&dom_document_prop_handlers, "version", dom_document_version_read, dom_document_version_write TSRMLS_CC);
  645. dom_register_prop_handler(&dom_document_prop_handlers, "xmlVersion", dom_document_version_read, dom_document_version_write TSRMLS_CC);
  646. dom_register_prop_handler(&dom_document_prop_handlers, "strictErrorChecking", dom_document_strict_error_checking_read, dom_document_strict_error_checking_write TSRMLS_CC);
  647. dom_register_prop_handler(&dom_document_prop_handlers, "documentURI", dom_document_document_uri_read, dom_document_document_uri_write TSRMLS_CC);
  648. dom_register_prop_handler(&dom_document_prop_handlers, "config", dom_document_config_read, NULL TSRMLS_CC);
  649. dom_register_prop_handler(&dom_document_prop_handlers, "formatOutput", dom_document_format_output_read, dom_document_format_output_write TSRMLS_CC);
  650. dom_register_prop_handler(&dom_document_prop_handlers, "validateOnParse", dom_document_validate_on_parse_read, dom_document_validate_on_parse_write TSRMLS_CC);
  651. dom_register_prop_handler(&dom_document_prop_handlers, "resolveExternals", dom_document_resolve_externals_read, dom_document_resolve_externals_write TSRMLS_CC);
  652. dom_register_prop_handler(&dom_document_prop_handlers, "preserveWhiteSpace", dom_document_preserve_whitespace_read, dom_document_preserve_whitespace_write TSRMLS_CC);
  653. dom_register_prop_handler(&dom_document_prop_handlers, "recover", dom_document_recover_read, dom_document_recover_write TSRMLS_CC);
  654. dom_register_prop_handler(&dom_document_prop_handlers, "substituteEntities", dom_document_substitue_entities_read, dom_document_substitue_entities_write TSRMLS_CC);
  655. zend_hash_merge(&dom_document_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  656. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_document_prop_handlers, sizeof(dom_document_prop_handlers), NULL);
  657. INIT_CLASS_ENTRY(ce, "DOMNodeList", php_dom_nodelist_class_functions);
  658. ce.create_object = dom_nnodemap_objects_new;
  659. dom_nodelist_class_entry = zend_register_internal_class_ex(&ce, NULL, NULL TSRMLS_CC);
  660. dom_nodelist_class_entry->get_iterator = php_dom_get_iterator;
  661. zend_class_implements(dom_nodelist_class_entry TSRMLS_CC, 1, zend_ce_traversable);
  662. zend_hash_init(&dom_nodelist_prop_handlers, 0, NULL, NULL, 1);
  663. dom_register_prop_handler(&dom_nodelist_prop_handlers, "length", dom_nodelist_length_read, NULL TSRMLS_CC);
  664. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_nodelist_prop_handlers, sizeof(dom_nodelist_prop_handlers), NULL);
  665. INIT_CLASS_ENTRY(ce, "DOMNamedNodeMap", php_dom_namednodemap_class_functions);
  666. ce.create_object = dom_nnodemap_objects_new;
  667. dom_namednodemap_class_entry = zend_register_internal_class_ex(&ce, NULL, NULL TSRMLS_CC);
  668. dom_namednodemap_class_entry->get_iterator = php_dom_get_iterator;
  669. zend_class_implements(dom_namednodemap_class_entry TSRMLS_CC, 1, zend_ce_traversable);
  670. zend_hash_init(&dom_namednodemap_prop_handlers, 0, NULL, NULL, 1);
  671. dom_register_prop_handler(&dom_namednodemap_prop_handlers, "length", dom_namednodemap_length_read, NULL TSRMLS_CC);
  672. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_namednodemap_prop_handlers, sizeof(dom_namednodemap_prop_handlers), NULL);
  673. REGISTER_DOM_CLASS(ce, "DOMCharacterData", dom_node_class_entry, php_dom_characterdata_class_functions, dom_characterdata_class_entry);
  674. zend_hash_init(&dom_characterdata_prop_handlers, 0, NULL, NULL, 1);
  675. dom_register_prop_handler(&dom_characterdata_prop_handlers, "data", dom_characterdata_data_read, dom_characterdata_data_write TSRMLS_CC);
  676. dom_register_prop_handler(&dom_characterdata_prop_handlers, "length", dom_characterdata_length_read, NULL TSRMLS_CC);
  677. zend_hash_merge(&dom_characterdata_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  678. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_characterdata_prop_handlers, sizeof(dom_characterdata_prop_handlers), NULL);
  679. REGISTER_DOM_CLASS(ce, "DOMAttr", dom_node_class_entry, php_dom_attr_class_functions, dom_attr_class_entry);
  680. zend_hash_init(&dom_attr_prop_handlers, 0, NULL, NULL, 1);
  681. dom_register_prop_handler(&dom_attr_prop_handlers, "name", dom_attr_name_read, NULL TSRMLS_CC);
  682. dom_register_prop_handler(&dom_attr_prop_handlers, "specified", dom_attr_specified_read, NULL TSRMLS_CC);
  683. dom_register_prop_handler(&dom_attr_prop_handlers, "value", dom_attr_value_read, dom_attr_value_write TSRMLS_CC);
  684. dom_register_prop_handler(&dom_attr_prop_handlers, "ownerElement", dom_attr_owner_element_read, NULL TSRMLS_CC);
  685. dom_register_prop_handler(&dom_attr_prop_handlers, "schemaTypeInfo", dom_attr_schema_type_info_read, NULL TSRMLS_CC);
  686. zend_hash_merge(&dom_attr_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  687. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_attr_prop_handlers, sizeof(dom_attr_prop_handlers), NULL);
  688. REGISTER_DOM_CLASS(ce, "DOMElement", dom_node_class_entry, php_dom_element_class_functions, dom_element_class_entry);
  689. zend_hash_init(&dom_element_prop_handlers, 0, NULL, NULL, 1);
  690. dom_register_prop_handler(&dom_element_prop_handlers, "tagName", dom_element_tag_name_read, NULL TSRMLS_CC);
  691. dom_register_prop_handler(&dom_element_prop_handlers, "schemaTypeInfo", dom_element_schema_type_info_read, NULL TSRMLS_CC);
  692. zend_hash_merge(&dom_element_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  693. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_element_prop_handlers, sizeof(dom_element_prop_handlers), NULL);
  694. REGISTER_DOM_CLASS(ce, "DOMText", dom_characterdata_class_entry, php_dom_text_class_functions, dom_text_class_entry);
  695. zend_hash_init(&dom_text_prop_handlers, 0, NULL, NULL, 1);
  696. dom_register_prop_handler(&dom_text_prop_handlers, "wholeText", dom_text_whole_text_read, NULL TSRMLS_CC);
  697. zend_hash_merge(&dom_text_prop_handlers, &dom_characterdata_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  698. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_text_prop_handlers, sizeof(dom_text_prop_handlers), NULL);
  699. REGISTER_DOM_CLASS(ce, "DOMComment", dom_characterdata_class_entry, php_dom_comment_class_functions, dom_comment_class_entry);
  700. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_characterdata_prop_handlers, sizeof(dom_typeinfo_prop_handlers), NULL);
  701. REGISTER_DOM_CLASS(ce, "DOMTypeinfo", NULL, php_dom_typeinfo_class_functions, dom_typeinfo_class_entry);
  702. zend_hash_init(&dom_typeinfo_prop_handlers, 0, NULL, NULL, 1);
  703. dom_register_prop_handler(&dom_typeinfo_prop_handlers, "typeName", dom_typeinfo_type_name_read, NULL TSRMLS_CC);
  704. dom_register_prop_handler(&dom_typeinfo_prop_handlers, "typeNamespace", dom_typeinfo_type_namespace_read, NULL TSRMLS_CC);
  705. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_typeinfo_prop_handlers, sizeof(dom_typeinfo_prop_handlers), NULL);
  706. REGISTER_DOM_CLASS(ce, "DOMUserDataHandler", NULL, php_dom_userdatahandler_class_functions, dom_userdatahandler_class_entry);
  707. REGISTER_DOM_CLASS(ce, "DOMDomError", NULL, php_dom_domerror_class_functions, dom_domerror_class_entry);
  708. zend_hash_init(&dom_domerror_prop_handlers, 0, NULL, NULL, 1);
  709. dom_register_prop_handler(&dom_domerror_prop_handlers, "severity", dom_domerror_severity_read, NULL TSRMLS_CC);
  710. dom_register_prop_handler(&dom_domerror_prop_handlers, "message", dom_domerror_message_read, NULL TSRMLS_CC);
  711. dom_register_prop_handler(&dom_domerror_prop_handlers, "type", dom_domerror_type_read, NULL TSRMLS_CC);
  712. dom_register_prop_handler(&dom_domerror_prop_handlers, "relatedException", dom_domerror_related_exception_read, NULL TSRMLS_CC);
  713. dom_register_prop_handler(&dom_domerror_prop_handlers, "related_data", dom_domerror_related_data_read, NULL TSRMLS_CC);
  714. dom_register_prop_handler(&dom_domerror_prop_handlers, "location", dom_domerror_location_read, NULL TSRMLS_CC);
  715. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_domerror_prop_handlers, sizeof(dom_domerror_prop_handlers), NULL);
  716. REGISTER_DOM_CLASS(ce, "DOMErrorHandler", NULL, php_dom_domerrorhandler_class_functions, dom_domerrorhandler_class_entry);
  717. REGISTER_DOM_CLASS(ce, "DOMLocator", NULL, php_dom_domlocator_class_functions, dom_domlocator_class_entry);
  718. zend_hash_init(&dom_domlocator_prop_handlers, 0, NULL, NULL, 1);
  719. dom_register_prop_handler(&dom_domlocator_prop_handlers, "lineNumber", dom_domlocator_line_number_read, NULL TSRMLS_CC);
  720. dom_register_prop_handler(&dom_domlocator_prop_handlers, "columnNumber", dom_domlocator_column_number_read, NULL TSRMLS_CC);
  721. dom_register_prop_handler(&dom_domlocator_prop_handlers, "offset", dom_domlocator_offset_read, NULL TSRMLS_CC);
  722. dom_register_prop_handler(&dom_domlocator_prop_handlers, "relatedNode", dom_domlocator_related_node_read, NULL TSRMLS_CC);
  723. dom_register_prop_handler(&dom_domlocator_prop_handlers, "uri", dom_domlocator_uri_read, NULL TSRMLS_CC);
  724. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_domlocator_prop_handlers, sizeof(dom_domlocator_prop_handlers), NULL);
  725. REGISTER_DOM_CLASS(ce, "DOMConfiguration", NULL, php_dom_domconfiguration_class_functions, dom_domconfiguration_class_entry);
  726. REGISTER_DOM_CLASS(ce, "DOMCdataSection", dom_text_class_entry, php_dom_cdatasection_class_functions, dom_cdatasection_class_entry);
  727. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_text_prop_handlers, sizeof(dom_documenttype_prop_handlers), NULL);
  728. REGISTER_DOM_CLASS(ce, "DOMDocumentType", dom_node_class_entry, php_dom_documenttype_class_functions, dom_documenttype_class_entry);
  729. zend_hash_init(&dom_documenttype_prop_handlers, 0, NULL, NULL, 1);
  730. dom_register_prop_handler(&dom_documenttype_prop_handlers, "name", dom_documenttype_name_read, NULL TSRMLS_CC);
  731. dom_register_prop_handler(&dom_documenttype_prop_handlers, "entities", dom_documenttype_entities_read, NULL TSRMLS_CC);
  732. dom_register_prop_handler(&dom_documenttype_prop_handlers, "notations", dom_documenttype_notations_read, NULL TSRMLS_CC);
  733. dom_register_prop_handler(&dom_documenttype_prop_handlers, "publicId", dom_documenttype_public_id_read, NULL TSRMLS_CC);
  734. dom_register_prop_handler(&dom_documenttype_prop_handlers, "systemId", dom_documenttype_system_id_read, NULL TSRMLS_CC);
  735. dom_register_prop_handler(&dom_documenttype_prop_handlers, "internalSubset", dom_documenttype_internal_subset_read, NULL TSRMLS_CC);
  736. zend_hash_merge(&dom_documenttype_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  737. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_documenttype_prop_handlers, sizeof(dom_documenttype_prop_handlers), NULL);
  738. REGISTER_DOM_CLASS(ce, "DOMNotation", dom_node_class_entry, php_dom_notation_class_functions, dom_notation_class_entry);
  739. zend_hash_init(&dom_notation_prop_handlers, 0, NULL, NULL, 1);
  740. dom_register_prop_handler(&dom_notation_prop_handlers, "publicId", dom_notation_public_id_read, NULL TSRMLS_CC);
  741. dom_register_prop_handler(&dom_notation_prop_handlers, "systemId", dom_notation_system_id_read, NULL TSRMLS_CC);
  742. zend_hash_merge(&dom_notation_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  743. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_notation_prop_handlers, sizeof(dom_notation_prop_handlers), NULL);
  744. REGISTER_DOM_CLASS(ce, "DOMEntity", dom_node_class_entry, php_dom_entity_class_functions, dom_entity_class_entry);
  745. zend_hash_init(&dom_entity_prop_handlers, 0, NULL, NULL, 1);
  746. dom_register_prop_handler(&dom_entity_prop_handlers, "publicId", dom_entity_public_id_read, NULL TSRMLS_CC);
  747. dom_register_prop_handler(&dom_entity_prop_handlers, "systemId", dom_entity_system_id_read, NULL TSRMLS_CC);
  748. dom_register_prop_handler(&dom_entity_prop_handlers, "notationName", dom_entity_notation_name_read, NULL TSRMLS_CC);
  749. dom_register_prop_handler(&dom_entity_prop_handlers, "actualEncoding", dom_entity_actual_encoding_read, dom_entity_actual_encoding_write TSRMLS_CC);
  750. dom_register_prop_handler(&dom_entity_prop_handlers, "encoding", dom_entity_encoding_read, dom_entity_encoding_write TSRMLS_CC);
  751. dom_register_prop_handler(&dom_entity_prop_handlers, "version", dom_entity_version_read, dom_entity_version_write TSRMLS_CC);
  752. zend_hash_merge(&dom_entity_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  753. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_entity_prop_handlers, sizeof(dom_entity_prop_handlers), NULL);
  754. REGISTER_DOM_CLASS(ce, "DOMEntityReference", dom_node_class_entry, php_dom_entityreference_class_functions, dom_entityreference_class_entry);
  755. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_node_prop_handlers, sizeof(dom_entity_prop_handlers), NULL);
  756. REGISTER_DOM_CLASS(ce, "DOMProcessingInstruction", dom_node_class_entry, php_dom_processinginstruction_class_functions, dom_processinginstruction_class_entry);
  757. zend_hash_init(&dom_processinginstruction_prop_handlers, 0, NULL, NULL, 1);
  758. dom_register_prop_handler(&dom_processinginstruction_prop_handlers, "target", dom_processinginstruction_target_read, NULL TSRMLS_CC);
  759. dom_register_prop_handler(&dom_processinginstruction_prop_handlers, "data", dom_processinginstruction_data_read, dom_processinginstruction_data_write TSRMLS_CC);
  760. zend_hash_merge(&dom_processinginstruction_prop_handlers, &dom_node_prop_handlers, NULL, NULL, sizeof(dom_prop_handler), 0);
  761. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_processinginstruction_prop_handlers, sizeof(dom_processinginstruction_prop_handlers), NULL);
  762. REGISTER_DOM_CLASS(ce, "DOMStringExtend", NULL, php_dom_string_extend_class_functions, dom_string_extend_class_entry);
  763. #if defined(LIBXML_XPATH_ENABLED)
  764. INIT_CLASS_ENTRY(ce, "DOMXPath", php_dom_xpath_class_functions);
  765. ce.create_object = dom_xpath_objects_new;
  766. dom_xpath_class_entry = zend_register_internal_class_ex(&ce, NULL, NULL TSRMLS_CC);
  767. zend_hash_init(&dom_xpath_prop_handlers, 0, NULL, NULL, 1);
  768. dom_register_prop_handler(&dom_xpath_prop_handlers, "document", dom_xpath_document_read, NULL TSRMLS_CC);
  769. zend_hash_add(&classes, ce.name, ce.name_length + 1, &dom_xpath_prop_handlers, sizeof(dom_xpath_prop_handlers), NULL);
  770. #endif
  771. REGISTER_LONG_CONSTANT("XML_ELEMENT_NODE", XML_ELEMENT_NODE, CONST_CS | CONST_PERSISTENT);
  772. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_NODE", XML_ATTRIBUTE_NODE, CONST_CS | CONST_PERSISTENT);
  773. REGISTER_LONG_CONSTANT("XML_TEXT_NODE", XML_TEXT_NODE, CONST_CS | CONST_PERSISTENT);
  774. REGISTER_LONG_CONSTANT("XML_CDATA_SECTION_NODE", XML_CDATA_SECTION_NODE, CONST_CS | CONST_PERSISTENT);
  775. REGISTER_LONG_CONSTANT("XML_ENTITY_REF_NODE", XML_ENTITY_REF_NODE, CONST_CS | CONST_PERSISTENT);
  776. REGISTER_LONG_CONSTANT("XML_ENTITY_NODE", XML_ENTITY_NODE, CONST_CS | CONST_PERSISTENT);
  777. REGISTER_LONG_CONSTANT("XML_PI_NODE", XML_PI_NODE, CONST_CS | CONST_PERSISTENT);
  778. REGISTER_LONG_CONSTANT("XML_COMMENT_NODE", XML_COMMENT_NODE, CONST_CS | CONST_PERSISTENT);
  779. REGISTER_LONG_CONSTANT("XML_DOCUMENT_NODE", XML_DOCUMENT_NODE, CONST_CS | CONST_PERSISTENT);
  780. REGISTER_LONG_CONSTANT("XML_DOCUMENT_TYPE_NODE", XML_DOCUMENT_TYPE_NODE, CONST_CS | CONST_PERSISTENT);
  781. REGISTER_LONG_CONSTANT("XML_DOCUMENT_FRAG_NODE", XML_DOCUMENT_FRAG_NODE, CONST_CS | CONST_PERSISTENT);
  782. REGISTER_LONG_CONSTANT("XML_NOTATION_NODE", XML_NOTATION_NODE, CONST_CS | CONST_PERSISTENT);
  783. REGISTER_LONG_CONSTANT("XML_HTML_DOCUMENT_NODE", XML_HTML_DOCUMENT_NODE, CONST_CS | CONST_PERSISTENT);
  784. REGISTER_LONG_CONSTANT("XML_DTD_NODE", XML_DTD_NODE, CONST_CS | CONST_PERSISTENT);
  785. REGISTER_LONG_CONSTANT("XML_ELEMENT_DECL_NODE", XML_ELEMENT_DECL, CONST_CS | CONST_PERSISTENT);
  786. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_DECL_NODE", XML_ATTRIBUTE_DECL, CONST_CS | CONST_PERSISTENT);
  787. REGISTER_LONG_CONSTANT("XML_ENTITY_DECL_NODE", XML_ENTITY_DECL, CONST_CS | CONST_PERSISTENT);
  788. REGISTER_LONG_CONSTANT("XML_NAMESPACE_DECL_NODE", XML_NAMESPACE_DECL, CONST_CS | CONST_PERSISTENT);
  789. #ifdef XML_GLOBAL_NAMESPACE
  790. REGISTER_LONG_CONSTANT("XML_GLOBAL_NAMESPACE", XML_GLOBAL_NAMESPACE, CONST_CS | CONST_PERSISTENT);
  791. #endif
  792. REGISTER_LONG_CONSTANT("XML_LOCAL_NAMESPACE", XML_LOCAL_NAMESPACE, CONST_CS | CONST_PERSISTENT);
  793. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_CDATA", XML_ATTRIBUTE_CDATA, CONST_CS | CONST_PERSISTENT);
  794. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_ID", XML_ATTRIBUTE_ID, CONST_CS | CONST_PERSISTENT);
  795. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_IDREF", XML_ATTRIBUTE_IDREF, CONST_CS | CONST_PERSISTENT);
  796. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_IDREFS", XML_ATTRIBUTE_IDREFS, CONST_CS | CONST_PERSISTENT);
  797. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_ENTITY", XML_ATTRIBUTE_ENTITIES, CONST_CS | CONST_PERSISTENT);
  798. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_NMTOKEN", XML_ATTRIBUTE_NMTOKEN, CONST_CS | CONST_PERSISTENT);
  799. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_NMTOKENS", XML_ATTRIBUTE_NMTOKENS, CONST_CS | CONST_PERSISTENT);
  800. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_ENUMERATION", XML_ATTRIBUTE_ENUMERATION, CONST_CS | CONST_PERSISTENT);
  801. REGISTER_LONG_CONSTANT("XML_ATTRIBUTE_NOTATION", XML_ATTRIBUTE_NOTATION, CONST_CS | CONST_PERSISTENT);
  802. /* DOMException Codes */
  803. REGISTER_LONG_CONSTANT("DOM_PHP_ERR", PHP_ERR, CONST_CS | CONST_PERSISTENT);
  804. REGISTER_LONG_CONSTANT("DOM_INDEX_SIZE_ERR", INDEX_SIZE_ERR, CONST_CS | CONST_PERSISTENT);
  805. REGISTER_LONG_CONSTANT("DOMSTRING_SIZE_ERR", DOMSTRING_SIZE_ERR, CONST_CS | CONST_PERSISTENT);
  806. REGISTER_LONG_CONSTANT("DOM_HIERARCHY_REQUEST_ERR", HIERARCHY_REQUEST_ERR, CONST_CS | CONST_PERSISTENT);
  807. REGISTER_LONG_CONSTANT("DOM_WRONG_DOCUMENT_ERR", WRONG_DOCUMENT_ERR, CONST_CS | CONST_PERSISTENT);
  808. REGISTER_LONG_CONSTANT("DOM_INVALID_CHARACTER_ERR", INVALID_CHARACTER_ERR, CONST_CS | CONST_PERSISTENT);
  809. REGISTER_LONG_CONSTANT("DOM_NO_DATA_ALLOWED_ERR", NO_DATA_ALLOWED_ERR, CONST_CS | CONST_PERSISTENT);
  810. REGISTER_LONG_CONSTANT("DOM_NO_MODIFICATION_ALLOWED_ERR", NO_MODIFICATION_ALLOWED_ERR, CONST_CS | CONST_PERSISTENT);
  811. REGISTER_LONG_CONSTANT("DOM_NOT_FOUND_ERR", NOT_FOUND_ERR, CONST_CS | CONST_PERSISTENT);
  812. REGISTER_LONG_CONSTANT("DOM_NOT_SUPPORTED_ERR", NOT_SUPPORTED_ERR, CONST_CS | CONST_PERSISTENT);
  813. REGISTER_LONG_CONSTANT("DOM_INUSE_ATTRIBUTE_ERR", INUSE_ATTRIBUTE_ERR, CONST_CS | CONST_PERSISTENT);
  814. REGISTER_LONG_CONSTANT("DOM_INVALID_STATE_ERR", INVALID_STATE_ERR, CONST_CS | CONST_PERSISTENT);
  815. REGISTER_LONG_CONSTANT("DOM_SYNTAX_ERR", SYNTAX_ERR, CONST_CS | CONST_PERSISTENT);
  816. REGISTER_LONG_CONSTANT("DOM_INVALID_MODIFICATION_ERR", INVALID_MODIFICATION_ERR, CONST_CS | CONST_PERSISTENT);
  817. REGISTER_LONG_CONSTANT("DOM_NAMESPACE_ERR", NAMESPACE_ERR, CONST_CS | CONST_PERSISTENT);
  818. REGISTER_LONG_CONSTANT("DOM_INVALID_ACCESS_ERR", INVALID_ACCESS_ERR, CONST_CS | CONST_PERSISTENT);
  819. REGISTER_LONG_CONSTANT("DOM_VALIDATION_ERR", VALIDATION_ERR, CONST_CS | CONST_PERSISTENT);
  820. php_libxml_register_export(dom_node_class_entry, php_dom_export_node);
  821. return SUCCESS;
  822. }
  823. /* }}} */
  824. /* {{{ */
  825. PHP_MINFO_FUNCTION(dom)
  826. {
  827. php_info_print_table_start();
  828. php_info_print_table_row(2, "DOM/XML", "enabled");
  829. php_info_print_table_row(2, "DOM/XML API Version", DOM_API_VERSION);
  830. php_info_print_table_row(2, "libxml Version", LIBXML_DOTTED_VERSION);
  831. #if defined(LIBXML_HTML_ENABLED)
  832. php_info_print_table_row(2, "HTML Support", "enabled");
  833. #endif
  834. #if defined(LIBXML_XPATH_ENABLED)
  835. php_info_print_table_row(2, "XPath Support", "enabled");
  836. #endif
  837. #if defined(LIBXML_XPTR_ENABLED)
  838. php_info_print_table_row(2, "XPointer Support", "enabled");
  839. #endif
  840. #ifdef LIBXML_SCHEMAS_ENABLED
  841. php_info_print_table_row(2, "Schema Support", "enabled");
  842. php_info_print_table_row(2, "RelaxNG Support", "enabled");
  843. #endif
  844. php_info_print_table_end();
  845. }
  846. /* }}} */
  847. PHP_MSHUTDOWN_FUNCTION(dom) /* {{{ */
  848. {
  849. zend_hash_destroy(&dom_domstringlist_prop_handlers);
  850. zend_hash_destroy(&dom_namelist_prop_handlers);
  851. zend_hash_destroy(&dom_domimplementationlist_prop_handlers);
  852. zend_hash_destroy(&dom_document_prop_handlers);
  853. zend_hash_destroy(&dom_node_prop_handlers);
  854. zend_hash_destroy(&dom_namespace_node_prop_handlers);
  855. zend_hash_destroy(&dom_nodelist_prop_handlers);
  856. zend_hash_destroy(&dom_namednodemap_prop_handlers);
  857. zend_hash_destroy(&dom_characterdata_prop_handlers);
  858. zend_hash_destroy(&dom_attr_prop_handlers);
  859. zend_hash_destroy(&dom_element_prop_handlers);
  860. zend_hash_destroy(&dom_text_prop_handlers);
  861. zend_hash_destroy(&dom_typeinfo_prop_handlers);
  862. zend_hash_destroy(&dom_domerror_prop_handlers);
  863. zend_hash_destroy(&dom_domlocator_prop_handlers);
  864. zend_hash_destroy(&dom_documenttype_prop_handlers);
  865. zend_hash_destroy(&dom_notation_prop_handlers);
  866. zend_hash_destroy(&dom_entity_prop_handlers);
  867. zend_hash_destroy(&dom_processinginstruction_prop_handlers);
  868. #if defined(LIBXML_XPATH_ENABLED)
  869. zend_hash_destroy(&dom_xpath_prop_handlers);
  870. #endif
  871. zend_hash_destroy(&classes);
  872. /* If you want do find memleaks in this module, compile libxml2 with --with-mem-debug and
  873. uncomment the following line, this will tell you the amount of not freed memory
  874. and the total used memory into apaches error_log */
  875. /* xmlMemoryDump();*/
  876. return SUCCESS;
  877. }
  878. /* }}} */
  879. /* {{{ node_list_unlink */
  880. void node_list_unlink(xmlNodePtr node TSRMLS_DC)
  881. {
  882. dom_object *wrapper;
  883. while (node != NULL) {
  884. wrapper = php_dom_object_get_data(node);
  885. if (wrapper != NULL ) {
  886. xmlUnlinkNode(node);
  887. } else {
  888. if (node->type == XML_ENTITY_REF_NODE)
  889. break;
  890. node_list_unlink(node->children TSRMLS_CC);
  891. switch (node->type) {
  892. case XML_ATTRIBUTE_DECL:
  893. case XML_DTD_NODE:
  894. case XML_DOCUMENT_TYPE_NODE:
  895. case XML_ENTITY_DECL:
  896. case XML_ATTRIBUTE_NODE:
  897. case XML_TEXT_NODE:
  898. break;
  899. default:
  900. node_list_unlink((xmlNodePtr) node->properties TSRMLS_CC);
  901. }
  902. }
  903. node = node->next;
  904. }
  905. }
  906. /* }}} end node_list_unlink */
  907. #if defined(LIBXML_XPATH_ENABLED)
  908. /* {{{ dom_xpath_objects_free_storage */
  909. void dom_xpath_objects_free_storage(void *object TSRMLS_DC)
  910. {
  911. dom_xpath_object *intern = (dom_xpath_object *)object;
  912. zend_object_std_dtor(&intern->std TSRMLS_CC);
  913. if (intern->ptr != NULL) {
  914. xmlXPathFreeContext((xmlXPathContextPtr) intern->ptr);
  915. php_libxml_decrement_doc_ref((php_libxml_node_object *) intern TSRMLS_CC);
  916. intern->ptr = NULL;
  917. }
  918. if (intern->registered_phpfunctions) {
  919. zend_hash_destroy(intern->registered_phpfunctions);
  920. FREE_HASHTABLE(intern->registered_phpfunctions);
  921. }
  922. if (intern->node_list) {
  923. zend_hash_destroy(intern->node_list);
  924. FREE_HASHTABLE(intern->node_list);
  925. }
  926. efree(object);
  927. }
  928. /* }}} */
  929. #endif
  930. /* {{{ dom_objects_free_storage */
  931. void dom_objects_free_storage(void *object TSRMLS_DC)
  932. {
  933. dom_object *intern = (dom_object *)object;
  934. #if defined(__GNUC__) && __GNUC__ >= 3
  935. int retcount __attribute__((unused)); /* keep compiler quiet */
  936. #else
  937. int retcount;
  938. #endif
  939. zend_object_std_dtor(&intern->std TSRMLS_CC);
  940. if (intern->ptr != NULL && ((php_libxml_node_ptr *)intern->ptr)->node != NULL) {
  941. if (((xmlNodePtr) ((php_libxml_node_ptr *)intern->ptr)->node)->type != XML_DOCUMENT_NODE && ((xmlNodePtr) ((php_libxml_node_ptr *)intern->ptr)->node)->type != XML_HTML_DOCUMENT_NODE) {
  942. php_libxml_node_decrement_resource((php_libxml_node_object *) intern TSRMLS_CC);
  943. } else {
  944. php_libxml_decrement_node_ptr((php_libxml_node_object *) intern TSRMLS_CC);
  945. retcount = php_libxml_decrement_doc_ref((php_libxml_node_object *)intern TSRMLS_CC);
  946. }
  947. intern->ptr = NULL;
  948. }
  949. efree(object);
  950. }
  951. /* }}} */
  952. void dom_namednode_iter(dom_object *basenode, int ntype, dom_object *intern, xmlHashTablePtr ht, xmlChar *local, xmlChar *ns TSRMLS_DC) /* {{{ */
  953. {
  954. dom_nnodemap_object *mapptr;
  955. zval *baseobj = NULL;
  956. mapptr = (dom_nnodemap_object *)intern->ptr;
  957. if (basenode) {
  958. MAKE_STD_ZVAL(baseobj);
  959. baseobj->type = IS_OBJECT;
  960. Z_SET_ISREF_P(baseobj);
  961. baseobj->value.obj.handle = basenode->handle;
  962. baseobj->value.obj.handlers = dom_get_obj_handlers(TSRMLS_C);
  963. zval_copy_ctor(baseobj);
  964. }
  965. mapptr->baseobjptr = baseobj;
  966. mapptr->baseobj = basenode;
  967. mapptr->nodetype = ntype;
  968. mapptr->ht = ht;
  969. mapptr->local = local;
  970. mapptr->ns = ns;
  971. }
  972. /* }}} */
  973. static dom_object* dom_objects_set_class(zend_class_entry *class_type, zend_bool hash_copy TSRMLS_DC) /* {{{ */
  974. {
  975. zend_class_entry *base_class;
  976. dom_object *intern;
  977. if (instanceof_function(class_type, dom_xpath_class_entry TSRMLS_CC)) {
  978. intern = emalloc(sizeof(dom_xpath_object));
  979. memset(intern, 0, sizeof(dom_xpath_object));
  980. } else {
  981. intern = emalloc(sizeof(dom_object));
  982. }
  983. intern->ptr = NULL;
  984. intern->prop_handler = NULL;
  985. intern->document = NULL;
  986. base_class = class_type;
  987. while(base_class->type != ZEND_INTERNAL_CLASS && base_class->parent != NULL) {
  988. base_class = base_class->parent;
  989. }
  990. zend_hash_find(&classes, base_class->name, base_class->name_length + 1, (void **) &intern->prop_handler);
  991. zend_object_std_init(&intern->std, class_type TSRMLS_CC);
  992. if (hash_copy) {
  993. object_properties_init(&intern->std, class_type);
  994. }
  995. return intern;
  996. }
  997. /* }}} */
  998. /* {{{ dom_objects_clone */
  999. void dom_objects_clone(void *object, void **object_clone TSRMLS_DC)
  1000. {
  1001. dom_object *intern = (dom_object *) object;
  1002. dom_object *clone;
  1003. xmlNodePtr node;
  1004. xmlNodePtr cloned_node;
  1005. clone = dom_objects_set_class(intern->std.ce, 0 TSRMLS_CC);
  1006. if (instanceof_function(intern->std.ce, dom_node_class_entry TSRMLS_CC)) {
  1007. node = (xmlNodePtr)dom_object_get_node((dom_object *) object);
  1008. if (node != NULL) {
  1009. cloned_node = xmlDocCopyNode(node, node->doc, 1);
  1010. if (cloned_node != NULL) {
  1011. /* If we cloned a document then we must create new doc proxy */
  1012. if (cloned_node->doc == node->doc) {
  1013. clone->document = intern->document;
  1014. }
  1015. php_libxml_increment_doc_ref((php_libxml_node_object *)clone, cloned_node->doc TSRMLS_CC);
  1016. php_libxml_increment_node_ptr((php_libxml_node_object *)clone, cloned_node, (void *)clone TSRMLS_CC);
  1017. if (intern->document != clone->document) {
  1018. dom_copy_doc_props(intern->document, clone->document);
  1019. }
  1020. }
  1021. }
  1022. }
  1023. *object_clone = (void *) clone;
  1024. }
  1025. /* }}} */
  1026. /* {{{ dom_objects_new */
  1027. zend_object_value dom_objects_new(zend_class_entry *class_type TSRMLS_DC)
  1028. {
  1029. zend_object_value retval;
  1030. dom_object *intern;
  1031. intern = dom_objects_set_class(class_type, 1 TSRMLS_CC);
  1032. retval.handle = zend_objects_store_put(intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, (zend_objects_free_object_storage_t)dom_objects_free_storage, dom_objects_clone TSRMLS_CC);
  1033. intern->handle = retval.handle;
  1034. retval.handlers = dom_get_obj_handlers(TSRMLS_C);
  1035. return retval;
  1036. }
  1037. /* }}} */
  1038. #if defined(LIBXML_XPATH_ENABLED)
  1039. /* {{{ zend_object_value dom_xpath_objects_new(zend_class_entry *class_type TSRMLS_DC) */
  1040. zend_object_value dom_xpath_objects_new(zend_class_entry *class_type TSRMLS_DC)
  1041. {
  1042. zend_object_value retval;
  1043. dom_xpath_object *intern;
  1044. intern = (dom_xpath_object *)dom_objects_set_class(class_type, 1 TSRMLS_CC);
  1045. intern->registerPhpFunctions = 0;
  1046. intern->registered_phpfunctions = NULL;
  1047. intern->node_list = NULL;
  1048. ALLOC_HASHTABLE(intern->registered_phpfunctions);
  1049. zend_hash_init(intern->registered_phpfunctions, 0, NULL, ZVAL_PTR_DTOR, 0);
  1050. retval.handle = zend_objects_store_put(intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, (zend_objects_free_object_storage_t)dom_xpath_objects_free_storage, dom_objects_clone TSRMLS_CC);
  1051. intern->handle = retval.handle;
  1052. retval.handlers = dom_get_obj_handlers(TSRMLS_C);
  1053. return retval;
  1054. }
  1055. /* }}} */
  1056. #endif
  1057. static void dom_nnodemap_object_dtor(void *object, zend_object_handle handle TSRMLS_DC) /* {{{ */
  1058. {
  1059. zval *baseobj;
  1060. dom_object *intern;
  1061. dom_nnodemap_object *objmap;
  1062. intern = (dom_object *)object;
  1063. objmap = (dom_nnodemap_object *)intern->ptr;
  1064. if (objmap) {
  1065. if (objmap->local) {
  1066. xmlFree(objmap->local);
  1067. }
  1068. if (objmap->ns) {
  1069. xmlFree(objmap->ns);
  1070. }
  1071. if (objmap->baseobjptr) {
  1072. baseobj = objmap->baseobjptr;
  1073. zval_ptr_dtor((zval **)&baseobj);
  1074. }
  1075. efree(objmap);
  1076. intern->ptr = NULL;
  1077. }
  1078. }
  1079. /* }}} */
  1080. void dom_nnodemap_objects_free_storage(void *object TSRMLS_DC) /* {{{ */
  1081. {
  1082. dom_object *intern = (dom_object *)object;
  1083. php_libxml_decrement_doc_ref((php_libxml_node_object *)intern TSRMLS_CC);
  1084. zend_object_std_dtor(&intern->std TSRMLS_CC);
  1085. efree(object);
  1086. }
  1087. /* }}} */
  1088. zend_object_value dom_nnodemap_objects_new(zend_class_entry *class_type TSRMLS_DC) /* {{{ */
  1089. {
  1090. zend_object_value retval;
  1091. dom_object *intern;
  1092. dom_nnodemap_object *objmap;
  1093. intern = dom_objects_set_class(class_type, 1 TSRMLS_CC);
  1094. intern->ptr = emalloc(sizeof(dom_nnodemap_object));
  1095. objmap = (dom_nnodemap_object *)intern->ptr;
  1096. objmap->baseobj = NULL;
  1097. objmap->baseobjptr = NULL;
  1098. objmap->nodetype = 0;
  1099. objmap->ht = NULL;
  1100. objmap->local = NULL;
  1101. objmap->ns = NULL;
  1102. retval.handle = zend_objects_store_put(intern, dom_nnodemap_object_dtor, (zend_objects_free_object_storage_t)dom_nnodemap_objects_free_storage, dom_objects_clone TSRMLS_CC);
  1103. intern->handle = retval.handle;
  1104. retval.handlers = &dom_nnodemap_object_handlers;
  1105. return retval;
  1106. }
  1107. /* }}} */
  1108. void php_dom_create_interator(zval *return_value, int ce_type TSRMLS_DC) /* {{{ */
  1109. {
  1110. zend_class_entry *ce;
  1111. if (ce_type == DOM_NAMEDNODEMAP) {
  1112. ce = dom_namednodemap_class_entry;
  1113. } else {
  1114. ce = dom_nodelist_class_entry;
  1115. }
  1116. object_init_ex(return_value, ce);
  1117. }
  1118. /* }}} */
  1119. /* {{{ php_dom_create_object */
  1120. PHP_DOM_EXPORT zval *php_dom_create_object(xmlNodePtr obj, int *found, zval *return_value, dom_object *domobj TSRMLS_DC)
  1121. {
  1122. zval *wrapper;
  1123. zend_class_entry *ce;
  1124. dom_object *intern;
  1125. *found = 0;
  1126. if (!obj) {
  1127. ALLOC_ZVAL(wrapper);
  1128. ZVAL_NULL(wrapper);
  1129. return wrapper;
  1130. }
  1131. if ((intern = (dom_object *) php_dom_object_get_data((void *) obj))) {
  1132. return_value->type = IS_OBJECT;
  1133. Z_SET_ISREF_P(return_value);
  1134. return_value->value.obj.handle = intern->handle;
  1135. return_value->value.obj.handlers = dom_get_obj_handlers(TSRMLS_C);
  1136. zval_copy_ctor(return_value);
  1137. *found = 1;
  1138. return return_value;
  1139. }
  1140. wrapper = return_value;
  1141. switch (obj->type) {
  1142. case XML_DOCUMENT_NODE:
  1143. case XML_HTML_DOCUMENT_NODE:
  1144. {
  1145. ce = dom_document_class_entry;
  1146. break;
  1147. }
  1148. case XML_DTD_NODE:
  1149. case XML_DOCUMENT_TYPE_NODE:
  1150. {
  1151. ce = dom_documenttype_class_entry;
  1152. break;
  1153. }
  1154. case XML_ELEMENT_NODE:
  1155. {
  1156. ce = dom_element_class_entry;
  1157. break;
  1158. }
  1159. case XML_ATTRIBUTE_NODE:
  1160. {
  1161. ce = dom_attr_class_entry;
  1162. break;
  1163. }
  1164. case XML_TEXT_NODE:
  1165. {
  1166. ce = dom_text_class_entry;
  1167. break;
  1168. }
  1169. case XML_COMMENT_NODE:
  1170. {
  1171. ce = dom_comment_class_entry;
  1172. break;
  1173. }
  1174. case XML_PI_NODE:
  1175. {
  1176. ce = dom_processinginstruction_class_entry;
  1177. break;
  1178. }
  1179. case XML_ENTITY_REF_NODE:
  1180. {
  1181. ce = dom_entityreference_class_entry;
  1182. break;
  1183. }
  1184. case XML_ENTITY_DECL:
  1185. case XML_ELEMENT_DECL:
  1186. {
  1187. ce = dom_entity_class_entry;
  1188. break;
  1189. }
  1190. case XML_CDATA_SECTION_NODE:
  1191. {
  1192. ce = dom_cdatasection_class_entry;
  1193. break;
  1194. }
  1195. case XML_DOCUMENT_FRAG_NODE:
  1196. {
  1197. ce = dom_documentfragment_class_entry;
  1198. break;
  1199. }
  1200. case XML_NOTATION_NODE:
  1201. {
  1202. ce = dom_notation_class_entry;
  1203. break;
  1204. }
  1205. case XML_NAMESPACE_DECL:
  1206. {
  1207. ce = dom_namespace_node_class_entry;
  1208. break;
  1209. }
  1210. default:
  1211. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unsupported node type: %d", Z_TYPE_P(obj));
  1212. ZVAL_NULL(wrapper);
  1213. return wrapper;
  1214. }
  1215. if (domobj && domobj->document) {
  1216. ce = dom_get_doc_classmap(domobj->document, ce TSRMLS_CC);
  1217. }
  1218. object_init_ex(wrapper, ce);
  1219. intern = (dom_object *)zend_objects_get_address(wrapper TSRMLS_CC);
  1220. if (obj->doc != NULL) {
  1221. if (domobj != NULL) {
  1222. intern->document = domobj->document;
  1223. }
  1224. php_libxml_increment_doc_ref((php_libxml_node_object *)intern, obj->doc TSRMLS_CC);
  1225. }
  1226. php_libxml_increment_node_ptr((php_libxml_node_object *)intern, obj, (void *)intern TSRMLS_CC);
  1227. return (wrapper);
  1228. }
  1229. /* }}} end php_domobject_new */
  1230. void php_dom_create_implementation(zval **retval TSRMLS_DC) {
  1231. object_init_ex(*retval, dom_domimplementation_class_entry);
  1232. }
  1233. /* {{{ int dom_hierarchy(xmlNodePtr parent, xmlNodePtr child) */
  1234. int dom_hierarchy(xmlNodePtr parent, xmlNodePtr child)
  1235. {
  1236. xmlNodePtr nodep;
  1237. if (parent == NULL || child == NULL || child->doc != parent->doc) {
  1238. return SUCCESS;
  1239. }
  1240. nodep = parent;
  1241. while (nodep) {
  1242. if (nodep == child) {
  1243. return FAILURE;
  1244. }
  1245. nodep = nodep->parent;
  1246. }
  1247. return SUCCESS;
  1248. }
  1249. /* }}} end dom_hierarchy */
  1250. /* {{{ dom_has_feature(char *feature, char *version) */
  1251. int dom_has_feature(char *feature, char *version)
  1252. {
  1253. int retval = 0;
  1254. if (!(strcmp (version, "1.0") && strcmp (version,"2.0") && strcmp(version, ""))) {
  1255. if ((!strcasecmp(feature, "Core") && !strcmp (version, "1.0")) || !strcasecmp(feature, "XML"))
  1256. retval = 1;
  1257. }
  1258. return retval;
  1259. }
  1260. /* }}} end dom_has_feature */
  1261. xmlNode *dom_get_elements_by_tag_name_ns_raw(xmlNodePtr nodep, char *ns, char *local, int *cur, int index) /* {{{ */
  1262. {
  1263. xmlNodePtr ret = NULL;
  1264. while (nodep != NULL && (*cur <= index || index == -1)) {
  1265. if (nodep->type == XML_ELEMENT_NODE) {
  1266. if (xmlStrEqual(nodep->name, (xmlChar *)local) || xmlStrEqual((xmlChar *)"*", (xmlChar *)local)) {
  1267. if (ns == NULL || (nodep->ns != NULL && (xmlStrEqual(nodep->ns->href, (xmlChar *)ns) || xmlStrEqual((xmlChar *)"*", (xmlChar *)ns)))) {
  1268. if (*cur == index) {
  1269. ret = nodep;
  1270. break;
  1271. }
  1272. (*cur)++;
  1273. }
  1274. }
  1275. ret = dom_get_elements_by_tag_name_ns_raw(nodep->children, ns, local, cur, index);
  1276. if (ret != NULL) {
  1277. break;
  1278. }
  1279. }
  1280. nodep = nodep->next;
  1281. }
  1282. return ret;
  1283. }
  1284. /* }}} */
  1285. /* }}} end dom_element_get_elements_by_tag_name_ns_raw */
  1286. /* {{{ void dom_normalize (xmlNodePtr nodep TSRMLS_DC) */
  1287. void dom_normalize (xmlNodePtr nodep TSRMLS_DC)
  1288. {
  1289. xmlNodePtr child, nextp, newnextp;
  1290. xmlAttrPtr attr;
  1291. xmlChar *strContent;
  1292. child = nodep->children;
  1293. while(child != NULL) {
  1294. switch (child->type) {
  1295. case XML_TEXT_NODE:
  1296. nextp = child->next;
  1297. while (nextp != NULL) {
  1298. if (nextp->type == XML_TEXT_NODE) {
  1299. newnextp = nextp->next;
  1300. strContent = xmlNodeGetContent(nextp);
  1301. xmlNodeAddContent(child, strContent);
  1302. xmlFree(strContent);
  1303. xmlUnlinkNode(nextp);
  1304. php_libxml_node_free_resource(nextp TSRMLS_CC);
  1305. nextp = newnextp;
  1306. } else {
  1307. break;
  1308. }
  1309. }
  1310. break;
  1311. case XML_ELEMENT_NODE:
  1312. dom_normalize (child TSRMLS_CC);
  1313. attr = child->properties;
  1314. while (attr != NULL) {
  1315. dom_normalize((xmlNodePtr) attr TSRMLS_CC);
  1316. attr = attr->next;
  1317. }
  1318. break;
  1319. case XML_ATTRIBUTE_NODE:
  1320. dom_normalize (child TSRMLS_CC);
  1321. break;
  1322. default:
  1323. break;
  1324. }
  1325. child = child->next;
  1326. }
  1327. }
  1328. /* }}} end dom_normalize */
  1329. /* {{{ void dom_set_old_ns(xmlDoc *doc, xmlNs *ns) */
  1330. void dom_set_old_ns(xmlDoc *doc, xmlNs *ns) {
  1331. xmlNs *cur;
  1332. if (doc == NULL)
  1333. return;
  1334. if (doc->oldNs == NULL) {
  1335. doc->oldNs = (xmlNsPtr) xmlMalloc(sizeof(xmlNs));
  1336. if (doc->oldNs == NULL) {
  1337. return;
  1338. }
  1339. memset(doc->oldNs, 0, sizeof(xmlNs));
  1340. doc->oldNs->type = XML_LOCAL_NAMESPACE;
  1341. doc->oldNs->href = xmlStrdup(XML_XML_NAMESPACE);
  1342. doc->oldNs->prefix = xmlStrdup((const xmlChar *)"xml");
  1343. }
  1344. cur = doc->oldNs;
  1345. while (cur->next != NULL) {
  1346. cur = cur->next;
  1347. }
  1348. cur->next = ns;
  1349. }
  1350. /* }}} end dom_set_old_ns */
  1351. /*
  1352. http://www.w3.org/TR/2004/REC-DOM-Level-3-Core-20040407/core.html#ID-DocCrElNS
  1353. NAMESPACE_ERR: Raised if
  1354. 1. the qualifiedName is a malformed qualified name
  1355. 2. the qualifiedName has a prefix and the namespaceURI is null
  1356. */
  1357. /* {{{ int dom_check_qname(char *qname, char **localname, char **prefix, int uri_len, int name_len) */
  1358. int dom_check_qname(char *qname, char **localname, char **prefix, int uri_len, int name_len) {
  1359. if (name_len == 0) {
  1360. return NAMESPACE_ERR;
  1361. }
  1362. *localname = (char *)xmlSplitQName2((xmlChar *)qname, (xmlChar **) prefix);
  1363. if (*localname == NULL) {
  1364. *localname = (char *)xmlStrdup((xmlChar *)qname);
  1365. if (*prefix == NULL && uri_len == 0) {
  1366. return 0;
  1367. }
  1368. }
  1369. /* 1 */
  1370. if (xmlValidateQName((xmlChar *) qname, 0) != 0) {
  1371. return NAMESPACE_ERR;
  1372. }
  1373. /* 2 */
  1374. if (*prefix != NULL && uri_len == 0) {
  1375. return NAMESPACE_ERR;
  1376. }
  1377. return 0;
  1378. }
  1379. /* }}} */
  1380. /*
  1381. http://www.w3.org/TR/2004/REC-DOM-Level-3-Core-20040407/core.html#ID-DocCrElNS
  1382. NAMESPACE_ERR: Raised if
  1383. 3. the qualifiedName has a prefix that is "xml" and the namespaceURI is different from "http://www.w3.org/XML/1998/namespace" [XML Namespaces]
  1384. 4. the qualifiedName or its prefix is "xmlns" and the namespaceURI is different from "http://www.w3.org/2000/xmlns/"
  1385. 5. the namespaceURI is "http://www.w3.org/2000/xmlns/" and neither the qualifiedName nor its prefix is "xmlns".
  1386. */
  1387. /* {{{ xmlNsPtr dom_get_ns(xmlNodePtr nodep, char *uri, int *errorcode, char *prefix) */
  1388. xmlNsPtr dom_get_ns(xmlNodePtr nodep, char *uri, int *errorcode, char *prefix) {
  1389. xmlNsPtr nsptr = NULL;
  1390. *errorcode = 0;
  1391. if (! ((prefix && !strcmp (prefix, "xml") && strcmp(uri, (char *)XML_XML_NAMESPACE)) ||
  1392. (prefix && !strcmp (prefix, "xmlns") && strcmp(uri, (char *)DOM_XMLNS_NAMESPACE)) ||
  1393. (prefix && !strcmp(uri, (char *)DOM_XMLNS_NAMESPACE) && strcmp (prefix, "xmlns")))) {
  1394. nsptr = xmlNewNs(nodep, (xmlChar *)uri, (xmlChar *)prefix);
  1395. }
  1396. if (nsptr == NULL) {
  1397. *errorcode = NAMESPACE_ERR;
  1398. }
  1399. return nsptr;
  1400. }
  1401. /* }}} end dom_get_ns */
  1402. /* {{{ xmlNsPtr dom_get_nsdecl(xmlNode *node, xmlChar *localName) */
  1403. xmlNsPtr dom_get_nsdecl(xmlNode *node, xmlChar *localName) {
  1404. xmlNsPtr cur;
  1405. xmlNs *ret = NULL;
  1406. if (node == NULL)
  1407. return NULL;
  1408. if (localName == NULL || xmlStrEqual(localName, (xmlChar *)"")) {
  1409. cur = node->nsDef;
  1410. while (cur != NULL) {
  1411. if (cur->prefix == NULL && cur->href != NULL) {
  1412. ret = cur;
  1413. break;
  1414. }
  1415. cur = cur->next;
  1416. }
  1417. } else {
  1418. cur = node->nsDef;
  1419. while (cur != NULL) {
  1420. if (cur->prefix != NULL && xmlStrEqual(localName, cur->prefix)) {
  1421. ret = cur;
  1422. break;
  1423. }
  1424. cur = cur->next;
  1425. }
  1426. }
  1427. return ret;
  1428. }
  1429. /* }}} end dom_get_nsdecl */
  1430. static inline long dom_get_long(zval *offset) /* {{{ */
  1431. {
  1432. if (Z_TYPE_P(offset) == IS_LONG) {
  1433. return Z_LVAL_P(offset);
  1434. } else {
  1435. zval tmp;
  1436. MAKE_COPY_ZVAL(&offset, &tmp);
  1437. convert_to_long(&tmp);
  1438. return Z_LVAL(tmp);
  1439. }
  1440. }
  1441. /* }}} */
  1442. zval *dom_nodelist_read_dimension(zval *object, zval *offset, int type TSRMLS_DC) /* {{{ */
  1443. {
  1444. zval *rv, offset_copy = zval_used_for_init;
  1445. if (!offset) {
  1446. return NULL;
  1447. }
  1448. ZVAL_LONG(&offset_copy, dom_get_long(offset));
  1449. zend_call_method_with_1_params(&object, Z_OBJCE_P(object), NULL, "item", &rv, &offset_copy);
  1450. Z_DELREF_P(rv);
  1451. return rv;
  1452. } /* }}} end dom_nodelist_read_dimension */
  1453. int dom_nodelist_has_dimension(zval *object, zval *member, int check_empty TSRMLS_DC)
  1454. {
  1455. long offset = dom_get_long(member);
  1456. if (offset < 0) {
  1457. return 0;
  1458. } else {
  1459. zval *length = zend_read_property(Z_OBJCE_P(object), object, "length", sizeof("length") - 1, 0 TSRMLS_CC);
  1460. int ret = length && offset < Z_LVAL_P(length);
  1461. FREE_ZVAL(length);
  1462. return ret;
  1463. }
  1464. } /* }}} end dom_nodelist_has_dimension */
  1465. #endif /* HAVE_DOM */
  1466. /*
  1467. * Local variables:
  1468. * tab-width: 4
  1469. * c-basic-offset: 4
  1470. * End:
  1471. * vim600: noet sw=4 ts=4 fdm=marker
  1472. * vim<600: noet sw=4 ts=4
  1473. */