mirror of
				https://gitlab.gnome.org/GNOME/libxml2.git
				synced 2025-10-24 13:33:01 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			984 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			984 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python3
 | |
| #
 | |
| # generate a tester program for the API
 | |
| #
 | |
| import sys
 | |
| import os
 | |
| import string
 | |
| try:
 | |
|     import libxml2
 | |
| except:
 | |
|     print("libxml2 python bindings not available, skipping testapi.c generation")
 | |
|     sys.exit(0)
 | |
| 
 | |
| if len(sys.argv) > 1:
 | |
|     srcPref = sys.argv[1] + '/'
 | |
| else:
 | |
|     srcPref = ''
 | |
| 
 | |
| #
 | |
| # Modules we want to skip in API test
 | |
| #
 | |
| skipped_modules = [ "SAX", "xlink", "threads", "globals",
 | |
|   "xmlmemory", "xmlversion", "xmlexports",
 | |
| ]
 | |
| 
 | |
| #
 | |
| # defines for each module
 | |
| #
 | |
| modules_defines = {
 | |
|     "HTMLparser": "LIBXML_HTML_ENABLED",
 | |
|     "catalog": "LIBXML_CATALOG_ENABLED",
 | |
|     "xmlreader": "LIBXML_READER_ENABLED",
 | |
|     "relaxng": "LIBXML_SCHEMAS_ENABLED",
 | |
|     "schemasInternals": "LIBXML_SCHEMAS_ENABLED",
 | |
|     "xmlschemas": "LIBXML_SCHEMAS_ENABLED",
 | |
|     "xmlschemastypes": "LIBXML_SCHEMAS_ENABLED",
 | |
|     "xpath": "LIBXML_XPATH_ENABLED",
 | |
|     "xpathInternals": "LIBXML_XPATH_ENABLED",
 | |
|     "xinclude": "LIBXML_XINCLUDE_ENABLED",
 | |
|     "xpointer": "LIBXML_XPTR_ENABLED",
 | |
|     "xmlregexp" : "LIBXML_REGEXP_ENABLED",
 | |
|     "xmlautomata" : "LIBXML_AUTOMATA_ENABLED",
 | |
|     "xmlsave" : "LIBXML_OUTPUT_ENABLED",
 | |
|     "xmlmodule" : "LIBXML_MODULES_ENABLED",
 | |
|     "pattern" : "LIBXML_PATTERN_ENABLED",
 | |
|     "schematron" : "LIBXML_SCHEMATRON_ENABLED",
 | |
| }
 | |
| 
 | |
| #
 | |
| # defines for specific functions
 | |
| #
 | |
| function_defines = {
 | |
|     "htmlDefaultSAXHandlerInit": "LIBXML_HTML_ENABLED",
 | |
|     "xmlSAX2EndElement" : "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAX2StartElement" : "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXDefaultVersion" : "LIBXML_SAX1_ENABLED",
 | |
|     "UTF8Toisolat1" : "LIBXML_OUTPUT_ENABLED",
 | |
|     "xmlIOParseDTD": "LIBXML_VALID_ENABLED",
 | |
|     "xmlParseDTD": "LIBXML_VALID_ENABLED",
 | |
|     "xmlParseDoc": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseMemory": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlRecoverDoc": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseFile": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlRecoverFile": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlRecoverMemory": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseFileWithData": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseMemory": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXUserParseMemory": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseDoc": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseDTD": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXUserParseFile": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseEntity": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseExternalEntity": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseMemoryWithData": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseBalancedChunkMemory": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlParseBalancedChunkMemoryRecover": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSetupParserForBuffer": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlStopParser": "LIBXML_PUSH_ENABLED",
 | |
|     "xmlAttrSerializeTxtContent": "LIBXML_OUTPUT_ENABLED",
 | |
|     "xmlSAXParseFile": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlSAXParseEntity": "LIBXML_SAX1_ENABLED",
 | |
|     "xmlNewTextChild": "LIBXML_TREE_ENABLED",
 | |
|     "xmlNewDocRawNode": "LIBXML_TREE_ENABLED",
 | |
|     "xmlNewProp": "LIBXML_TREE_ENABLED",
 | |
|     "xmlReconciliateNs": "LIBXML_TREE_ENABLED",
 | |
|     "xmlValidateNCName": "LIBXML_TREE_ENABLED",
 | |
|     "xmlValidateNMToken": "LIBXML_TREE_ENABLED",
 | |
|     "xmlValidateName": "LIBXML_TREE_ENABLED",
 | |
|     "xmlNewChild": "LIBXML_TREE_ENABLED",
 | |
|     "xmlValidateQName": "LIBXML_TREE_ENABLED",
 | |
|     "xmlSprintfElementContent": "LIBXML_OUTPUT_ENABLED",
 | |
|     "xmlValidGetPotentialChildren" : "LIBXML_VALID_ENABLED",
 | |
|     "xmlValidGetValidElements" : "LIBXML_VALID_ENABLED",
 | |
|     "xmlTextReaderPreservePattern" : "LIBXML_PATTERN_ENABLED",
 | |
| }
 | |
| 
 | |
| #
 | |
| # Some functions really need to be skipped for the tests.
 | |
| #
 | |
