mirror of
				https://gitlab.gnome.org/GNOME/libxml2.git
				synced 2025-10-24 13:33:01 +03:00 
			
		
		
		
	save: Report malloc failures
Fix places where malloc failures aren't report. Introduce a new API function xmlSaveFinish which returns an error code.
This commit is contained in:
		| @@ -73,6 +73,8 @@ XMLPUBFUN int | |||||||
| 		xmlSaveFlush		(xmlSaveCtxtPtr ctxt); | 		xmlSaveFlush		(xmlSaveCtxtPtr ctxt); | ||||||
| XMLPUBFUN int | XMLPUBFUN int | ||||||
| 		xmlSaveClose		(xmlSaveCtxtPtr ctxt); | 		xmlSaveClose		(xmlSaveCtxtPtr ctxt); | ||||||
|  | XMLPUBFUN int | ||||||
|  | 		xmlSaveFinish		(xmlSaveCtxtPtr ctxt); | ||||||
| XMLPUBFUN int | XMLPUBFUN int | ||||||
| 		xmlSaveSetEscape	(xmlSaveCtxtPtr ctxt, | 		xmlSaveSetEscape	(xmlSaveCtxtPtr ctxt, | ||||||
| 					 xmlCharEncodingOutputFunc escape); | 					 xmlCharEncodingOutputFunc escape); | ||||||
|   | |||||||
							
								
								
									
										139
									
								
								xmlsave.c
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								xmlsave.c
									
									
									
									
									
								
							| @@ -422,19 +422,17 @@ xmlAttrSerializeContent(xmlOutputBufferPtr buf, xmlAttrPtr attr) | |||||||
