mirror of
				https://gitlab.gnome.org/GNOME/libxml2.git
				synced 2025-10-26 00:37:43 +03:00 
			
		
		
		
	valid: Improve addition and deletion of IDs
Introduce a new API function xmlAddIDSafe that returns a separate error code if a memory allocation fails. Store a pointer to the ID struct in xmlAttr so attributes can be freed without allocating memory. It's impossible to report malloc failures in deallocation code.
This commit is contained in:
		| @@ -440,6 +440,7 @@ struct _xmlAttr { | ||||
|     xmlNs           *ns;        /* pointer to the associated namespace */ | ||||
|     xmlAttributeType atype;     /* the attribute type if validating */ | ||||
|     void            *psvi;	/* for type/PSVI information */ | ||||
|     struct _xmlID   *id;        /* the ID struct */ | ||||
| }; | ||||
|  | ||||
| /** | ||||
|   | ||||
| @@ -249,6 +249,12 @@ XMLPUBFUN void | ||||
| #endif /* LIBXML_OUTPUT_ENABLED */ | ||||
|  | ||||
| /* IDs */ | ||||
| XMLPUBFUN int | ||||
| 		xmlAddIDSafe	       (xmlDocPtr doc, | ||||
| 					const xmlChar *value, | ||||
| 					xmlAttrPtr attr, | ||||
| 					int streaming, | ||||
| 					xmlIDPtr *id); | ||||
| XMLPUBFUN xmlIDPtr | ||||
| 		xmlAddID	       (xmlValidCtxtPtr ctxt, | ||||
| 					xmlDocPtr doc, | ||||
|   | ||||
							
								
								
									
										174
									
								
								valid.c
									
									
									
									
									
								
							
							
						
						
									
										174
									
								
								valid.c
									
									
									
									
									
								
							| @@ -2502,6 +2502,96 @@ xmlFreeID(xmlIDPtr id) { | ||||
| } | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * xmlAddIDSafe: | ||||
|  * @doc:  pointer to the document | ||||
|  * @value:  the value name | ||||
|  * @attr:  the attribute holding the ID | ||||
|  * @id:  pointer to new xmlIdPtr (optional) | ||||
|  * | ||||
|  * Register a new id declaration | ||||
|  * | ||||
|  * Returns 1 on success, 0 if the ID already exists, -1 if a memory | ||||
|  * allocation fails. | ||||
|  */ | ||||
| int | ||||
| xmlAddIDSafe(xmlDocPtr doc, const xmlChar *value, xmlAttrPtr attr, | ||||
|              int streaming, xmlIDPtr *id) { | ||||
|     xmlIDPtr ret; | ||||
|     xmlIDTablePtr table; | ||||
|     int res; | ||||
|  | ||||
|     if (id != NULL) | ||||
|         *id = NULL; | ||||
|  | ||||
|     if (doc == NULL) { | ||||
| 	return(-1); | ||||
|     } | ||||
|     if ((value == NULL) || (value[0] == 0)) { | ||||
| 	return(0); | ||||
|     } | ||||
|     if (attr == NULL) { | ||||
| 	return(-1); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * Create the ID table if needed. | ||||
|      */ | ||||
|     table = (xmlIDTablePtr) doc->ids; | ||||
|     if (table == NULL)  { | ||||
|         doc->ids = table = xmlHashCreateDict(0, doc->dict); | ||||
|     } | ||||
|     if (table == NULL) | ||||
|         return(-1); | ||||
|  | ||||
|     ret = (xmlIDPtr) xmlMalloc(sizeof(xmlID)); | ||||
|     if (ret == NULL) | ||||
| 	return(-1); | ||||
|     memset(ret, 0, sizeof(*ret)); | ||||
|  | ||||
|     /* | ||||
|      * fill the structure. | ||||
|      */ | ||||
|     ret->doc = doc; | ||||
|     ret->value = xmlStrdup(value); | ||||
|     if (ret->value == NULL) { | ||||
|         xmlFreeID(ret); | ||||
|         return(-1); | ||||
|     } | ||||
|     if (streaming) { | ||||
| 	/* | ||||
| 	 * Operating in streaming mode, attr is gonna disappear | ||||
| 	 */ | ||||
| 	if (doc->dict != NULL) | ||||
| 	    ret->name = xmlDictLookup(doc->dict, attr->name, -1); | ||||
| 	else | ||||
| 	    ret->name = xmlStrdup(attr->name); | ||||
|         if (ret->name == NULL) { | ||||
|             xmlFreeID(ret); | ||||
|             return(-1); | ||||
|         } | ||||
| 	ret->attr = NULL; | ||||
|     } else { | ||||
| 	ret->attr = attr; | ||||
| 	ret->name = NULL; | ||||
|     } | ||||
|     ret->lineno = xmlGetLineNo(attr->parent); | ||||
|  | ||||
|     res = xmlHashAdd(table, value, ret); | ||||
|     if (res <= 0) { | ||||
| 	xmlFreeID(ret); | ||||
| 	return(res); | ||||
|     } | ||||
|     if (attr != NULL) { | ||||
| 	attr->atype = XML_ATTRIBUTE_ID; | ||||
|         attr->id = ret; | ||||
|     } | ||||
|  | ||||
|     if (id != NULL) | ||||
|         *id = ret; | ||||
|     return(1); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * xmlAddID: | ||||
|  * @ctxt:  the validation context | ||||
| @@ -2516,74 +2606,26 @@ xmlFreeID(xmlIDPtr id) { | ||||
| xmlIDPtr | ||||
| xmlAddID(xmlValidCtxtPtr ctxt, xmlDocPtr doc, const xmlChar *value, | ||||
|          xmlAttrPtr attr) { | ||||
|     xmlIDPtr ret; | ||||
|     xmlIDTablePtr table; | ||||
|     xmlIDPtr id; | ||||
|     int res; | ||||
|  | ||||
|     if (doc == NULL) { | ||||
| 	return(NULL); | ||||
|     } | ||||
|     if ((value == NULL) || (value[0] == 0)) { | ||||
| 	return(NULL); | ||||
|     } | ||||
|     if (attr == NULL) { | ||||
| 	return(NULL); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * Create the ID table if needed. | ||||
|      */ | ||||
|     table = (xmlIDTablePtr) doc->ids; | ||||
|     if (table == NULL)  { | ||||
|         doc->ids = table = xmlHashCreateDict(0, doc->dict); | ||||
|     } | ||||
|     if (table == NULL) { | ||||
| 	xmlVErrMemory(ctxt, | ||||
| 		"xmlAddID: Table creation failed!\n"); | ||||
|         return(NULL); | ||||
|     } | ||||
|  | ||||
|     ret = (xmlIDPtr) xmlMalloc(sizeof(xmlID)); | ||||
|     if (ret == NULL) { | ||||
|     res = xmlAddIDSafe(doc, value, attr, xmlIsStreaming(ctxt), &id); | ||||
|     if (res < 0) { | ||||
|         xmlVErrMemory(ctxt, "malloc failed"); | ||||
| 	return(NULL); | ||||
|     } | ||||
|  | ||||
|     /* | ||||
|      * fill the structure. | ||||
|      */ | ||||
|     ret->value = xmlStrdup(value); | ||||
|     ret->doc = doc; | ||||
|     if (xmlIsStreaming(ctxt)) { | ||||
| 	/* | ||||
| 	 * Operating in streaming mode, attr is gonna disappear | ||||
| 	 */ | ||||
| 	if (doc->dict != NULL) | ||||
| 	    ret->name = xmlDictLookup(doc->dict, attr->name, -1); | ||||
| 	else | ||||
| 	    ret->name = xmlStrdup(attr->name); | ||||
| 	ret->attr = NULL; | ||||
|     } else { | ||||
| 	ret->attr = attr; | ||||
| 	ret->name = NULL; | ||||
|     } | ||||
|     ret->lineno = xmlGetLineNo(attr->parent); | ||||
|  | ||||
|     if (xmlHashAddEntry(table, value, ret) < 0) { | ||||
| #ifdef LIBXML_VALID_ENABLED | ||||
|     else if (res == 0) { | ||||
|         if (ctxt != NULL) { | ||||
|             /* | ||||
|              * The id is already defined in this DTD. | ||||
|              */ | ||||
| 	if (ctxt != NULL) { | ||||
|             xmlErrValidNode(ctxt, attr->parent, XML_DTD_ID_REDEFINED, | ||||
|                             "ID %s already defined\n", value, NULL, NULL); | ||||
|         } | ||||
| #endif /* LIBXML_VALID_ENABLED */ | ||||
| 	xmlFreeID(ret); | ||||
| 	return(NULL); | ||||
|     } | ||||
|     if (attr != NULL) | ||||
| 	attr->atype = XML_ATTRIBUTE_ID; | ||||
|     return(ret); | ||||
| #endif /* LIBXML_VALID_ENABLED */ | ||||
|  | ||||
|     return(id); | ||||
| } | ||||
|  | ||||
| static void | ||||
| @@ -2679,30 +2721,20 @@ xmlIsID(xmlDocPtr doc, xmlNodePtr elem, xmlAttrPtr attr) { | ||||
| int | ||||
| xmlRemoveID(xmlDocPtr doc, xmlAttrPtr attr) { | ||||
|     xmlIDTablePtr table; | ||||
|     xmlIDPtr id; | ||||
|     xmlChar *ID; | ||||
|  | ||||
|     if (doc == NULL) return(-1); | ||||
|     if (attr == NULL) return(-1); | ||||
|     if ((attr == NULL) || (attr->id == NULL)) return(-1); | ||||
|  | ||||
|     table = (xmlIDTablePtr) doc->ids; | ||||
|     if (table == NULL) | ||||
|         return(-1); | ||||
|  | ||||
|     ID = xmlNodeListGetString(doc, attr->children, 1); | ||||
|     if (ID == NULL) | ||||
|     if (xmlHashRemoveEntry(table, attr->id->value, xmlFreeIDTableEntry) < 0) | ||||
|         return(-1); | ||||
|     xmlValidNormalizeString(ID); | ||||
|  | ||||
|     id = xmlHashLookup(table, ID); | ||||
|     if (id == NULL || id->attr != attr) { | ||||
|         xmlFree(ID); | ||||
|         return(-1); | ||||
|     } | ||||
|  | ||||
|     xmlHashRemoveEntry(table, ID, xmlFreeIDTableEntry); | ||||
|     xmlFree(ID); | ||||
|     attr->atype = 0; | ||||
|     attr->id = NULL; | ||||
|  | ||||
|     return(0); | ||||
| } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user