diff options
Diffstat (limited to 'Source/cmparseMSBuildXML.py')
-rwxr-xr-x | Source/cmparseMSBuildXML.py | 294 |
1 files changed, 294 insertions, 0 deletions
diff --git a/Source/cmparseMSBuildXML.py b/Source/cmparseMSBuildXML.py new file mode 100755 index 0000000..0d1959c --- /dev/null +++ b/Source/cmparseMSBuildXML.py @@ -0,0 +1,294 @@ +# This python script parses the spec files from MSBuild to create +# mappings from compiler options to IDE XML specifications. For +# more information see here: + +# http://blogs.msdn.com/vcblog/archive/2008/12/16/msbuild-task.aspx +# cl.xml +# +# BoolProperty <Name>true|false</Name> +# simple example: +# <BoolProperty ReverseSwitch="Oy-" Name="OmitFramePointers" +# Category="Optimization" Switch="Oy"> +# <BoolProperty.DisplayName> <BoolProperty.Description> +# <CLCompile> +# <OmitFramePointers>true</OmitFramePointers> +# </ClCompile> +# +# argument means it might be this: /MP3 +# example with argument: +# <BoolProperty Name="MultiProcessorCompilation" Category="General" Switch="MP"> +# <BoolProperty.DisplayName> +# <sys:String>Multi-processor Compilation</sys:String> +# </BoolProperty.DisplayName> +# <BoolProperty.Description> +# <sys:String>Multi-processor Compilation</sys:String> +# </BoolProperty.Description> +# <Argument Property="ProcessorNumber" IsRequired="false" /> +# </BoolProperty> +# <CLCompile> +# <MultiProcessorCompilation>true</MultiProcessorCompilation> +# <ProcessorNumber>4</ProcessorNumber> +# </ClCompile> +# IntProperty +# not used AFIT +# <IntProperty Name="ProcessorNumber" Category="General" Visible="false"> + + +# per config options example +# <EnableFiberSafeOptimizations Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">true</EnableFiberSafeOptimizations> +# +# EnumProperty +# <EnumProperty Name="Optimization" Category="Optimization"> +# <EnumProperty.DisplayName> +# <sys:String>Optimization</sys:String> +# </EnumProperty.DisplayName> +# <EnumProperty.Description> +# <sys:String>Select option for code optimization; choose Custom to use specific optimization options. (/Od, /O1, /O2, /Ox)</sys:String> +# </EnumProperty.Description> +# <EnumValue Name="MaxSpeed" Switch="O2"> +# <EnumValue.DisplayName> +# <sys:String>Maximize Speed</sys:String> +# </EnumValue.DisplayName> +# <EnumValue.Description> +# <sys:String>Equivalent to /Og /Oi /Ot /Oy /Ob2 /Gs /GF /Gy</sys:String> +# </EnumValue.Description> +# </EnumValue> +# <EnumValue Name="MinSpace" Switch="O1"> +# <EnumValue.DisplayName> +# <sys:String>Minimize Size</sys:String> +# </EnumValue.DisplayName> +# <EnumValue.Description> +# <sys:String>Equivalent to /Og /Os /Oy /Ob2 /Gs /GF /Gy</sys:String> +# </EnumValue.Description> +# </EnumValue> +# example for O2 would be this: +# <Optimization>MaxSpeed</Optimization> +# example for O1 would be this: +# <Optimization>MinSpace</Optimization> +# +# StringListProperty +# <StringListProperty Name="PreprocessorDefinitions" Category="Preprocessor" Switch="D "> +# <StringListProperty.DisplayName> +# <sys:String>Preprocessor Definitions</sys:String> +# </StringListProperty.DisplayName> +# <StringListProperty.Description> +# <sys:String>Defines a preprocessing symbols for your source file.</sys:String> +# </StringListProperty.Description> +# </StringListProperty> + +# <StringListProperty Subtype="folder" Name="AdditionalIncludeDirectories" Category="General" Switch="I"> +# <StringListProperty.DisplayName> +# <sys:String>Additional Include Directories</sys:String> +# </StringListProperty.DisplayName> +# <StringListProperty.Description> +# <sys:String>Specifies one or more directories to add to the include path; separate with semi-colons if more than one. (/I[path])</sys:String> +# </StringListProperty.Description> +# </StringListProperty> +# StringProperty + +# Example add bill include: + +# <AdditionalIncludeDirectories>..\..\..\..\..\..\bill;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories> + + +import sys +from xml.dom.minidom import parse, parseString + +def getText(node): + nodelist = node.childNodes + rc = "" + for child in nodelist: + if child.nodeType == child.TEXT_NODE: + rc = rc + child.data + return rc + +def print_tree(document, spaces=""): + for i in range(len(document.childNodes)): + if document.childNodes[i].nodeType == document.childNodes[i].ELEMENT_NODE: + print spaces+str(document.childNodes[i].nodeName ) + print_tree(document.childNodes[i],spaces+"----") + pass + +########################################################################################### +#Data structure that stores a property of MSBuild +class Property: + #type = type of MSBuild property (ex. if the property is EnumProperty type should be "Enum") + #attributeNames = a list of any attributes that this property could have (ex. if this was a EnumProperty it should be ["Name","Category"]) + #document = the dom file that's root node is the Property node (ex. if you were parsing a BoolProperty the root node should be something like <BoolProperty Name="RegisterOutput" Category="General" IncludeInCommandLine="false"> + def __init__(self,type,attributeNames,document=None): + self.suffix_type = "Property" + self.prefix_type = type + self.attributeNames = attributeNames + self.attributes = {} + self.DisplayName = "" + self.Description = "" + self.argumentProperty = "" + self.argumentIsRequired = "" + self.values = [] + if document is not None: + self.populate(document) + pass + + #document = the dom file that's root node is the Property node (ex. if you were parsing a BoolProperty the root node should be something like <BoolProperty Name="RegisterOutput" Category="General" IncludeInCommandLine="false"> + #spaces = do not use + def populate(self,document, spaces = ""): + if document.nodeName == self.prefix_type+self.suffix_type: + for i in self.attributeNames: + self.attributes[i] = document.getAttribute(i) + for i in range(len(document.childNodes)): + child = document.childNodes[i] + if child.nodeType == child.ELEMENT_NODE: + if child.nodeName == self.prefix_type+self.suffix_type+".DisplayName": + self.DisplayName = getText(child.childNodes[1]) + if child.nodeName == self.prefix_type+self.suffix_type+".Description": + self.Description = getText(child.childNodes[1]) + if child.nodeName == "Argument": + self.argumentProperty = child.getAttribute("Property") + self.argumentIsRequired = child.getAttribute("IsRequired") + if child.nodeName == self.prefix_type+"Value": + va = Property(self.prefix_type,["Name","Switch"]) + va.suffix_type = "Value" + va.populate(child) + self.values.append(va) + self.populate(child,spaces+"----") + pass + + #toString function + def __str__(self): + toReturn = self.prefix_type+self.suffix_type+":" + for i in self.attributeNames: + toReturn += "\n "+i+": "+self.attributes[i] + if self.argumentProperty != "": + toReturn += "\n Argument:\n Property: "+self.argumentProperty+"\n IsRequired: "+self.argumentIsRequired + for i in self.values: + toReturn+="\n "+str(i).replace("\n","\n ") + return toReturn +########################################################################################### + +########################################################################################### +#Class that populates itself from an MSBuild file and outputs it in CMake +#format + +class MSBuildToCMake: + #document = the entire MSBuild xml file + def __init__(self,document=None): + self.enumProperties = [] + self.stringProperties = [] + self.stringListProperties = [] + self.boolProperties = [] + self.intProperties = [] + if document!=None : + self.populate(document) + pass + + #document = the entire MSBuild xml file + #spaces = don't use + #To add a new property (if they exist) copy and paste this code and fill in appropriate places + # + #if child.nodeName == "<Name>Property": + # self.<Name>Properties.append(Property("<Name>",[<List of attributes>],child)) + # + #Replace <Name> with the name of the new property (ex. if property is StringProperty replace <Name> with String) + #Replace <List of attributes> with a list of attributes in your property's root node + #in the __init__ function add the line self.<Name>Properties = [] + # + #That is all that is required to add new properties + # + def populate(self,document, spaces=""): + for i in range(len(document.childNodes)): + child = document.childNodes[i] + if child.nodeType == child.ELEMENT_NODE: + if child.nodeName == "EnumProperty": + self.enumProperties.append(Property("Enum",["Name","Category"],child)) + if child.nodeName == "StringProperty": + self.stringProperties.append(Property("String",["Name","Subtype","Separator","Category","Visible","IncludeInCommandLine","Switch","ReadOnly"],child)) + if child.nodeName == "StringListProperty": + self.stringListProperties.append(Property("StringList",["Name","Category","Switch","Subtype"],child)) + if child.nodeName == "BoolProperty": + self.boolProperties.append(Property("Bool",["ReverseSwitch","Name","Category","Switch","SwitchPrefix","IncludeInCommandLine"],child)) + if child.nodeName == "IntProperty": + self.intProperties.append(Property("Int",["Name","Category","Visible"],child)) + self.populate(child,spaces+"----") + pass + + #outputs information that CMake needs to know about MSBuild xml files + def toCMake(self): + toReturn = "static cmVS7FlagTable cmVS10CxxTable[] =\n{\n" + toReturn += "\n //Enum Properties\n" + for i in self.enumProperties: + for j in i.values: + toReturn+=" {\""+i.attributes["Name"]+"\", \""+j.attributes["Switch"]+"\", \""+j.DisplayName+"\", \""+j.attributes["Name"]+"\", 0},\n" + toReturn += "\n" + + + toReturn += "\n //Bool Properties\n" + for i in self.boolProperties: + if i.argumentProperty == "": + if i.attributes["ReverseSwitch"] != "": + toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["ReverseSwitch"]+"\", \"\", \"false\", 0},\n" + if i.attributes["Switch"] != "": + toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+"\", \"\", \"true\", 0},\n" + + toReturn += "\n //Bool Properties With Argument\n" + for i in self.boolProperties: + if i.argumentProperty != "": + if i.attributes["ReverseSwitch"] != "": + toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["ReverseSwitch"]+"\", \"\", \"false\", cmVS7FlagTable::Continue},\n" + toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["ReverseSwitch"]+"\", \""+i.DisplayName+"\", \"\", cmVS7FlagTable::UserValueRequired},\n" + if i.attributes["Switch"] != "": + toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+"\", \"\", \"true\", cmVS7FlagTable::Continue},\n" + toReturn += " {\""+i.argumentProperty+"\", \""+i.attributes["Switch"]+"\", \""+i.DisplayName+"\", \"\", cmVS7FlagTable::UserValueRequired},\n" + + toReturn += "\n //String List Properties\n" + for i in self.stringListProperties: + toReturn+=" {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+"\", \""+i.DisplayName+"\", \"\", cmVS7FlagTable::UserValue | cmVS7FlagTable::SemicolonAppendable},\n" + + toReturn += " {0,0,0,0,0}\n};" + return toReturn + pass + + #toString function + def __str__(self): + toReturn = "" + allList = [self.enumProperties,self.stringProperties,self.stringListProperties,self.boolProperties,self.intProperties] + for p in allList: + for i in p: + toReturn += "==================================================\n"+str(i).replace("\n","\n ")+"\n==================================================\n" + + return toReturn +########################################################################################### + +########################################################################################### +# main function +def main(argv): + xml_file = None + help = """ + Please specify an input xml file with -x + + Exiting... + Have a nice day :)""" + for i in range(0,len(argv)): + if argv[i] == "-x": + xml_file = argv[i+1] + if argv[i] == "-h": + print help + sys.exit(0) + pass + if xml_file == None: + print help + sys.exit(1) + + f = open(xml_file,"r") + xml_str = f.read() + xml_dom = parseString(xml_str) + + convertor = MSBuildToCMake(xml_dom) + print convertor.toCMake() + + xml_dom.unlink() +########################################################################################### +# main entry point +if __name__ == "__main__": + main(sys.argv) + +sys.exit(0) |