@@ -1105,8 +1105,9 @@ static void php_dom_transfer_document_ref(xmlNodePtr node, php_libxml_ref_obj *n
1105
1105
}
1106
1106
}
1107
1107
1108
- /* Workaround for bug that was fixed in https://github.com/GNOME/libxml2/commit/4bc3ebf3eaba352fbbce2ef70ad00a3c7752478a */
1109
- #if LIBXML_VERSION < 21000
1108
+ /* Workaround for bug that was fixed in https://github.com/GNOME/libxml2/commit/4bc3ebf3eaba352fbbce2ef70ad00a3c7752478a
1109
+ * and https://github.com/GNOME/libxml2/commit/bc7ab5a2e61e4b36accf6803c5b0e245c11154b1 */
1110
+ #if LIBXML_VERSION < 21300
1110
1111
static xmlChar * libxml_copy_dicted_string (xmlDictPtr src_dict , xmlDictPtr dst_dict , xmlChar * str )
1111
1112
{
1112
1113
if (str == NULL ) {
@@ -1123,30 +1124,43 @@ static xmlChar *libxml_copy_dicted_string(xmlDictPtr src_dict, xmlDictPtr dst_di
1123
1124
1124
1125
static void libxml_fixup_name_and_content (xmlDocPtr src_doc , xmlDocPtr dst_doc , xmlNodePtr node )
1125
1126
{
1126
- if (src_doc != NULL && dst_doc != src_doc && src_doc -> dict != NULL ) {
1127
+ if (node -> type == XML_ENTITY_REF_NODE ) {
1128
+ node -> children = NULL ; /* Break link with original document. */
1129
+ }
1130
+ if (src_doc != NULL && src_doc -> dict != NULL ) {
1131
+ ZEND_ASSERT (dst_doc != src_doc );
1127
1132
node -> name = libxml_copy_dicted_string (src_doc -> dict , dst_doc -> dict , BAD_CAST node -> name );
1128
1133
node -> content = libxml_copy_dicted_string (src_doc -> dict , NULL , node -> content );
1129
1134
}
1130
1135
}
1131
1136
1132
- static void libxml_fixup_name_and_content_element (xmlDocPtr src_doc , xmlDocPtr dst_doc , xmlNodePtr node )
1137
+ static void libxml_fixup_name_and_content_outer (xmlDocPtr src_doc , xmlDocPtr dst_doc , xmlNodePtr node )
1133
1138
{
1134
1139
libxml_fixup_name_and_content (src_doc , dst_doc , node );
1135
- for (xmlAttrPtr attr = node -> properties ; attr != NULL ; attr = attr -> next ) {
1136
- libxml_fixup_name_and_content (src_doc , dst_doc , (xmlNodePtr ) attr );
1140
+
1141
+ if (node -> type == XML_ELEMENT_NODE ) {
1142
+ for (xmlAttrPtr attr = node -> properties ; attr != NULL ; attr = attr -> next ) {
1143
+ libxml_fixup_name_and_content (src_doc , dst_doc , (xmlNodePtr ) attr );
1144
+ for (xmlNodePtr attr_child = attr -> children ; attr_child != NULL ; attr_child = attr_child -> next ) {
1145
+ libxml_fixup_name_and_content (src_doc , dst_doc , attr_child );
1146
+ }
1147
+ }
1137
1148
}
1138
1149
1139
- for (xmlNodePtr child = node -> children ; child != NULL ; child = child -> next ) {
1140
- libxml_fixup_name_and_content_element (src_doc , dst_doc , child );
1150
+ if (node -> type == XML_ELEMENT_NODE || node -> type == XML_ATTRIBUTE_NODE ) {
1151
+ for (xmlNodePtr child = node -> children ; child != NULL ; child = child -> next ) {
1152
+ libxml_fixup_name_and_content_outer (src_doc , dst_doc , child );
1153
+ }
1141
1154
}
1142
1155
}
1143
1156
#endif
1144
1157
1145
1158
bool php_dom_adopt_node (xmlNodePtr nodep , dom_object * dom_object_new_document , xmlDocPtr new_document )
1146
1159
{
1147
- xmlDocPtr original_document = nodep -> doc ;
1148
- php_libxml_invalidate_node_list_cache_from_doc (original_document );
1149
- if (nodep -> doc != new_document ) {
1160
+ xmlDocPtr old_doc = nodep -> doc ;
1161
+
1162
+ php_libxml_invalidate_node_list_cache_from_doc (old_doc );
1163
+ if (old_doc != new_document ) {
1150
1164
php_libxml_invalidate_node_list_cache (dom_object_new_document -> document );
1151
1165
1152
1166
/* Note for ATTRIBUTE_NODE: specified is always true in ext/dom,
@@ -1156,16 +1170,18 @@ bool php_dom_adopt_node(xmlNodePtr nodep, dom_object *dom_object_new_document, x
1156
1170
xmlSetTreeDoc (nodep , new_document );
1157
1171
php_dom_libxml_ns_mapper * ns_mapper = php_dom_get_ns_mapper (dom_object_new_document );
1158
1172
php_dom_libxml_reconcile_modern (ns_mapper , nodep );
1159
- #if LIBXML_VERSION < 21000
1160
- libxml_fixup_name_and_content_element (original_document , new_document , nodep );
1161
- #endif
1162
1173
} else {
1163
- int ret = xmlDOMWrapAdoptNode (NULL , original_document , nodep , new_document , NULL , /* options, unused */ 0 );
1174
+ int ret = xmlDOMWrapAdoptNode (NULL , old_doc , nodep , new_document , NULL , /* options, unused */ 0 );
1164
1175
if (UNEXPECTED (ret != 0 )) {
1165
1176
return false;
1166
1177
}
1167
1178
}
1168
1179
1180
+ #if LIBXML_VERSION < 21300
1181
+ /* Must be first before transferring the ref to ensure the old document dictionary stays alive. */
1182
+ libxml_fixup_name_and_content_outer (old_doc , new_document , nodep );
1183
+ #endif
1184
+
1169
1185
php_dom_transfer_document_ref (nodep , dom_object_new_document -> document );
1170
1186
} else {
1171
1187
xmlUnlinkNode (nodep );
0 commit comments