| skipped_functions = [
 | |
| # block on I/O
 | |
| "xmlFdRead", "xmlReadFd", "xmlCtxtReadFd",
 | |
| "htmlFdRead", "htmlReadFd", "htmlCtxtReadFd",
 | |
| "xmlReaderNewFd", "xmlReaderForFd",
 | |
| "xmlIORead", "xmlReadIO", "xmlCtxtReadIO",
 | |
| "htmlIORead", "htmlReadIO", "htmlCtxtReadIO",
 | |
| "xmlReaderNewIO", "xmlBufferDump", "xmlNanoFTPConnect",
 | |
| "xmlNanoFTPConnectTo", "xmlNanoHTTPMethod", "xmlNanoHTTPMethodRedir",
 | |
| # Complex I/O APIs
 | |
| "xmlCreateIOParserCtxt", "xmlParserInputBufferCreateIO",
 | |
| "xmlRegisterInputCallbacks", "xmlReaderForIO",
 | |
| "xmlOutputBufferCreateIO", "xmlRegisterOutputCallbacks",
 | |
| "xmlSaveToIO", "xmlIOHTTPOpenW",
 | |
| # library state cleanup, generate false leak information and other
 | |
| # troubles, heavillyb tested otherwise.
 | |
| "xmlCleanupParser", "xmlRelaxNGCleanupTypes", "xmlSetListDoc",
 | |
| "xmlSetTreeDoc", "xmlUnlinkNode",
 | |
| # hard to avoid leaks in the tests
 | |
| "xmlStrcat", "xmlStrncat", "xmlCatalogAddLocal", "xmlNewTextWriterDoc",
 | |
| "xmlXPathNewValueTree", "xmlXPathWrapString",
 | |
| # unimplemented
 | |
| "xmlTextReaderReadInnerXml", "xmlTextReaderReadOuterXml",
 | |
| "xmlTextReaderReadString",
 | |
| # destructor
 | |
| "xmlListDelete", "xmlOutputBufferClose", "xmlNanoFTPClose", "xmlNanoHTTPClose",
 | |
| # deprecated
 | |
| "xmlCatalogGetPublic", "xmlCatalogGetSystem", "xmlEncodeEntities",
 | |
| "xmlNewGlobalNs", "xmlHandleEntity", "xmlNamespaceParseNCName",
 | |
| "xmlNamespaceParseNSDef", "xmlNamespaceParseQName",
 | |
| "xmlParseNamespace", "xmlParseQuotedString", "xmlParserHandleReference",
 | |
| "xmlScanName",
 | |
| "xmlDecodeEntities", 
 | |
| # allocators
 | |
| "xmlMemFree",
 | |
| # verbosity
 | |
| "xmlCatalogSetDebug", "xmlShellPrintXPathError", "xmlShellPrintNode",
 | |
| # Internal functions, no user space should really call them
 | |
| "xmlParseAttribute", "xmlParseAttributeListDecl", "xmlParseName",
 | |
| "xmlParseNmtoken", "xmlParseEntityValue", "xmlParseAttValue",
 | |
| "xmlParseSystemLiteral", "xmlParsePubidLiteral", "xmlParseCharData",
 | |
| "xmlParseExternalID", "xmlParseComment", "xmlParsePITarget", "xmlParsePI",
 | |
| "xmlParseNotationDecl", "xmlParseEntityDecl", "xmlParseDefaultDecl",
 | |
| "xmlParseNotationType", "xmlParseEnumerationType", "xmlParseEnumeratedType",
 | |
| "xmlParseAttributeType", "xmlParseAttributeListDecl",
 | |
| "xmlParseElementMixedContentDecl", "xmlParseElementChildrenContentDecl",
 | |
| "xmlParseElementContentDecl", "xmlParseElementDecl", "xmlParseMarkupDecl",
 | |
| "xmlParseCharRef", "xmlParseEntityRef", "xmlParseReference",
 | |
| "xmlParsePEReference", "xmlParseDocTypeDecl", "xmlParseAttribute",
 | |
| "xmlParseStartTag", "xmlParseEndTag", "xmlParseCDSect", "xmlParseContent",
 | |
| "xmlParseElement", "xmlParseVersionNum", "xmlParseVersionInfo",
 | |
| "xmlParseEncName", "xmlParseEncodingDecl", "xmlParseSDDecl",
 | |
| "xmlParseXMLDecl", "xmlParseTextDecl", "xmlParseMisc",
 | |
| "xmlParseExternalSubset", "xmlParserHandlePEReference",
 | |
| "xmlSkipBlankChars",
 | |
| # Legacy
 | |
| "xmlCleanupPredefinedEntities", "xmlInitializePredefinedEntities",
 | |
| "xmlSetFeature", "xmlGetFeature", "xmlGetFeaturesList",
 | |
| # location sets
 | |
| "xmlXPtrLocationSetAdd",
 | |
| "xmlXPtrLocationSetCreate",
 | |
| "xmlXPtrLocationSetDel",
 | |
| "xmlXPtrLocationSetMerge",
 | |
| "xmlXPtrLocationSetRemove",
 | |
| "xmlXPtrWrapLocationSet",
 | |
| ]
 | |
| 
 | |
| #
 | |
| # These functions have side effects on the global state
 | |
| # and hence generate errors on memory allocation tests
 | |
| #
 | |
| skipped_memcheck = [ "xmlLoadCatalog", "xmlAddEncodingAlias",
 | |
|    "xmlSchemaInitTypes", "xmlNanoFTPProxy", "xmlNanoFTPScanProxy",
 | |
|    "xmlNanoHTTPScanProxy", "xmlResetLastError", "xmlCatalogConvert",
 | |
|    "xmlCatalogRemove", "xmlLoadCatalogs", "xmlCleanupCharEncodingHandlers",
 | |
|    "xmlInitCharEncodingHandlers", "xmlCatalogCleanup",
 | |
|    "xmlSchemaGetBuiltInType",
 | |
|    "htmlParseFile", "htmlCtxtReadFile", # loads the catalogs
 | |
|    "xmlTextReaderSchemaValidate", "xmlSchemaCleanupTypes", # initialize the schemas type system
 | |
|    "xmlCatalogResolve", "xmlIOParseDTD" # loads the catalogs
 | |
| ]
 | |
| 
 | |
| #
 | |
| # Extra code needed for some test cases
 | |
| #
 | |
