mirror of
				https://gitlab.gnome.org/GNOME/libxml2.git
				synced 2025-10-24 13:33:01 +03:00 
			
		
		
		
	parser: Acknowledge that entities with namespaces are broken
Entities which reference out-of-scope namespace have always been broken.
xmlParseBalancedChunkMemoryInternal tried to reuse the namespaces
currently in scope but these namespaces were ignored by the SAX handler.
Besides, there could be different namespaces in scope when expanding the
entity again. For example:
    <!DOCTYPE doc [
      <!ENTITY ent "<ns:elem/>">
    ]>
    <doc>
      <decl1 xmlns:ns="urn:ns1">
        &ent;
      </decl1>
      <decl2 xmlns:ns="urn:ns2">
        &ent;
      </decl2>
    </doc>
Add some comments outlining possible solutions to this problem.
For now, we stop copying namespaces to the temporary parser context
in xmlParseBalancedChunkMemoryInternal. This has never really worked
and the recent changes contained a partial fix which uncovered other
problems like a use-after-free with the XML Reader interface, found
by OSS-Fuzz.
			
			
This commit is contained in:
		
							
								
								
									
										82
									
								
								parser.c
									
									
									
									
									
								
							
							
						
						
									
										82
									
								
								parser.c
									
									
									
									
									
								
							| @@ -7267,6 +7267,39 @@ xmlParseReference(xmlParserCtxtPtr ctxt) { | |||||||
|      * of validating, or substituting entities were given. Doing so is |      * of validating, or substituting entities were given. Doing so is | ||||||
|      * far more secure as the parser will only process data coming from |      * far more secure as the parser will only process data coming from | ||||||
|      * the document entity by default. |      * the document entity by default. | ||||||
|  |      * | ||||||
|  |      * FIXME: This doesn't work correctly since entities can be | ||||||
|  |      * expanded with different namespace declarations in scope. | ||||||
|  |      * For example: | ||||||
|  |      * | ||||||
|  |      * <!DOCTYPE doc [ | ||||||
|  |      *   <!ENTITY ent "<ns:elem/>"> | ||||||
|  |      * ]> | ||||||
|  |      * <doc> | ||||||
|  |      *   <decl1 xmlns:ns="urn:ns1"> | ||||||
|  |      *     &ent; | ||||||
|  |      *   </decl1> | ||||||
|  |      *   <decl2 xmlns:ns="urn:ns2"> | ||||||
|  |      *     &ent; | ||||||
|  |      *   </decl2> | ||||||
|  |      * </doc> | ||||||
|  |      * | ||||||
|  |      * Proposed fix: | ||||||
|  |      * | ||||||
|  |      * - Remove the ent->owner optimization which tries to avoid the | ||||||
|  |      *   initial copy of the entity. Always make entities own the | ||||||
|  |      *   subtree. | ||||||
|  |      * - Ignore current namespace declarations when parsing the | ||||||
|  |      *   entity. If a prefix can't be resolved, don't report an error | ||||||
|  |      *   but mark it as unresolved. | ||||||
|  |      * - Try to resolve these prefixes when expanding the entity. | ||||||
|  |      *   This will require a specialized version of xmlStaticCopyNode | ||||||
|  |      *   which can also make use of the namespace hash table to avoid | ||||||
|  |      *   quadratic behavior. | ||||||
|  |      * | ||||||
|  |      * Alternatively, we could simply reparse the entity on each | ||||||
|  |      * expansion like we already do with custom SAX callbacks. | ||||||
|  |      * External entity content should be cached in this case. | ||||||
|      */ |      */ | ||||||
|     if (((ent->flags & XML_ENT_PARSED) == 0) && |     if (((ent->flags & XML_ENT_PARSED) == 0) && | ||||||
|         ((ent->etype != XML_EXTERNAL_GENERAL_PARSED_ENTITY) || |         ((ent->etype != XML_EXTERNAL_GENERAL_PARSED_ENTITY) || | ||||||
| @@ -12876,7 +12909,9 @@ xmlParseBalancedChunkMemoryInternal(xmlParserCtxtPtr oldctxt, | |||||||
|     xmlNodePtr content = NULL; |     xmlNodePtr content = NULL; | ||||||
|     xmlNodePtr last = NULL; |     xmlNodePtr last = NULL; | ||||||
|     xmlParserErrors ret = XML_ERR_OK; |     xmlParserErrors ret = XML_ERR_OK; | ||||||
|  | #if 0 | ||||||
|     unsigned i; |     unsigned i; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|     if (((oldctxt->depth > 40) && ((oldctxt->options & XML_PARSE_HUGE) == 0)) || |     if (((oldctxt->depth > 40) && ((oldctxt->options & XML_PARSE_HUGE) == 0)) || | ||||||
|         (oldctxt->depth >  100)) { |         (oldctxt->depth >  100)) { | ||||||
| @@ -12906,29 +12941,38 @@ xmlParseBalancedChunkMemoryInternal(xmlParserCtxtPtr oldctxt, | |||||||
|     ctxt->str_xmlns = xmlDictLookup(ctxt->dict, BAD_CAST "xmlns", 5); |     ctxt->str_xmlns = xmlDictLookup(ctxt->dict, BAD_CAST "xmlns", 5); | ||||||
|     ctxt->str_xml_ns = xmlDictLookup(ctxt->dict, XML_XML_NAMESPACE, 36); |     ctxt->str_xml_ns = xmlDictLookup(ctxt->dict, XML_XML_NAMESPACE, 36); | ||||||
|  |  | ||||||
|     /* propagate namespaces down the entity */ |     /* | ||||||
|     if (oldctxt->nsdb != NULL) { |      * Propagate namespaces down the entity | ||||||
|         for (i = 0; i < oldctxt->nsdb->hashSize; i++) { |      * | ||||||
|             xmlParserNsBucket *bucket = &oldctxt->nsdb->hash[i]; |      * This is disabled for now. The pre-2.12 code was already broken | ||||||
|             xmlHashedString hprefix, huri; |      * since the SAX handler was using xmlSearchNs which didn't see the | ||||||
|             const xmlChar **ns; |      * namespaces added here. | ||||||
|             xmlParserNsExtra *extra; |      * | ||||||
|             unsigned nsIndex; |      * Making entities and namespaces work correctly requires additional | ||||||
|  |      * changes, see xmlParseReference. | ||||||
|  |      */ | ||||||
|  | #if 0 | ||||||
|  |     for (i = 0; i < oldctxt->nsdb->hashSize; i++) { | ||||||
|  |         xmlParserNsBucket *bucket = &oldctxt->nsdb->hash[i]; | ||||||
|  |         xmlHashedString hprefix, huri; | ||||||
|  |         const xmlChar **ns; | ||||||
|  |         xmlParserNsExtra *extra; | ||||||
|  |         unsigned nsIndex; | ||||||
|  |  | ||||||
|             if ((bucket->hashValue != 0) && |         if ((bucket->hashValue != 0) && | ||||||
|                 (bucket->index != INT_MAX)) { |             (bucket->index != INT_MAX)) { | ||||||
|                 nsIndex = bucket->index; |             nsIndex = bucket->index; | ||||||
|                 ns = &oldctxt->nsTab[nsIndex * 2]; |             ns = &oldctxt->nsTab[nsIndex * 2]; | ||||||
|                 extra = &oldctxt->nsdb->extra[nsIndex]; |             extra = &oldctxt->nsdb->extra[nsIndex]; | ||||||
|  |  | ||||||
|                 hprefix.name = ns[0]; |             hprefix.name = ns[0]; | ||||||
|                 hprefix.hashValue = bucket->hashValue; |             hprefix.hashValue = bucket->hashValue; | ||||||
|                 huri.name = ns[1]; |             huri.name = ns[1]; | ||||||
|                 huri.hashValue = extra->uriHashValue; |             huri.hashValue = extra->uriHashValue; | ||||||
|                 xmlParserNsPush(ctxt, &hprefix, &huri, extra->saxData, 0); |             xmlParserNsPush(ctxt, &hprefix, &huri, extra->saxData, 0); | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|     oldsax = ctxt->sax; |     oldsax = ctxt->sax; | ||||||
|     ctxt->sax = oldctxt->sax; |     ctxt->sax = oldctxt->sax; | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								tree.c
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								tree.c
									
									
									
									
									
								
							| @@ -4206,7 +4206,10 @@ xmlStaticCopyNode(xmlNodePtr node, xmlDocPtr doc, xmlNodePtr parent, | |||||||
| 	    /* | 	    /* | ||||||
| 	     * Humm, we are copying an element whose namespace is defined | 	     * Humm, we are copying an element whose namespace is defined | ||||||
| 	     * out of the new tree scope. Search it in the original tree | 	     * out of the new tree scope. Search it in the original tree | ||||||
| 	     * and add it at the top of the new tree | 	     * and add it at the top of the new tree. | ||||||
|  |              * | ||||||
|  |              * TODO: Searching the original tree seems unnecessary. We | ||||||
|  |              * already have a namespace URI. | ||||||
| 	     */ | 	     */ | ||||||
| 	    ns = xmlSearchNs(node->doc, node, node->ns->prefix); | 	    ns = xmlSearchNs(node->doc, node, node->ns->prefix); | ||||||
| 	    if (ns != NULL) { | 	    if (ns != NULL) { | ||||||
| @@ -4214,8 +4217,8 @@ xmlStaticCopyNode(xmlNodePtr node, xmlDocPtr doc, xmlNodePtr parent, | |||||||
|  |  | ||||||
| 		while (root->parent != NULL) root = root->parent; | 		while (root->parent != NULL) root = root->parent; | ||||||
| 		ret->ns = xmlNewNs(root, ns->href, ns->prefix); | 		ret->ns = xmlNewNs(root, ns->href, ns->prefix); | ||||||
| 		} else { |             } else { | ||||||
| 			ret->ns = xmlNewReconciledNs(doc, ret, node->ns); |                 ret->ns = xmlNewReconciledNs(doc, ret, node->ns); | ||||||
| 	    } | 	    } | ||||||
| 	} else { | 	} else { | ||||||
| 	    /* | 	    /* | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user