|  * This will dump the content of the notation table as an XML DTD definition |  * This will dump the content of the notation table as an XML DTD definition | ||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| xmlBufDumpNotationTable(xmlBufPtr buf, xmlNotationTablePtr table) { | xmlBufDumpNotationTable(xmlOutputBufferPtr stream, xmlNotationTablePtr table) { | ||||||
|     xmlBufferPtr buffer; |     xmlBufferPtr buffer; | ||||||
|  |  | ||||||
|     buffer = xmlBufferCreate(); |     buffer = xmlBufferCreate(); | ||||||
|     if (buffer == NULL) { |     if (buffer == NULL) { | ||||||
|         /* |         stream->error = XML_ERR_NO_MEMORY; | ||||||
|          * TODO set the error in buf |  | ||||||
|          */ |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); |     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); | ||||||
|     xmlDumpNotationTable(buffer, table); |     xmlDumpNotationTable(buffer, table); | ||||||
|     xmlBufMergeBuffer(buf, buffer); |     xmlBufMergeBuffer(stream->buffer, buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -446,19 +444,17 @@ xmlBufDumpNotationTable(xmlBufPtr buf, xmlNotationTablePtr table) { | |||||||
|  * DTD definition |  * DTD definition | ||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| xmlBufDumpElementDecl(xmlBufPtr buf, xmlElementPtr elem) { | xmlBufDumpElementDecl(xmlOutputBufferPtr stream, xmlElementPtr elem) { | ||||||
|     xmlBufferPtr buffer; |     xmlBufferPtr buffer; | ||||||
|  |  | ||||||
|     buffer = xmlBufferCreate(); |     buffer = xmlBufferCreate(); | ||||||
|     if (buffer == NULL) { |     if (buffer == NULL) { | ||||||
|         /* |         stream->error = XML_ERR_NO_MEMORY; | ||||||
|          * TODO set the error in buf |  | ||||||
|          */ |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); |     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); | ||||||
|     xmlDumpElementDecl(buffer, elem); |     xmlDumpElementDecl(buffer, elem); | ||||||
|     xmlBufMergeBuffer(buf, buffer); |     xmlBufMergeBuffer(stream->buffer, buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -470,19 +466,17 @@ xmlBufDumpElementDecl(xmlBufPtr buf, xmlElementPtr elem) { | |||||||
|  * DTD definition |  * DTD definition | ||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| xmlBufDumpAttributeDecl(xmlBufPtr buf, xmlAttributePtr attr) { | xmlBufDumpAttributeDecl(xmlOutputBufferPtr stream, xmlAttributePtr attr) { | ||||||
|     xmlBufferPtr buffer; |     xmlBufferPtr buffer; | ||||||
|  |  | ||||||
|     buffer = xmlBufferCreate(); |     buffer = xmlBufferCreate(); | ||||||
|     if (buffer == NULL) { |     if (buffer == NULL) { | ||||||
|         /* |         stream->error = XML_ERR_NO_MEMORY; | ||||||
|          * TODO set the error in buf |  | ||||||
|          */ |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); |     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); | ||||||
|     xmlDumpAttributeDecl(buffer, attr); |     xmlDumpAttributeDecl(buffer, attr); | ||||||
|     xmlBufMergeBuffer(buf, buffer); |     xmlBufMergeBuffer(stream->buffer, buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -493,19 +487,17 @@ xmlBufDumpAttributeDecl(xmlBufPtr buf, xmlAttributePtr attr) { | |||||||
|  * This will dump the content of the entity table as an XML DTD definition |  * This will dump the content of the entity table as an XML DTD definition | ||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| xmlBufDumpEntityDecl(xmlBufPtr buf, xmlEntityPtr ent) { | xmlBufDumpEntityDecl(xmlOutputBufferPtr stream, xmlEntityPtr ent) { | ||||||
|     xmlBufferPtr buffer; |     xmlBufferPtr buffer; | ||||||
|  |  | ||||||
|     buffer = xmlBufferCreate(); |     buffer = xmlBufferCreate(); | ||||||
|     if (buffer == NULL) { |     if (buffer == NULL) { | ||||||
|         /* |         stream->error = XML_ERR_NO_MEMORY; | ||||||
|          * TODO set the error in buf |  | ||||||
|          */ |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); |     xmlBufferSetAllocationScheme(buffer, XML_BUFFER_ALLOC_DOUBLEIT); | ||||||
|     xmlDumpEntityDecl(buffer, ent); |     xmlDumpEntityDecl(buffer, ent); | ||||||
|     xmlBufMergeBuffer(buf, buffer); |     xmlBufMergeBuffer(stream->buffer, buffer); | ||||||
| } | } | ||||||
|  |  | ||||||
| /************************************************************************ | /************************************************************************ | ||||||
| @@ -518,18 +510,21 @@ static int xmlSaveSwitchEncoding(xmlSaveCtxtPtr ctxt, const char *encoding) { | |||||||
|     xmlOutputBufferPtr buf = ctxt->buf; |     xmlOutputBufferPtr buf = ctxt->buf; | ||||||
|  |  | ||||||
|     if ((encoding != NULL) && (buf->encoder == NULL) && (buf->conv == NULL)) { |     if ((encoding != NULL) && (buf->encoder == NULL) && (buf->conv == NULL)) { | ||||||
| 	buf->encoder = xmlFindCharEncodingHandler((const char *)encoding); |         xmlCharEncodingHandler *handler; | ||||||
| 	if (buf->encoder == NULL) { |         int res; | ||||||
| 	    xmlSaveErr(XML_SAVE_UNKNOWN_ENCODING, NULL, |  | ||||||
| 		       (const char *)encoding); | 	res = xmlOpenCharEncodingHandler((const char *) encoding, &handler); | ||||||
|  |         if (res != 0) { | ||||||
|  |             buf->error = res; | ||||||
|             return(-1); |             return(-1); | ||||||
|         } |         } | ||||||
| 	buf->conv = xmlBufCreate(); | 	buf->conv = xmlBufCreate(); | ||||||
| 	if (buf->conv == NULL) { | 	if (buf->conv == NULL) { | ||||||
| 	    xmlCharEncCloseFunc(buf->encoder); | 	    xmlCharEncCloseFunc(handler); | ||||||
| 	    xmlSaveErrMemory("creating encoding buffer"); |             buf->error = XML_ERR_NO_MEMORY; | ||||||
| 	    return(-1); | 	    return(-1); | ||||||
| 	} | 	} | ||||||
|  |         buf->encoder = handler; | ||||||
| 	/* | 	/* | ||||||
| 	 * initialize the state, e.g. if outputting a BOM | 	 * initialize the state, e.g. if outputting a BOM | ||||||
| 	 */ | 	 */ | ||||||
| @@ -696,8 +691,7 @@ xmlDtdDumpOutput(xmlSaveCtxtPtr ctxt, xmlDtdPtr dtd) { | |||||||
|      */ |      */ | ||||||
|     if ((dtd->notations != NULL) && ((dtd->doc == NULL) || |     if ((dtd->notations != NULL) && ((dtd->doc == NULL) || | ||||||
|         (dtd->doc->intSubset == dtd))) { |         (dtd->doc->intSubset == dtd))) { | ||||||
|         xmlBufDumpNotationTable(buf->buffer, |         xmlBufDumpNotationTable(buf, (xmlNotationTablePtr) dtd->notations); | ||||||
|                                 (xmlNotationTablePtr) dtd->notations); |  | ||||||
|     } |     } | ||||||
|     format = ctxt->format; |     format = ctxt->format; | ||||||
|     level = ctxt->level; |     level = ctxt->level; | ||||||
| @@ -735,7 +729,16 @@ xmlAttrDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) { | |||||||
|     } |     } | ||||||
|     xmlOutputBufferWriteString(buf, (const char *)cur->name); |     xmlOutputBufferWriteString(buf, (const char *)cur->name); | ||||||
|     xmlOutputBufferWrite(buf, 2, "=\""); |     xmlOutputBufferWrite(buf, 2, "=\""); | ||||||
|  |     if ((ctxt->options & XML_SAVE_XHTML) && | ||||||
|  |         (cur->ns == NULL) && | ||||||
|  |         ((cur->children == NULL) || | ||||||
|  |          (cur->children->content == NULL) || | ||||||
|  |          (cur->children->content[0] == 0)) && | ||||||
|  |         (htmlIsBooleanAttr(cur->name))) { | ||||||
|  |         xmlOutputBufferWriteString(buf, (const char *) cur->name); | ||||||
|  |     } else { | ||||||
|         xmlAttrSerializeContent(buf, cur); |         xmlAttrSerializeContent(buf, cur); | ||||||
|  |     } | ||||||
|     xmlOutputBufferWrite(buf, 1, "\""); |     xmlOutputBufferWrite(buf, 1, "\""); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -839,15 +842,15 @@ xmlNodeDumpOutputInternal(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) { | |||||||
| 	    break; | 	    break; | ||||||
|  |  | ||||||
|         case XML_ELEMENT_DECL: |         case XML_ELEMENT_DECL: | ||||||
|             xmlBufDumpElementDecl(buf->buffer, (xmlElementPtr) cur); |             xmlBufDumpElementDecl(buf, (xmlElementPtr) cur); | ||||||
|             break; |             break; | ||||||
|  |  | ||||||
|         case XML_ATTRIBUTE_DECL: |         case XML_ATTRIBUTE_DECL: | ||||||
|             xmlBufDumpAttributeDecl(buf->buffer, (xmlAttributePtr) cur); |             xmlBufDumpAttributeDecl(buf, (xmlAttributePtr) cur); | ||||||
|             break; |             break; | ||||||
|  |  | ||||||
|         case XML_ENTITY_DECL: |         case XML_ENTITY_DECL: | ||||||
|             xmlBufDumpEntityDecl(buf->buffer, (xmlEntityPtr) cur); |             xmlBufDumpEntityDecl(buf, (xmlEntityPtr) cur); | ||||||
|             break; |             break; | ||||||
|  |  | ||||||
|         case XML_ELEMENT_NODE: |         case XML_ELEMENT_NODE: | ||||||
| @@ -1329,17 +1332,6 @@ xhtmlAttrListDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) { | |||||||
| 	if ((cur->ns != NULL) && (xmlStrEqual(cur->name, BAD_CAST "lang")) && | 	if ((cur->ns != NULL) && (xmlStrEqual(cur->name, BAD_CAST "lang")) && | ||||||
| 	    (xmlStrEqual(cur->ns->prefix, BAD_CAST "xml"))) | 	    (xmlStrEqual(cur->ns->prefix, BAD_CAST "xml"))) | ||||||
| 	    xml_lang = cur; | 	    xml_lang = cur; | ||||||
| 	else if ((cur->ns == NULL) && |  | ||||||
| 		 ((cur->children == NULL) || |  | ||||||
| 		  (cur->children->content == NULL) || |  | ||||||
| 		  (cur->children->content[0] == 0)) && |  | ||||||
| 		 (htmlIsBooleanAttr(cur->name))) { |  | ||||||
| 	    if (cur->children != NULL) |  | ||||||
| 		xmlFreeNode(cur->children); |  | ||||||
| 	    cur->children = xmlNewDocText(cur->doc, cur->name); |  | ||||||
| 	    if (cur->children != NULL) |  | ||||||
| 		cur->children->parent = (xmlNodePtr) cur; |  | ||||||
| 	} |  | ||||||
|         xmlAttrDumpOutput(ctxt, cur); |         xmlAttrDumpOutput(ctxt, cur); | ||||||
| 	cur = cur->next; | 	cur = cur->next; | ||||||
|     } |     } | ||||||
| @@ -1390,13 +1382,16 @@ xhtmlAttrListDumpOutput(xmlSaveCtxtPtr ctxt, xmlAttrPtr cur) { | |||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) { | xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) { | ||||||
|     int format = ctxt->format, addmeta; |     int format = ctxt->format, addmeta, oldoptions; | ||||||
|     xmlNodePtr tmp, root, unformattedNode = NULL; |     xmlNodePtr tmp, root, unformattedNode = NULL; | ||||||
|     xmlChar *start, *end; |     xmlChar *start, *end; | ||||||
|     xmlOutputBufferPtr buf = ctxt->buf; |     xmlOutputBufferPtr buf = ctxt->buf; | ||||||
|  |  | ||||||
|     if (cur == NULL) return; |     if (cur == NULL) return; | ||||||
|  |  | ||||||
|  |     oldoptions = ctxt->options; | ||||||
|  |     ctxt->options |= XML_SAVE_XHTML; | ||||||
|  |  | ||||||
|     root = cur; |     root = cur; | ||||||
|     while (1) { |     while (1) { | ||||||
|         switch (cur->type) { |         switch (cur->type) { | ||||||
| @@ -1421,15 +1416,15 @@ xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) { | |||||||
|             break; |             break; | ||||||
|  |  | ||||||
|         case XML_ELEMENT_DECL: |         case XML_ELEMENT_DECL: | ||||||
|             xmlBufDumpElementDecl(buf->buffer, (xmlElementPtr) cur); |             xmlBufDumpElementDecl(buf, (xmlElementPtr) cur); | ||||||
| 	    break; | 	    break; | ||||||
|  |  | ||||||
|         case XML_ATTRIBUTE_DECL: |         case XML_ATTRIBUTE_DECL: | ||||||
|             xmlBufDumpAttributeDecl(buf->buffer, (xmlAttributePtr) cur); |             xmlBufDumpAttributeDecl(buf, (xmlAttributePtr) cur); | ||||||
| 	    break; | 	    break; | ||||||
|  |  | ||||||
|         case XML_ENTITY_DECL: |         case XML_ENTITY_DECL: | ||||||
|             xmlBufDumpEntityDecl(buf->buffer, (xmlEntityPtr) cur); |             xmlBufDumpEntityDecl(buf, (xmlEntityPtr) cur); | ||||||
| 	    break; | 	    break; | ||||||
|  |  | ||||||
|         case XML_ELEMENT_NODE: |         case XML_ELEMENT_NODE: | ||||||
| @@ -1697,6 +1692,8 @@ xhtmlNodeDumpOutput(xmlSaveCtxtPtr ctxt, xmlNodePtr cur) { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     ctxt->options = oldoptions; | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| @@ -1915,6 +1912,28 @@ xmlSaveClose(xmlSaveCtxtPtr ctxt) | |||||||
|     return(ret); |     return(ret); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * xmlSaveFinish: | ||||||
|  |  * @ctxt:  a document saving context | ||||||
|  |  * | ||||||
|  |  * Close a document saving context, i.e. make sure that all bytes have | ||||||
|  |  * been output and free the associated data. | ||||||
|  |  * | ||||||
|  |  * Returns an xmlParserErrors code. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | xmlSaveFinish(xmlSaveCtxtPtr ctxt) | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |  | ||||||
|  |     if (ctxt == NULL) | ||||||
|  |         return(XML_ERR_INTERNAL_ERROR); | ||||||
|  |     xmlSaveFlush(ctxt); | ||||||
|  |     ret = ctxt->buf->error; | ||||||
|  |     xmlFreeSaveCtxt(ctxt); | ||||||
|  |     return(ret); | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * xmlSaveSetEscape: |  * xmlSaveSetEscape: | ||||||
|  * @ctxt:  a document saving context |  * @ctxt:  a document saving context | ||||||
| @@ -2311,6 +2330,8 @@ xmlDocDumpFormatMemoryEnc(xmlDocPtr out_doc, xmlChar **doc_txt_ptr, | |||||||
|     int                         dummy = 0; |     int                         dummy = 0; | ||||||
|     xmlOutputBufferPtr          out_buff = NULL; |     xmlOutputBufferPtr          out_buff = NULL; | ||||||
|     xmlCharEncodingHandlerPtr   conv_hdlr = NULL; |     xmlCharEncodingHandlerPtr   conv_hdlr = NULL; | ||||||
|  |     xmlChar *content; | ||||||
|  |     int len; | ||||||
|  |  | ||||||
|     if (doc_txt_len == NULL) { |     if (doc_txt_len == NULL) { | ||||||
|         doc_txt_len = &dummy;   /*  Continue, caller just won't get length */ |         doc_txt_len = &dummy;   /*  Continue, caller just won't get length */ | ||||||
| @@ -2361,19 +2382,27 @@ xmlDocDumpFormatMemoryEnc(xmlDocPtr out_doc, xmlChar **doc_txt_ptr, | |||||||
|     xmlDocContentDumpOutput(&ctxt, out_doc); |     xmlDocContentDumpOutput(&ctxt, out_doc); | ||||||
|     xmlOutputBufferFlush(out_buff); |     xmlOutputBufferFlush(out_buff); | ||||||
|     if (out_buff->conv != NULL) { |     if (out_buff->conv != NULL) { | ||||||
| 	*doc_txt_len = xmlBufUse(out_buff->conv); |         if (xmlBufContent(out_buff->buffer) == NULL) | ||||||
| 	*doc_txt_ptr = xmlStrndup(xmlBufContent(out_buff->conv), *doc_txt_len); |             goto error; | ||||||
|  |         content = xmlBufContent(out_buff->conv); | ||||||
|  |         len = xmlBufUse(out_buff->conv); | ||||||
|     } else { |     } else { | ||||||
| 	*doc_txt_len = xmlBufUse(out_buff->buffer); |         content = xmlBufContent(out_buff->buffer); | ||||||
| 	*doc_txt_ptr = xmlStrndup(xmlBufContent(out_buff->buffer),*doc_txt_len); |         len = xmlBufUse(out_buff->buffer); | ||||||
|     } |     } | ||||||
|     (void)xmlOutputBufferClose(out_buff); |     if (content == NULL) | ||||||
|  |         goto error; | ||||||
|  |     *doc_txt_ptr = xmlStrndup(content, len); | ||||||
|  |     if (*doc_txt_ptr == NULL) | ||||||
|  |         goto error; | ||||||
|  |     *doc_txt_len = len; | ||||||
|  |     xmlOutputBufferClose(out_buff); | ||||||
|  |  | ||||||
|     if ((*doc_txt_ptr == NULL) && (*doc_txt_len > 0)) { |     return; | ||||||
|         *doc_txt_len = 0; |  | ||||||
|  | error: | ||||||
|     xmlSaveErrMemory("creating output"); |     xmlSaveErrMemory("creating output"); | ||||||
|     } |     xmlOutputBufferClose(out_buff); | ||||||
|  |  | ||||||
|     return; |     return; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user