| extra_pre_call = {
 | |
|    "xmlSAXUserParseFile": """
 | |
| #ifdef LIBXML_SAX1_ENABLED
 | |
|         if (sax == (xmlSAXHandlerPtr)&xmlDefaultSAXHandler) user_data = NULL;
 | |
| #endif
 | |
| """,
 | |
|    "xmlSAXUserParseMemory": """
 | |
| #ifdef LIBXML_SAX1_ENABLED
 | |
|         if (sax == (xmlSAXHandlerPtr)&xmlDefaultSAXHandler) user_data = NULL;
 | |
| #endif
 | |
| """,
 | |
|    "xmlParseBalancedChunkMemory": """
 | |
| #ifdef LIBXML_SAX1_ENABLED
 | |
|         if (sax == (xmlSAXHandlerPtr)&xmlDefaultSAXHandler) user_data = NULL;
 | |
| #endif
 | |
| """,
 | |
|    "xmlParseBalancedChunkMemoryRecover": """
 | |
| #ifdef LIBXML_SAX1_ENABLED
 | |
|         if (sax == (xmlSAXHandlerPtr)&xmlDefaultSAXHandler) user_data = NULL;
 | |
| #endif
 | |
| """,
 | |
|    "xmlParserInputBufferCreateFd":
 | |
|        "if (fd >= 0) fd = -1;",
 | |
| }
 | |
| extra_post_call = {
 | |
|    "xmlAddChild": 
 | |
|        "if (ret_val == NULL) { xmlFreeNode(cur) ; cur = NULL ; }",
 | |
|    "xmlAddEntity":
 | |
|        "if (ret_val != NULL) { xmlFreeNode(ret_val) ; ret_val = NULL; }",
 | |
|    "xmlAddChildList": 
 | |
|        "if (ret_val == NULL) { xmlFreeNodeList(cur) ; cur = NULL ; }",
 | |
|    "xmlAddSibling":
 | |
|        "if (ret_val == NULL) { xmlFreeNode(elem) ; elem = NULL ; }",
 | |
|    "xmlAddNextSibling":
 | |
|        "if (ret_val == NULL) { xmlFreeNode(elem) ; elem = NULL ; }",
 | |
|    "xmlAddPrevSibling": 
 | |
|        "if (ret_val == NULL) { xmlFreeNode(elem) ; elem = NULL ; }",
 | |
|    "xmlDocSetRootElement": 
 | |
|        "if (doc == NULL) { xmlFreeNode(root) ; root = NULL ; }",
 | |
|    "xmlReplaceNode": 
 | |
|        """if (cur != NULL) {
 | |
|               xmlUnlinkNode(cur);
 | |
|               xmlFreeNode(cur) ; cur = NULL ; }
 | |
|           if (old != NULL) {
 | |
|               xmlUnlinkNode(old);
 | |
|               xmlFreeNode(old) ; old = NULL ; }
 | |
| \t  ret_val = NULL;""",
 | |
|    "xmlTextMerge": 
 | |
|        """if ((first != NULL) && (first->type != XML_TEXT_NODE)) {
 | |
|               xmlUnlinkNode(second);
 | |
|               xmlFreeNode(second) ; second = NULL ; }""",
 | |
|    "xmlBuildQName": 
 | |
|        """if ((ret_val != NULL) && (ret_val != ncname) &&
 | |
|               (ret_val != prefix) && (ret_val != memory))
 | |
|               xmlFree(ret_val);
 | |
| \t  ret_val = NULL;""",
 | |
|    "xmlNewDocElementContent":
 | |
|        """xmlFreeDocElementContent(doc, ret_val); ret_val = NULL;""",
 | |
|    "xmlDictReference": "xmlDictFree(dict);",
 | |
|    # Functions which deallocates one of their parameters
 | |
|    "xmlXPathConvertBoolean": """val = NULL;""",
 | |
|    "xmlXPathConvertNumber": """val = NULL;""",
 | |
|    "xmlXPathConvertString": """val = NULL;""",
 | |
|    "xmlSaveFileTo": """buf = NULL;""",
 | |
|    "xmlSaveFormatFileTo": """buf = NULL;""",
 | |
|    "xmlIOParseDTD": "input = NULL;",
 | |
|    "xmlRemoveProp": "cur = NULL;",
 | |
|    "xmlNewNs": "if ((node == NULL) && (ret_val != NULL)) xmlFreeNs(ret_val);",
 | |
|    "xmlCopyNamespace": "if (ret_val != NULL) xmlFreeNs(ret_val);",
 | |
|    "xmlCopyNamespaceList": "if (ret_val != NULL) xmlFreeNsList(ret_val);",
 | |
|    "xmlNewTextWriter": "if (ret_val != NULL) out = NULL;",
 | |
|    "xmlNewTextWriterPushParser": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;} if (ret_val != NULL) ctxt = NULL;",
 | |
|    "xmlNewIOInputStream": "if (ret_val != NULL) input = NULL;",
 | |
|    "htmlParseChunk": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;}",
 | |
|    "htmlParseDocument": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;}",
 | |
|    "xmlParseDocument": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;}",
 | |
|    "xmlParseChunk": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;}",
 | |
|    "xmlParseExtParsedEnt": "if (ctxt != NULL) {xmlFreeDoc(ctxt->myDoc); ctxt->myDoc = NULL;}",
 | |
|    "xmlDOMWrapAdoptNode": "if ((node != NULL) && (node->parent == NULL)) {xmlUnlinkNode(node);xmlFreeNode(node);node = NULL;}",
 | |
| }
 | |
| 
 | |
| modules = []
 | |
| 
 | |
| def is_skipped_module(name):
 | |
|     for mod in skipped_modules:
 | |
|         if mod == name:
 | |
|             return 1
 | |
|     return 0
 | |
| 
 | |
| def is_skipped_function(name):
 | |
|     for fun in skipped_functions:
 | |
|         if fun == name:
 | |
|             return 1
 | |
|     # Do not test destructors
 | |
|     if name.find('Free') != -1:
 | |
