mirror of
https://github.com/reactos/CMake.git
synced 2025-01-10 05:31:02 +00:00
9b2dda5db5
Run cmparseMSBuildXML.py on cl.xml, lib.xml, and link.xml to generate our flag tables: python cmparseMSBuildXML.py -x ".../MSBuild/Microsoft.Cpp/v4.0/V110/1033/cl.xml" > cmVS11CLFlagTable.h python cmparseMSBuildXML.py -x ".../MSBuild/Microsoft.Cpp/v4.0/V110/1033/lib.xml" > cmVS11LibFlagTable.h python cmparseMSBuildXML.py -x ".../MSBuild/Microsoft.Cpp/v4.0/V110/1033/link.xml" > cmVS11LinkFlagTable.h Fix up the declaration names at the top of each file. Finally, teach cmVisualStudio10TargetGenerator to select the version of the table matching the version of VS.
333 lines
15 KiB
Python
Executable File
333 lines
15 KiB
Python
Executable File
# 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
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/1033/cl.xml"
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/1033/lib.xml"
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/1033/link.xml"
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/V110/1033/cl.xml"
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/V110/1033/lib.xml"
|
|
# "${PROGRAMFILES}/MSBuild/Microsoft.Cpp/v4.0/V110/1033/link.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","DisplayName","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","DisplayName","ReadOnly"],child))
|
|
if child.nodeName == "StringListProperty":
|
|
self.stringListProperties.append(Property("StringList",["Name","Category","Switch","DisplayName","Subtype"],child))
|
|
if child.nodeName == "BoolProperty":
|
|
self.boolProperties.append(Property("Bool",["ReverseSwitch","Name","Category","Switch","DisplayName","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"
|
|
lastProp = {}
|
|
for i in self.enumProperties:
|
|
if i.attributes["Name"] == "CompileAsManaged":
|
|
#write these out after the rest of the enumProperties
|
|
lastProp = i
|
|
continue
|
|
for j in i.values:
|
|
#hardcore Brad King's manual fixes for cmVS10CLFlagTable.h
|
|
if i.attributes["Name"] == "PrecompiledHeader" and j.attributes["Switch"] != "":
|
|
toReturn+=" {\""+i.attributes["Name"]+"\", \""+j.attributes["Switch"]+"\",\n \""+j.attributes["DisplayName"]+"\", \""+j.attributes["Name"]+"\",\n cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue},\n"
|
|
else:
|
|
#default (normal, non-hardcoded) case
|
|
toReturn+=" {\""+i.attributes["Name"]+"\", \""+j.attributes["Switch"]+"\",\n \""+j.attributes["DisplayName"]+"\", \""+j.attributes["Name"]+"\", 0},\n"
|
|
toReturn += "\n"
|
|
|
|
if lastProp != {}:
|
|
for j in lastProp.values:
|
|
toReturn+=" {\""+lastProp.attributes["Name"]+"\", \""+j.attributes["Switch"]+"\",\n \""+j.attributes["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\",\n cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue},\n"
|
|
toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["ReverseSwitch"]+"\", \""+i.attributes["DisplayName"]+"\", \"\",\n cmVS7FlagTable::UserValueRequired},\n"
|
|
if i.attributes["Switch"] != "":
|
|
toReturn += " {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+"\", \"\", \"true\",\n cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue},\n"
|
|
toReturn += " {\""+i.argumentProperty+"\", \""+i.attributes["Switch"]+"\", \""+i.attributes["DisplayName"]+"\", \"\",\n cmVS7FlagTable::UserValueRequired},\n"
|
|
|
|
toReturn += "\n //String List Properties\n"
|
|
for i in self.stringListProperties:
|
|
if i.attributes["Switch"] == "":
|
|
toReturn += " // Skip [" + i.attributes["Name"] + "] - no command line Switch.\n";
|
|
else:
|
|
toReturn +=" {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+"\",\n \""+i.attributes["DisplayName"]+"\",\n \"\", cmVS7FlagTable::UserValue | cmVS7FlagTable::SemicolonAppendable},\n"
|
|
|
|
toReturn += "\n //String Properties\n"
|
|
for i in self.stringProperties:
|
|
if i.attributes["Switch"] == "":
|
|
if i.attributes["Name"] == "PrecompiledHeaderFile":
|
|
#more hardcoding
|
|
toReturn += " {\"PrecompiledHeaderFile\", \"Yc\",\n"
|
|
toReturn += " \"Precompiled Header Name\",\n"
|
|
toReturn += " \"\", cmVS7FlagTable::UserValueRequired},\n"
|
|
toReturn += " {\"PrecompiledHeaderFile\", \"Yu\",\n"
|
|
toReturn += " \"Precompiled Header Name\",\n"
|
|
toReturn += " \"\", cmVS7FlagTable::UserValueRequired},\n"
|
|
else:
|
|
toReturn += " // Skip [" + i.attributes["Name"] + "] - no command line Switch.\n";
|
|
else:
|
|
toReturn +=" {\""+i.attributes["Name"]+"\", \""+i.attributes["Switch"]+i.attributes["Separator"]+"\",\n \""+i.attributes["DisplayName"]+"\",\n \"\", cmVS7FlagTable::UserValue},\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)
|