|         return 1
 | |
|     return 0
 | |
| 
 | |
| def is_skipped_memcheck(name):
 | |
|     for fun in skipped_memcheck:
 | |
|         if fun == name:
 | |
|             return 1
 | |
|     return 0
 | |
| 
 | |
| missing_types = {}
 | |
| def add_missing_type(name, func):
 | |
|     try:
 | |
|         list = missing_types[name]
 | |
|         list.append(func)
 | |
|     except:
 | |
|         missing_types[name] = [func]
 | |
| 
 | |
| generated_param_types = []
 | |
| def add_generated_param_type(name):
 | |
|     generated_param_types.append(name)
 | |
| 
 | |
| generated_return_types = []
 | |
| def add_generated_return_type(name):
 | |
|     generated_return_types.append(name)
 | |
| 
 | |
| missing_functions = {}
 | |
| missing_functions_nr = 0
 | |
| def add_missing_functions(name, module):
 | |
|     global missing_functions_nr
 | |
| 
 | |
|     missing_functions_nr = missing_functions_nr + 1
 | |
|     try:
 | |
|         list = missing_functions[module]
 | |
|         list.append(name)
 | |
|     except:
 | |
|         missing_functions[module] = [name]
 | |
| 
 | |
| #
 | |
| # Provide the type generators and destructors for the parameters
 | |
| #
 | |
| 
 | |
| def type_convert(str, name, info, module, function, pos):
 | |
| #    res = str.replace("    ", " ")
 | |
| #    res = str.replace("   ", " ")
 | |
| #    res = str.replace("  ", " ")
 | |
|     res = str.replace(" *", "_ptr")
 | |
| #    res = str.replace("*", "_ptr")
 | |
|     res = res.replace(" ", "_")
 | |
|     if res == 'const_char_ptr':
 | |
|         if name.find("file") != -1 or \
 | |
|            name.find("uri") != -1 or \
 | |
|            name.find("URI") != -1 or \
 | |
|            info.find("filename") != -1 or \
 | |
|            info.find("URI") != -1 or \
 | |
|            info.find("URL") != -1:
 | |
|             if function.find("Save") != -1 or \
 | |
|                function.find("Create") != -1 or \
 | |
|                function.find("Write") != -1 or \
 | |
|                function.find("Fetch") != -1:
 | |
|                 return('fileoutput')
 | |
|             return('filepath')
 | |
|     if res == 'void_ptr':
 | |
|         if module == 'nanoftp' and name == 'ctx':
 | |
|             return('xmlNanoFTPCtxtPtr')
 | |
|         if function == 'xmlNanoFTPNewCtxt' or \
 | |
|            function == 'xmlNanoFTPConnectTo' or \
 | |
|            function == 'xmlNanoFTPOpen':
 | |
|             return('xmlNanoFTPCtxtPtr')
 | |
|         if module == 'nanohttp' and name == 'ctx':
 | |
|             return('xmlNanoHTTPCtxtPtr')
 | |
|         if function == 'xmlNanoHTTPMethod' or \
 | |
|            function == 'xmlNanoHTTPMethodRedir' or \
 | |
|            function == 'xmlNanoHTTPOpen' or \
 | |
|            function == 'xmlNanoHTTPOpenRedir':
 | |
|             return('xmlNanoHTTPCtxtPtr');
 | |
|         if function == 'xmlIOHTTPOpen':
 | |
|             return('xmlNanoHTTPCtxtPtr')
 | |
|         if name.find("data") != -1:
 | |
|             return('userdata')
 | |
|         if name.find("user") != -1:
 | |
|             return('userdata')
 | |
|     if res == 'xmlDoc_ptr':
 | |
|         res = 'xmlDocPtr'
 | |
|     if res == 'xmlNode_ptr':
 | |
|         res = 'xmlNodePtr'
 | |
|     if res == 'xmlDict_ptr':
 | |
|         res = 'xmlDictPtr'
 | |
|     if res == 'xmlNodePtr' and pos != 0:
 | |
|         if (function == 'xmlAddChild' and pos == 2) or \
 | |
|            (function == 'xmlAddChildList' and pos == 2) or \
 | |
|            (function == 'xmlAddNextSibling' and pos == 2) or \
 | |
|            (function == 'xmlAddSibling' and pos == 2) or \
 | |
|            (function == 'xmlDocSetRootElement' and pos == 2) or \
 | |
|            (function == 'xmlReplaceNode' and pos == 2) or \
 | |
|            (function == 'xmlTextMerge') or \
 | |
|            (function == 'xmlAddPrevSibling' and pos == 2):
 | |
|             return('xmlNodePtr_in');
 | |
|     if res == 'const xmlBufferPtr':
 | |
|         res = 'xmlBufferPtr'
 | |
|     if res == 'xmlChar_ptr' and name == 'name' and \
 | |
|        function.find("EatName") != -1:
 | |
|         return('eaten_name')
 | |
|     if res == 'void_ptr*':
 | |
|         res = 'void_ptr_ptr'
 | |
|     if res == 'char_ptr*':
 | |
|         res = 'char_ptr_ptr'
 | |
|     if res == 'xmlChar_ptr*':
 | |
|         res = 'xmlChar_ptr_ptr'
 | |
|     if res == 'const_xmlChar_ptr*':
 | |
|         res = 'const_xmlChar_ptr_ptr'
 | |
|     if res == 'const_char_ptr*':
 | |
|         res = 'const_char_ptr_ptr'
 | |
|     if res == 'FILE_ptr' and module == 'debugXML':
 | |
|         res = 'debug_FILE_ptr';
 | |
|     if res == 'int' and name == 'options':
 | |
|         if module == 'parser' or module == 'xmlreader':
 | |
|             res = 'parseroptions'
 | |
| 
 | |
|     return res
 | |
| 
 | |
| known_param_types = []
 | |
| 
 | |
| def is_known_param_type(name):
 | |
|     for type in known_param_types:
 | |
|         if type == name:
 | |
|             return 1
 | |
|     return name[-3:] == 'Ptr' or name[-4:] == '_ptr'
 | |
| 
 | |
| def generate_param_type(name, rtype):
 | |
|     global test
 | |
|     for type in known_param_types:
 | |
|         if type == name:
 | |
|             return
 | |
|     for type in generated_param_types:
 | |
|         if type == name:
 | |
|             return
 | |
| 
 | |
|     if name[-3:] == 'Ptr' or name[-4:] == '_ptr':
 | |
|         if rtype[0:6] == 'const ':
 | |
|             crtype = rtype[6:]
 | |
|         else:
 | |
|             crtype = rtype
 | |
| 
 | |
|         define = 0
 | |
|         if module in modules_defines:
 | |
|             test.write("#ifdef %s\n" % (modules_defines[module]))
 | |
|             define = 1
 | |
|         test.write("""
 | |
| #define gen_nb_%s 1
 | |
| #define gen_%s(no, nr) NULL
 | |
| #define des_%s(no, val, nr)
 | |
| """ % (name, name, name))
 | |
|         if define == 1:
 | |
|             test.write("#endif\n\n")
 | |
|         add_generated_param_type(name)
 | |
| 
 | |
| #
 | |
| # Provide the type destructors for the return values
 | |
| #
 | |
| 
 | |
| known_return_types = []
 | |
| 
 | |
| def is_known_return_type(name):
 | |
|     for type in known_return_types:
 | |
|         if type == name:
 | |
|             return 1
 | |
|     return 0
 | |
| 
 | |
| #
 | |
| # Copy the beginning of the C test program result
 | |
| #
 | |
| 
 | |
| try:
 | |
|     input = open("testapi.c", "r")
 | |
| except:
 | |
|     input = open(srcPref + "testapi.c", "r")
 | |
| test = open('testapi.c.new', 'w')
 | |
| 
 | |
| def compare_and_save():
 | |
|     global test
 | |
| 
 | |
|     test.close()
 | |
|     try:
 | |
|         input = open("testapi.c", "r").read()
 | |
|     except:
 | |
|         input = ''
 | |
|     test = open('testapi.c.new', "r").read()
 | |
|     if input != test:
 | |
|         try:
 | |
|             os.system("rm testapi.c; mv testapi.c.new testapi.c")
 | |
|         except:
 | |
|             os.system("mv testapi.c.new testapi.c")
 | |
|         print("Updated testapi.c")
 | |
|     else:
 | |
|         print("Generated testapi.c is identical")
 | |
| 
 | |
| line = input.readline()
 | |
| while line != "":
 | |
|     if line == "/* CUT HERE: everything below that line is generated */\n":
 | |
|         break;
 | |
|     if line[0:15] == "#define gen_nb_":
 | |
|         type = line[15:].split()[0]
 | |
|         known_param_types.append(type)
 | |
|     if line[0:19] == "static void desret_":
 | |
|         type = line[19:].split('(')[0]
 | |
|         known_return_types.append(type)
 | |
|     test.write(line)
 | |
|     line = input.readline()
 | |
| input.close()
 | |
| 
 | |
| if line == "":
 | |
|     print("Could not find the CUT marker in testapi.c skipping generation")
 | |
|     test.close()
 | |
|     sys.exit(0)
 | |
| 
 | |
| print("Scanned testapi.c: found %d parameters types and %d return types\n" % (
 | |
|       len(known_param_types), len(known_return_types)))
 | |
| test.write("/* CUT HERE: everything below that line is generated */\n")
 | |
| 
 | |
| 
 | |
| #
 | |
| # Open the input API description
 | |
| #
 | |
| doc = libxml2.readFile(srcPref + 'doc/libxml2-api.xml', None, 0)
 | |
| if doc == None:
 | |
|     print("Failed to load doc/libxml2-api.xml")
 | |
|     sys.exit(1)
 | |
| ctxt = doc.xpathNewContext()
 | |
| 
 | |
| #
 | |
| # Generate a list of all function parameters and select only
 | |
| # those used in the api tests
 | |
| #
 | |
| argtypes = {}
 | |
| args = ctxt.xpathEval("/api/symbols/function/arg")
 | |
| for arg in args:
 | |
|     mod = arg.xpathEval('string(../@file)')
 | |
|     func = arg.xpathEval('string(../@name)')
 | |
|     if (mod not in skipped_modules) and (func not in skipped_functions):
 | |
|         type = arg.xpathEval('string(@type)')
 | |
|         if type not in argtypes:
 | |
|             argtypes[type] = func
 | |
| 
 | |
| # similarly for return types
 | |
| rettypes = {}
 | |
| rets = ctxt.xpathEval("/api/symbols/function/return")
 | |
| for ret in rets:
 | |
|     mod = ret.xpathEval('string(../@file)')
 | |
|     func = ret.xpathEval('string(../@name)')
 | |
|     if (mod not in skipped_modules) and (func not in skipped_functions):
 | |
|         type = ret.xpathEval('string(@type)')
 | |
|         if type not in rettypes:
 | |
|             rettypes[type] = func
 | |
| 
 | |
| #
 | |
| # Generate constructors and return type handling for all enums
 | |
| # which are used as function parameters
 | |
| #
 | |
| enums = ctxt.xpathEval("/api/symbols/typedef[@type='enum']")
 | |
| for enum in enums:
 | |
|     module = enum.xpathEval('string(@file)')
 | |
|     name = enum.xpathEval('string(@name)')
 | |
|     #
 | |
|     # Skip any enums which are not in our filtered lists
 | |
|     #
 | |
|     if (name == None) or ((name not in argtypes) and (name not in rettypes)):
 | |
|         continue;
 | |
|     define = 0
 | |
| 
 | |
|     if (name in argtypes) and is_known_param_type(name) == 0:
 | |
|         values = ctxt.xpathEval("/api/symbols/enum[@type='%s']" % name)
 | |
|         i = 0
 | |
|         vals = []
 | |
|         for value in values:
 | |
|             vname = value.xpathEval('string(@name)')
 | |
|             if vname == None:
 | |
|                 continue;
 | |
|             i = i + 1
 | |
|             if i >= 5:
 | |
|                 break;
 | |
|             vals.append(vname)
 | |
|         if vals == []:
 | |
|             print("Didn't find any value for enum %s" % (name))
 | |
|             continue
 | |
|         if module in modules_defines:
 | |
|             test.write("#ifdef %s\n" % (modules_defines[module]))
 | |
|             define = 1
 | |
|         test.write("#define gen_nb_%s %d\n" % (name, len(vals)))
 | |
|         test.write("""static %s gen_%s(int no, int nr ATTRIBUTE_UNUSED) {\n""" %
 | |
|                    (name, name))
 | |
|         i = 1
 | |
|         for value in vals:
 | |
|             test.write("    if (no == %d) return(%s);\n" % (i, value))
 | |
|             i = i + 1
 | |
|         test.write("""    return(0);
 | |
| }
 | |
| 
 | |
| static void des_%s(int no ATTRIBUTE_UNUSED, %s val ATTRIBUTE_UNUSED, int nr ATTRIBUTE_UNUSED) {
 | |
| }
 | |
| 
 | |
| """ % (name, name));
 | |
|         known_param_types.append(name)
 | |
| 
 | |
|     if (is_known_return_type(name) == 0) and (name in rettypes):
 | |
|         if define == 0 and (module in modules_defines):
 | |
|             test.write("#ifdef %s\n" % (modules_defines[module]))
 | |
|             define = 1
 | |
|         test.write("""static void desret_%s(%s val ATTRIBUTE_UNUSED) {
 | |
| }
 | |
| 
 | |
| """ % (name, name))
 | |
|         known_return_types.append(name)
 | |
|     if define == 1:
 | |
|         test.write("#endif\n\n")
 | |
| 
 | |
| #
 | |
| # Load the interfaces
 | |
| # 
 | |
| headers = ctxt.xpathEval("/api/files/file")
 | |
| for file in headers:
 | |
|     name = file.xpathEval('string(@name)')
 | |
|     if (name == None) or (name == ''):
 | |
|         continue
 | |
| 
 | |
|     #
 | |
|     # Some module may be skipped because they don't really consists
 | |
|     # of user callable APIs
 | |
|     #
 | |
|     if is_skipped_module(name):
 | |
|         continue
 | |
| 
 | |
|     #
 | |
|     # do not test deprecated APIs
 | |
|     #
 | |
|     desc = file.xpathEval('string(description)')
 | |
|     if desc.find('DEPRECATED') != -1:
 | |
|         print("Skipping deprecated interface %s" % name)
 | |
|         continue;
 | |
| 
 | |
|     test.write("#include <libxml/%s.h>\n" % name)
 | |
|     modules.append(name)
 | |
|         
 | |
| #
 | |
| # Generate the callers signatures
 | |
| # 
 | |
| for module in modules:
 | |
|     test.write("static int test_%s(void);\n" % module);
 | |
| 
 | |
| #
 | |
| # Generate the top caller
 | |
| # 
 | |
| 
 | |
| test.write("""
 | |
| /**
 | |
|  * testlibxml2:
 | |
|  *
 | |
|  * Main entry point of the tester for the full libxml2 module,
 | |
|  * it calls all the tester entry point for each module.
 | |
|  *
 | |
|  * Returns the number of error found
 | |
|  */
 | |
| static int
 | |
| testlibxml2(void)
 | |
| {
 | |
|     int test_ret = 0;
 | |
| 
 | |
| """)
 | |
| 
 | |
| for module in modules:
 | |
|     test.write("    test_ret += test_%s();\n" % module)
 | |
| 
 | |
| test.write("""
 | |
|     printf("Total: %d functions, %d tests, %d errors\\n",
 | |
|            function_tests, call_tests, test_ret);
 | |
|     return(test_ret);
 | |
| }
 | |
| 
 | |
| """)
 | |
| 
 | |
| #
 | |
| # How to handle a function
 | |
| # 
 | |
| nb_tests = 0
 | |
| 
 | |
| def generate_test(module, node):
 | |
|     global test
 | |
|     global nb_tests
 | |
|     nb_cond = 0
 | |
|     no_gen = 0
 | |
| 
 | |
|     name = node.xpathEval('string(@name)')
 | |
|     if is_skipped_function(name):
 | |
|         return
 | |
| 
 | |
|     #
 | |
|     # check we know how to handle the args and return values
 | |
|     # and store the information for the generation
 | |
|     #
 | |
|     try:
 | |
|         args = node.xpathEval("arg")
 | |
|     except:
 | |
|         args = []
 | |
|     t_args = []
 | |
|     n = 0
 | |
|     for arg in args:
 | |
|         n = n + 1
 | |
|         rtype = arg.xpathEval("string(@type)")
 | |
|         if rtype == 'void':
 | |
|             break;
 | |
|         info = arg.xpathEval("string(@info)")
 | |
|         nam = arg.xpathEval("string(@name)")
 | |
|         type = type_convert(rtype, nam, info, module, name, n)
 | |
|         if is_known_param_type(type) == 0:
 | |
|             add_missing_type(type, name);
 | |
|             no_gen = 1
 | |
|         if (type[-3:] == 'Ptr' or type[-4:] == '_ptr') and \
 | |
|             rtype[0:6] == 'const ':
 | |
|             crtype = rtype[6:]
 | |
|         else:
 | |
|             crtype = rtype
 | |
|         t_args.append((nam, type, rtype, crtype, info))
 | |
|     
 | |
|     try:
 | |
|         rets = node.xpathEval("return")
 | |
|     except:
 | |
|         rets = []
 | |
|     t_ret = None
 | |
|     for ret in rets:
 | |
|         rtype = ret.xpathEval("string(@type)")
 | |
|         info = ret.xpathEval("string(@info)")
 | |
|         type = type_convert(rtype, 'return', info, module, name, 0)
 | |
|         if rtype == 'void':
 | |
|             break
 | |
|         if is_known_return_type(type) == 0:
 | |
|             add_missing_type(type, name);
 | |
|             no_gen = 1
 | |
|         t_ret = (type, rtype, info)
 | |
|         break
 | |
| 
 | |
|     if no_gen == 0:
 | |
|         for t_arg in t_args:
 | |
|             (nam, type, rtype, crtype, info) = t_arg
 | |
|             generate_param_type(type, rtype)
 | |
| 
 | |
|     test.write("""
 | |
| static int
 | |
| test_%s(void) {
 | |
|     int test_ret = 0;
 | |
| 
 | |
| """ % (name))
 | |
| 
 | |
|     if no_gen == 1:
 | |
|         add_missing_functions(name, module)
 | |
|         test.write("""
 | |
|     /* missing type support */
 | |
|     return(test_ret);
 | |
| }
 | |
| 
 | |
| """)
 | |
|         return
 | |
| 
 | |
|     try:
 | |
|         conds = node.xpathEval("cond")
 | |
|         for cond in conds:
 | |
|             test.write("#if %s\n" % (cond.get_content()))
 | |
|             nb_cond = nb_cond + 1
 | |
|     except:
 | |
|         pass
 | |
| 
 | |
|     define = 0
 | |
|     if name in function_defines:
 | |
|         test.write("#ifdef %s\n" % (function_defines[name]))
 | |
|         define = 1
 | |
|     
 | |
|     # Declare the memory usage counter
 | |
|     no_mem = is_skipped_memcheck(name)
 | |
|     if no_mem == 0:
 | |
|         test.write("    int mem_base;\n");
 | |
| 
 | |
|     # Declare the return value
 | |
|     if t_ret != None:
 | |
|         test.write("    %s ret_val;\n" % (t_ret[1]))
 | |
| 
 | |
|     # Declare the arguments
 | |
|     for arg in t_args:
 | |
|         (nam, type, rtype, crtype, info) = arg;
 | |
|         # add declaration
 | |
|         test.write("    %s %s; /* %s */\n" % (crtype, nam, info))
 | |
|         test.write("    int n_%s;\n" % (nam))
 | |
|     test.write("\n")
 | |
| 
 | |
|     # Cascade loop on of each argument list of values
 | |
|     for arg in t_args:
 | |
|         (nam, type, rtype, crtype, info) = arg;
 | |
|         #
 | |
|         test.write("    for (n_%s = 0;n_%s < gen_nb_%s;n_%s++) {\n" % (
 | |
|                    nam, nam, type, nam))
 | |
|     
 | |
|     # log the memory usage
 | |
|     if no_mem == 0:
 | |
|         test.write("        mem_base = xmlMemBlocks();\n");
 | |
| 
 | |
|     # prepare the call
 | |
|     i = 0;
 | |
|     for arg in t_args:
 | |
|         (nam, type, rtype, crtype, info) = arg;
 | |
|         #
 | |
|         test.write("        %s = gen_%s(n_%s, %d);\n" % (nam, type, nam, i))
 | |
|         i = i + 1;
 | |
| 
 | |
|     # add checks to avoid out-of-bounds array access
 | |
|     i = 0;
 | |
|     for arg in t_args:
 | |
|         (nam, type, rtype, crtype, info) = arg;
 | |
|         # assume that "size", "len", and "start" parameters apply to either
 | |
|         # the nearest preceding or following char pointer
 | |
|         if type == "int" and (nam == "size" or nam == "len" or nam == "start"):
 | |
|             for j in (list(range(i - 1, -1, -1)) + list(range(i + 1, len(t_args)))):
 | |
|                 (bnam, btype) = t_args[j][:2]
 | |
|                 if btype == "const_char_ptr" or btype == "const_xmlChar_ptr":
 | |
|                     test.write(
 | |
|                         "        if ((%s != NULL) &&\n"
 | |
|                         "            (%s > xmlStrlen(BAD_CAST %s)))\n"
 | |
|                         "            %s = 0;\n"
 | |
|                         % (bnam, nam, bnam, nam))
 | |
|                     break
 | |
|         i = i + 1;
 | |
| 
 | |
|     # do the call, and clanup the result
 | |
|     if name in extra_pre_call:
 | |
|         test.write("        %s\n"% (extra_pre_call[name]))
 | |
|     if t_ret != None:
 | |
|         test.write("\n        ret_val = %s(" % (name))
 | |
|         need = 0
 | |
|         for arg in t_args:
 | |
|             (nam, type, rtype, crtype, info) = arg
 | |
|             if need:
 | |
|                 test.write(", ")
 | |
|             else:
 | |
|                 need = 1
 | |
|             if rtype != crtype:
 | |
|                 test.write("(%s)" % rtype)
 | |
|             test.write("%s" % nam);
 | |
|         test.write(");\n")
 | |
|         if name in extra_post_call:
 | |
|             test.write("        %s\n"% (extra_post_call[name]))
 | |
|         test.write("        desret_%s(ret_val);\n" % t_ret[0])
 | |
|     else:
 | |
|         test.write("\n        %s(" % (name));
 | |
|         need = 0;
 | |
|         for arg in t_args:
 | |
|             (nam, type, rtype, crtype, info) = arg;
 | |
|             if need:
 | |
|                 test.write(", ")
 | |
|             else:
 | |
|                 need = 1
 | |
|             if rtype != crtype:
 | |
|                 test.write("(%s)" % rtype)
 | |
|             test.write("%s" % nam)
 | |
|         test.write(");\n")
 | |
|         if name in extra_post_call:
 | |
|             test.write("        %s\n"% (extra_post_call[name]))
 | |
| 
 | |
|     test.write("        call_tests++;\n");
 | |
| 
 | |
|     # Free the arguments
 | |
|     i = 0;
 | |
|     for arg in t_args:
 | |
|         (nam, type, rtype, crtype, info) = arg;
 | |
|         # This is a hack to prevent generating a destructor for the
 | |
|         # 'input' argument in xmlTextReaderSetup.  There should be
 | |
|         # a better, more generic way to do this!
 | |
|         if info.find('destroy') == -1:
 | |
|             test.write("        des_%s(n_%s, " % (type, nam))
 | |
|             if rtype != crtype:
 | |
|                 test.write("(%s)" % rtype)
 | |
|             test.write("%s, %d);\n" % (nam, i))
 | |
|         i = i + 1;
 | |
| 
 | |
|     test.write("        xmlResetLastError();\n");
 | |
|     # Check the memory usage
 | |
|     if no_mem == 0:
 | |
|         test.write("""        if (mem_base != xmlMemBlocks()) {
 | |
|             printf("Leak of %%d blocks found in %s",
 | |
| \t           xmlMemBlocks() - mem_base);
 | |
| \t    test_ret++;
 | |
| """ % (name));
 | |
|         for arg in t_args:
 | |
|             (nam, type, rtype, crtype, info) = arg;
 | |
|             test.write("""            printf(" %%d", n_%s);\n""" % (nam))
 | |
|         test.write("""            printf("\\n");\n""")
 | |
|         test.write("        }\n")
 | |
| 
 | |
|     for arg in t_args:
 | |
|         test.write("    }\n")
 | |
| 
 | |
|     test.write("    function_tests++;\n")
 | |
|     #
 | |
|     # end of conditional
 | |
|     #
 | |
|     while nb_cond > 0:
 | |
|         test.write("#endif\n")
 | |
|         nb_cond = nb_cond -1
 | |
|     if define == 1:
 | |
|         test.write("#endif\n")
 | |
| 
 | |
|     nb_tests = nb_tests + 1;
 | |
| 
 | |
|     test.write("""
 | |
|     return(test_ret);
 | |
| }
 | |
| 
 | |
| """)
 | |
|     
 | |
| #
 | |
| # Generate all module callers
 | |
| #
 | |
| for module in modules:
 | |
|     # gather all the functions exported by that module
 | |
|     try:
 | |
|         functions = ctxt.xpathEval("/api/symbols/function[@file='%s']" % (module))
 | |
|     except:
 | |
|         print("Failed to gather functions from module %s" % (module))
 | |
|         continue;
 | |
| 
 | |
|     # iterate over all functions in the module generating the test
 | |
|     i = 0
 | |
|     nb_tests_old = nb_tests
 | |
|     for function in functions:
 | |
|         i = i + 1
 | |
|         generate_test(module, function);
 | |
| 
 | |
|     # header
 | |
|     test.write("""static int
 | |
| test_%s(void) {
 | |
|     int test_ret = 0;
 | |
| 
 | |
|     if (quiet == 0) printf("Testing %s : %d of %d functions ...\\n");
 | |
| """ % (module, module, nb_tests - nb_tests_old, i))
 | |
| 
 | |
|     # iterate over all functions in the module generating the call
 | |
|     for function in functions:
 | |
|         name = function.xpathEval('string(@name)')
 | |
|         if is_skipped_function(name):
 | |
|             continue
 | |
|         test.write("    test_ret += test_%s();\n" % (name))
 | |
| 
 | |
|     # footer
 | |
|     test.write("""
 | |
|     if (test_ret != 0)
 | |
| \tprintf("Module %s: %%d errors\\n", test_ret);
 | |
|     return(test_ret);
 | |
| }
 | |
| """ % (module))
 | |
| 
 | |
| #
 | |
| # Generate direct module caller
 | |
| #
 | |
| test.write("""static int
 | |
| test_module(const char *module) {
 | |
| """);
 | |
| for module in modules:
 | |
|     test.write("""    if (!strcmp(module, "%s")) return(test_%s());\n""" % (
 | |
|         module, module))
 | |
| test.write("""    return(0);
 | |
| }
 | |
| """);
 | |
| 
 | |
| print("Generated test for %d modules and %d functions" %(len(modules), nb_tests))
 | |
| 
 | |
| compare_and_save()
 | |
| 
 | |
| missing_list = []
 | |
| for missing in missing_types.keys():
 | |
|     if missing == 'va_list' or missing == '...':
 | |
|         continue;
 | |
| 
 | |
|     n = len(missing_types[missing])
 | |
|     missing_list.append((n, missing))
 | |
| 
 | |
| missing_list.sort(key=lambda a: a[0])
 | |
| print("Missing support for %d functions and %d types see missing.lst" % (missing_functions_nr, len(missing_list)))
 | |
| lst = open("missing.lst", "w")
 | |
| lst.write("Missing support for %d types" % (len(missing_list)))
 | |
| lst.write("\n")
 | |
| for miss in missing_list:
 | |
|     lst.write("%s: %d :" % (miss[1], miss[0]))
 | |
|     i = 0
 | |
|     for n in missing_types[miss[1]]:
 | |
|         i = i + 1
 | |
|         if i > 5:
 | |
|             lst.write(" ...")
 | |
|             break
 | |
|         lst.write(" %s" % (n))
 | |
|     lst.write("\n")
 | |
| lst.write("\n")
 | |
| lst.write("\n")
 | |
| lst.write("Missing support per module");
 | |
| for module in missing_functions.keys():
 | |
|     lst.write("module %s:\n   %s\n" % (module, missing_functions[module]))
 | |
| 
 | |
| lst.close()
 | |
| 
 | |
| 
 |