Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2)

Unified Diff: site_scons/site_tools/_Node_MSVS.py

Issue 42667: Remove Hammer files.... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: Created 11 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « site_scons/site_tools/MSVSNew.py ('k') | site_scons/site_tools/atlmfc_vc80.py » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: site_scons/site_tools/_Node_MSVS.py
===================================================================
--- site_scons/site_tools/_Node_MSVS.py (revision 12583)
+++ site_scons/site_tools/_Node_MSVS.py (working copy)
@@ -1,1463 +0,0 @@
-#
-# __COPYRIGHT__
-#
-# Permission is hereby granted, free of charge, to any person obtaining
-# a copy of this software and associated documentation files (the
-# "Software"), to deal in the Software without restriction, including
-# without limitation the rights to use, copy, modify, merge, publish,
-# distribute, sublicense, and/or sell copies of the Software, and to
-# permit persons to whom the Software is furnished to do so, subject to
-# the following conditions:
-#
-# The above copyright notice and this permission notice shall be included
-# in all copies or substantial portions of the Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
-# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
-# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
-# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
-# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
-# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
-# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
-#
-
-__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
-
-__doc__ = """SCons.Node.MSVS
-
-New implementation of Visual Studio project generation for SCons.
-"""
-
-import md5
-import os
-import random
-import re
-import UserList
-import xml.dom
-import xml.dom.minidom
-
-import SCons.Node.FS
-import SCons.Script
-import SCons.Util
-
-from SCons.Debug import Trace
-TODO = 0
-
-# Initialize random number generator
-random.seed()
-
-
-#------------------------------------------------------------------------------
-# Entry point for supplying a fixed map of GUIDs for testing.
-
-GUIDMap = {}
-
-
-#------------------------------------------------------------------------------
-# Helper functions
-
-
-def MakeGuid(name, seed='msvs_new'):
- """Returns a GUID for the specified target name.
-
- Args:
- name: Target name.
- seed: Seed for MD5 hash.
- Returns:
- A GUID-line string calculated from the name and seed.
-
- This generates something which looks like a GUID, but depends only on the
- name and seed. This means the same name/seed will always generate the same
- GUID, so that projects and solutions which refer to each other can explicitly
- determine the GUID to refer to explicitly. It also means that the GUID will
- not change when the project for a target is rebuilt.
- """
- # Calculate a MD5 signature for the seed and name.
- d = md5.new(str(seed) + str(name)).hexdigest().upper()
- # Convert most of the signature to GUID form (discard the rest)
- guid = ('{' + d[:8] + '-' + d[8:12] + '-' + d[12:16] + '-' + d[16:20]
- + '-' + d[20:32] + '}')
- return guid
-
-
-#------------------------------------------------------------------------------
-# Global look up of string names.
-
-class LookupError(Exception):
- def __str__(self):
- string, expanded = self.args
- if string == expanded:
- return string
- else:
- return '%s (%s)' % (string, expanded)
-
-_lookup_dict = {}
-
-def LookupAdd(item, result):
- _lookup_dict[item] = result
- _lookup_dict[result] = result
-
-def Lookup(item):
- """Looks up an MSVS item in the global dictionary.
-
- Args:
- item: A path (string) or instance for looking up.
- Returns:
- An instance from the global _lookup_dict.
-
- Raises an exception if the item does not exist in the _lookup_dict.
- """
- global _lookup_dict
- try:
- return _lookup_dict[item]
- except KeyError:
- return SCons.Node.FS.default_fs.Entry(item, create=False)
-
-def LookupCreate(klass, item, *args, **kw):
- """Looks up an MSVS item, creating it if it doesn't already exist.
-
- Args:
- klass: The class of item being looked up, or created if it
- doesn't already exist in the global _lookup_dict.
- item: The a string (or instance) being looked up.
- *args: positional arguments passed to the klass.initialize() method.
- **kw: keyword arguments passed to the klass.initialize() method.
- Returns:
- An instance from the global _lookup_dict, or None if the item does
- not exist in the _lookup_dict.
-
- This raises a LookupError if the found instance doesn't match the
- requested klass.
-
- When creating a new instance, this populates the _lookup_dict with
- both the item and the instance itself as keys, so that looking up
- the instance will return itself.
- """
- global _lookup_dict
- result = _lookup_dict.get(item)
- if result:
- if not isinstance(result, klass):
- raise LookupError, "tried to redefine %s as a %s" % (item, klass)
- return result
- result = klass()
- result.initialize(item, *args, **kw)
- LookupAdd(item, result)
- return result
-
-
-#------------------------------------------------------------------------------
-
-class FileList(object):
- def __init__(self, entries=None):
- if isinstance(entries, FileList):
- entries = entries.entries
- self.entries = entries or []
- def __getitem__(self, i):
- return self.entries[i]
- def __setitem__(self, i, item):
- self.entries[i] = item
- def __delitem__(self, i):
- del self.entries[i]
- def __add__(self, other):
- if isinstance(other, FileList):
- return self.__class__(self.entries + other.entries)
- elif isinstance(other, type(self.entries)):
- return self.__class__(self.entries + other)
- else:
- return self.__class__(self.entries + list(other))
- def __radd__(self, other):
- if isinstance(other, FileList):
- return self.__class__(other.entries + self.entries)
- elif isinstance(other, type(self.entries)):
- return self.__class__(other + self.entries)
- else:
- return self.__class__(list(other) + self.entries)
- def __iadd__(self, other):
- if isinstance(other, FileList):
- self.entries += other.entries
- elif isinstance(other, type(self.entries)):
- self.entries += other
- else:
- self.entries += list(other)
- return self
- def append(self, item):
- return self.entries.append(item)
- def extend(self, item):
- return self.entries.extend(item)
- def index(self, item, *args):
- return self.entries.index(item, *args)
- def remove(self, item):
- return self.entries.remove(item)
-
-def FileListWalk(top, topdown=True, onerror=None):
- """
- """
- try:
- entries = top.entries
- except AttributeError, err:
- if onerror is not None:
- onerror(err)
- return
-
- dirs, nondirs = [], []
- for entry in entries:
- if hasattr(entry, 'entries'):
- dirs.append(entry)
- else:
- nondirs.append(entry)
-
- if topdown:
- yield top, dirs, nondirs
- for entry in dirs:
- for x in FileListWalk(entry, topdown, onerror):
- yield x
- if not topdown:
- yield top, dirs, nondirs
-
-#------------------------------------------------------------------------------
-
-class _MSVSFolder(FileList):
- """Folder in a Visual Studio solution."""
-
- entry_type_guid = '{2150E333-8FDC-42A3-9474-1A3956D46DE8}'
-
- def initialize(self, path, name=None, entries=None, guid=None, items=None):
- """Initializes the folder.
-
- Args:
- path: The unique name of the folder, by which other MSVS Nodes can
- refer to it. This is not necessarily the name that gets printed
- in the .sln file.
- name: The name of this folder as actually written in a generated
- .sln file. The default is
- entries: List of folder entries to nest inside this folder. May contain
- Folder or Project objects. May be None, if the folder is empty.
- guid: GUID to use for folder, if not None.
- items: List of solution items to include in the folder project. May be
- None, if the folder does not directly contain items.
- """
- super(_MSVSFolder, self).__init__(entries)
-
- # For folder entries, the path is the same as the name
- self.msvs_path = path
- self.msvs_name = name or path
-
- self.guid = guid
-
- # Copy passed lists (or set to empty lists)
- self.items = list(items or [])
-
- def get_guid(self):
- if self.guid is None:
- guid = GUIDMap.get(self.msvs_path)
- if not guid:
- # The GUID for the folder can be random, since it's used only inside
- # solution files and doesn't need to be consistent across runs.
- guid = MakeGuid(random.random())
- self.guid = guid
- return self.guid
-
- def get_msvs_path(self, sln):
- return self.msvs_name
-
-def MSVSFolder(env, item, *args, **kw):
- return LookupCreate(_MSVSFolder, item, *args, **kw)
-
-#------------------------------------------------------------------------------
-
-class MSVSConfig(object):
- """Visual Studio configuration."""
- def __init__(self, Name, config_type, tools=None, **attrs):
- """Initializes the configuration.
-
- Args:
- **attrs: Configuration attributes.
- """
- # Special handling for attributes that we want to make more
- # convenient for the user.
- ips = attrs.get('InheritedPropertySheets')
- if ips:
- if isinstance(ips, list):
- ips = ';'.join(ips)
- attrs['InheritedPropertySheets'] = ips.replace('/', '\\')
-
- self.Name = Name
- self.config_type = config_type
- self.tools = tools
- self.attrs = attrs
-
- def CreateElement(self, doc, project):
- """Creates an element for the configuration.
-
- Args:
- doc: xml.dom.Document object to use for node creation.
-
- Returns:
- A new xml.dom.Element for the configuration.
- """
- node = doc.createElement(self.config_type)
- node.setAttribute('Name', self.Name)
- for k, v in self.attrs.items():
- node.setAttribute(k, v)
-
- tools = self.tools
- if tools is None:
- tools = project.tools or []
- if not SCons.Util.is_List(tools):
- tools = [tools]
- tool_objects = []
- for t in tools:
- if not isinstance(t, MSVSTool):
- t = MSVSTool(t)
- tool_objects.append(t)
- for t in tool_objects:
- node.appendChild(t.CreateElement(doc))
-
- return node
-
-
-class MSVSFileListBase(FileList):
- """Base class for a file list in a Visual Studio project file."""
-
- def CreateElement(self, doc, node_func=lambda x: x):
- """Creates an element for an MSVSFileListBase subclass.
-
- Args:
- doc: xml.dom.Document object to use for node creation.
- node_func: Function to use to return Nodes for objects that
- don't have a CreateElement() method of their own.
-
- Returns:
- A new xml.dom.Element for the MSVSFileListBase object.
- """
- node = doc.createElement(self.element_name)
- for entry in self.entries:
- if hasattr(entry, 'CreateElement'):
- n = entry.CreateElement(doc, node_func)
- else:
- n = node_func(entry)
- node.appendChild(n)
- return node
-
-
-class MSVSFiles(MSVSFileListBase):
- """Files list in a Visual Studio project file."""
- element_name = 'Files'
-
-
-class MSVSFilter(MSVSFileListBase):
- """Filter (that is, a virtual folder) in a Visual Studio project file."""
-
- element_name = 'Filter'
-
- def __init__(self, Name, entries=None):
- """Initializes the folder.
-
- Args:
- Name: Filter (folder) name.
- entries: List of filenames and/or Filter objects contained.
- """
- super(MSVSFilter, self).__init__(entries)
- self.Name = Name
-
- def CreateElement(self, doc, node_func=lambda x: x):
- """Creates an element for the Filter.
-
- Args:
- doc: xml.dom.Document object to use for node creation.
- node_func: Function to use to return Nodes for objects that
- don't have a CreateElement() method of their own.
-
- Returns:
- A new xml.dom.Element for the filter.
- """
- node = super(MSVSFilter, self).CreateElement(doc, node_func)
- node.setAttribute('Name', self.Name)
- return node
-
-
-class MSVSTool(object):
- """Visual Studio tool."""
-
- def __init__(self, Name, **attrs):
- """Initializes the tool.
-
- Args:
- Name: Tool name.
- **attrs: Tool attributes.
- """
-
- val = attrs.get('AdditionalDependencies')
- if val:
- if isinstance(val, list):
- val = ' '.join(val)
- attrs['AdditionalDependencies'] = val.replace('/', '\\')
-
- val = attrs.get('AdditionalIncludeDirectories')
- if val:
- if isinstance(val, list):
- val = ';'.join(val)
- attrs['AdditionalIncludeDirectories'] = val.replace('/', '\\')
-
- val = attrs.get('AdditionalManifestFiles')
- if val:
- if isinstance(val, list):
- val = ';'.join(val)
- attrs['AdditionalManifestFiles'] = val.replace('/', '\\')
-
- val = attrs.get('CommandLine')
- if val:
- if isinstance(val, list):
- val = '\r\n'.join(val)
- attrs['CommandLine'] = val.replace('/', '\\')
-
- val = attrs.get('PreprocessorDefinitions')
- if val:
- if isinstance(val, list):
- val = ';'.join(val)
- attrs['PreprocessorDefinitions'] = val
-
- for a in ('ImportLibrary',
- 'ObjectFile',
- 'OutputFile',
- 'Outputs',
- 'XMLDocumentationFileName'):
- val = attrs.get(a)
- if val:
- val = val.replace('/', '\\')
- attrs[a] = val
-
- self.Name = Name
- self.attrs = attrs
-
- def CreateElement(self, doc):
- """Creates an element for the tool.
-
- Args:
- doc: xml.dom.Document object to use for node creation.
-
- Returns:
- A new xml.dom.Element for the tool.
- """
- node = doc.createElement('Tool')
- node.setAttribute('Name', self.Name)
- for k, v in self.attrs.items():
- node.setAttribute(k, v)
- return node
-
- def _format(self):
- """Formats a tool specification for debug printing"""
- xml_impl = xml.dom.getDOMImplementation()
- doc = xml_impl.createDocument(None, 'VisualStudioProject', None)
- return self.CreateElement(doc).toprettyxml()
-
- def diff(self, other):
- for key, value in self.attrs.items():
- if other.attrs[key] == value:
- del self.attrs[key]
-
-
-class MSVSToolFile(object):
- """Visual Studio tool file specification."""
-
- def __init__(self, node, **attrs):
- """Initializes the tool.
-
- Args:
- node: Node for the Tool File
- **attrs: Tool File attributes.
- """
- self.node = node
-
- def CreateElement(self, doc, project):
- result = doc.createElement('ToolFile')
- result.setAttribute('RelativePath', project.get_rel_path(self.node))
- return result
-
-
-#------------------------------------------------------------------------------
-
-def MSVSAction(target, source, env):
- target[0].Write(env)
-
-MSVSProjectAction = SCons.Script.Action(MSVSAction,
- "Generating Visual Studio project `$TARGET' ...")
-
-class _MSVSProject(SCons.Node.FS.File):
- """Visual Studio project."""
-
- entry_type_guid = '{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}'
- initialized = False
-
- def initialize(self, env, path, name = None,
- dependencies = None,
- guid = None,
- buildtargets = [],
- files = [],
- root_namespace = None,
- keyword = None,
- relative_path_prefix = None,
- local_directory_prefix = None,
- relative_path_substitutions = [],
- tools = None,
- configurations = None,
- **attrs):
- """Initializes the project.
-
- Args:
- path: Relative path to project file.
- name: Name of project. If None, the name will be the same as the base
- name of the project file.
- dependencies: List of other Project objects this project is dependent
- upon, if not None.
- guid: GUID to use for project, if not None.
- buildtargets: List of target(s) being built by this project.
- files: List of source files for the project. This will be
- supplemented by any source files of buildtargets.
- root_namespace: The value of the RootNamespace attribute of the
- project, if not None. The default is to use the same
- string as the name.
- relative_path_prefix: A prefix to be appended to the beginning of
- every file name in the list. The canonical use is to specify
- './' to make files explicitly relative to the local directory.
- tools: A list of MSVSTool objects or strings representing
- tools to be used to build this project. This will be used
- for any configurations that don't provide their own
- per-configuration tool list.
- configurations: A list of MSVSConfig objects representing
- configurations built by this project.
- """
- if name is None:
- if buildtargets:
- name = os.path.splitext(buildtargets[0].name)[0]
- else:
- name = os.path.splitext(os.path.basename(path))[0]
- if not root_namespace:
- root_namespace or name
-
- if self.initialized:
- # TODO(sgk): fill in
- if self.msvs_name != name:
- pass
- if self.root_namespace != root_namespace:
- pass
- if self.relative_path_prefix != relative_path_prefix:
- pass
- if self.guid != guid:
- pass
- #if self.env != env:
- # pass
- else:
- self.buildtargets = []
- self.configurations = []
- self.dependencies = []
- self.file_configurations = {}
- self.files = MSVSFiles([])
- self.tool_files = []
- self.file_lists = []
- self.initialized = True
- self.keyword = None
- self.local_directory_prefix = ''
- self.relative_path_prefix = ''
- self.relative_path_substitutions = []
- self.root_namespace = name
-
- self.attrs = attrs
- self.env = env
- self.guid = guid
- self.msvs_name = name
- self.msvs_path = path
- if relative_path_prefix:
- self.relative_path_prefix = relative_path_prefix
- if local_directory_prefix:
- self.local_directory_prefix = local_directory_prefix
- for left, right in relative_path_substitutions:
- t = (left.replace('/', '\\'), right.replace('/', '\\'))
- self.relative_path_substitutions.append(t)
- if root_namespace:
- self.root_namespace = root_namespace
- if keyword:
- self.keyword = keyword
- self.tools = tools
-
- self.buildtargets.extend(buildtargets)
- self.configurations.extend(configurations or [])
- self.dependencies.extend(list(dependencies or []))
- self.AddFiles(files)
-
- env.Command(self, [], MSVSProjectAction)
-
- def args2nodes(self, entries):
- result = []
- for entry in entries:
- if SCons.Util.is_String(entry):
- entry = self.env.File(entry)
- result.append(entry.srcnode())
- elif hasattr(entry, 'entries'):
- entry.entries = self.args2nodes(entry.entries)
- result.append(entry)
- elif isinstance(entry, (list, UserList.UserList)):
- result.extend(self.args2nodes(entry))
- elif hasattr(entry, 'sources') and entry.sources:
- result.extend(entry.sources)
- else:
- result.append(entry.srcnode())
- return result
-
- def FindFile(self, node):
- try:
- flat_file_dict = self.flat_file_dict
- except AttributeError:
- flat_file_dict = {}
- file_list = self.files[:]
- while file_list:
- entry = file_list.pop(0)
- if not isinstance(entry, (list, UserList.UserList)):
- entry = [entry]
- for f in entry:
- if hasattr(f, 'entries'):
- file_list.extend(f.entries)
- else:
- flat_file_dict[f] = True
- flat_file_dict[f.srcnode()] = True
- if hasattr(f, 'sources'):
- for s in f.sources:
- flat_file_dict[s] = True
- flat_file_dict[s.srcnode()] = True
- self.flat_file_dict = flat_file_dict
-
- return flat_file_dict.get(node)
-
- def get_guid(self):
- if self.guid is None:
- guid = GUIDMap.get(self.msvs_path)
- if not guid:
- # Set GUID from path
- # TODO(rspangler): This is fragile.
- # 1. We can't just use the project filename sans path, since there
- # could be multiple projects with the same base name (for example,
- # foo/unittest.vcproj and bar/unittest.vcproj).
- # 2. The path needs to be relative to $SOURCE_ROOT, so that the project
- # GUID is the same whether it's included from base/base.sln or
- # foo/bar/baz/baz.sln.
- # 3. The GUID needs to be the same each time this builder is invoked,
- # so that we don't need to rebuild the solution when the
- # project changes.
- # 4. We should be able to handle pre-built project files by reading the
- # GUID from the files.
- guid = MakeGuid(self.msvs_path)
- self.guid = guid
- return self.guid
-
- def get_msvs_path(self, sln):
- return sln.rel_path(self).replace('/', '\\')
-
- def get_rel_path(self, node):
- result = self.rel_path(node)
- if self.relative_path_prefix:
- if not result.startswith('..'):
- result = self.relative_path_prefix + result
- elif not os.path.split(result)[0]:
- result = self.local_directory_prefix + result
- result = result.replace('/', '\\')
- for left, right in self.relative_path_substitutions:
- result = result.replace(left, right)
- return result
-
- def AddConfig(self, Name, tools=None, **attrs):
- """Adds a configuration to the parent node.
-
- Args:
- Name: The name of the configuration.
- tools: List of tools (strings or Tool objects); may be None.
- **attrs: Configuration attributes.
- """
- if tools is None:
- # No tool list specifically for this configuration,
- # use the Project's as a default.
- tools = self.tools
- if not attrs.has_key('ConfigurationType'):
- # No ConfigurationType specifically for this configuration,
- # use the Project's as a default.
- try:
- attrs['ConfigurationType'] = self.attrs['ConfigurationType']
- except KeyError:
- pass
- if attrs.has_key('InheritedPropertySheets'):
- ips = attrs['InheritedPropertySheets']
- attrs['InheritedPropertySheets'] = self.env.subst(ips)
- c = MSVSConfig(Name, 'Configuration', tools=tools, **attrs)
- self.configurations.append(c)
-
- def AddFiles(self, files):
- """Adds files to the project.
-
- Args:
- files: A list of Filter objects and/or relative paths to files.
-
- This makes a copy of the file/filter tree at the time of this call. If you
- later add files to a Filter object which was passed into a previous call
- to AddFiles(), it will not be reflected in this project.
- """
- self.file_lists.append(self.args2nodes(files))
-
- def _FilesToSourceFiles(self, files):
- file_list = files[:]
- result = []
- while file_list:
- entry = file_list.pop(0)
- if not isinstance(entry, (list, UserList.UserList)):
- entry = [entry]
- for f in entry:
- if hasattr(f, 'entries'):
- self._FilesToSourceFiles(f.entries)
- result.append(f)
- else:
- if f.sources:
- flist = f.sources
- else:
- flist = [f]
- for x in flist:
- result.append(x.srcnode())
- files[:] = result
-
- def _MergeFiles(self, dest_list, src_list):
- for f in src_list:
- if f not in dest_list:
- dest_list.append(f)
- continue
- #if hasattr(f, 'entries'):
- # self._FilesToSourceFiles(f.entries)
-
- def AddFileConfig(self, path, Name, tools=None, **attrs):
- """Adds a configuration to a file.
-
- Args:
- path: Relative path to the file.
- Name: Name of configuration to add.
- tools: List of tools (strings or MSVSTool objects); may be None.
- **attrs: Configuration attributes.
-
- Raises:
- ValueError: Relative path does not match any file added via AddFiles().
- """
- # Store as the VariantDir node, not as the source node.
- node = self.env.File(path)
- c = MSVSConfig(Name, 'FileConfiguration', tools=tools, **attrs)
- config_list = self.file_configurations.get(node)
- if config_list is None:
- config_list = []
- self.file_configurations[node] = config_list
- config_list.append(c)
-
- def AddToolFile(self, path):
- """Adds a tool file to the project.
-
- Args:
- path: Relative path from project to tool file.
- """
- tf = MSVSToolFile(self.env.File(path))
- self.tool_files.append(tf)
-
- def Create(self):
- """Creates the project document.
-
- Args:
- name: Name of the project.
- guid: GUID to use for project, if not None.
- """
- # Create XML doc
- xml_impl = xml.dom.getDOMImplementation()
- self.doc = xml_impl.createDocument(None, 'VisualStudioProject', None)
-
- # Add attributes to root element
- root = self.doc.documentElement
- root.setAttribute('ProjectType', 'Visual C++')
- root.setAttribute('Version', '8.00')
- root.setAttribute('Name', self.msvs_name)
- root.setAttribute('ProjectGUID', self.get_guid())
- root.setAttribute('RootNamespace', self.root_namespace)
- if self.keyword:
- root.setAttribute('Keyword', self.keyword)
-
- # Add platform list
- platforms = self.doc.createElement('Platforms')
- root.appendChild(platforms)
- n = self.doc.createElement('Platform')
- n.setAttribute('Name', 'Win32')
- platforms.appendChild(n)
-
- # Add tool files section
- tool_files = self.doc.createElement('ToolFiles')
- root.appendChild(tool_files)
- for tf in self.tool_files:
- tool_files.appendChild(tf.CreateElement(self.doc, self))
-
- # Add configurations section
- configs = self.doc.createElement('Configurations')
- root.appendChild(configs)
- for c in self.configurations:
- configs.appendChild(c.CreateElement(self.doc, self))
-
- # Add empty References section
- root.appendChild(self.doc.createElement('References'))
-
- # Add files section
- root.appendChild(self.files.CreateElement(self.doc, self.CreateFileElement))
-
- # Add empty Globals section
- root.appendChild(self.doc.createElement('Globals'))
-
- def CreateFileElement(self, file):
- """Create a DOM node for the specified file.
-
- Args:
- file: The file Node being considered.
-
- Returns:
- A DOM Node for the File, with a relative path to the current
- project object, and any file configurations attached to the
- project.
- """
-
- node = self.doc.createElement('File')
- node.setAttribute('RelativePath', self.get_rel_path(file))
- for c in self.file_configurations.get(file, []):
- node.appendChild(c.CreateElement(self.doc, self))
- return node
-
- def VCCLCompilerTool(self, args):
- default_attrs = {
- 'BufferSecurityCheck' : "false",
- 'CompileAs' : 0, # default
- 'DebugInformationFormat' : 0, # TODO(???)
- 'DisableSpecificWarnings' : [],
- 'EnableFiberSafeOptimizations' : "false",
- 'EnableFunctionLevelLinking' : "false",
- 'EnableIntrinsicFunctions' : "false",
- 'FavorSizeOrSpeed' : 0, # favorNone
- 'InlineFunctionExpansion' : 1, # expandDisable
- 'MinimalRebuild' : "false",
- 'OmitFramePointers' : "false",
- 'Optimization' : 1, # optimizeDisabled TODO(???)
- 'PreprocessorDefinitions' : [],
- 'RuntimeLibrary' : TODO,
- 'RuntimeTypeInfo' : "false",
- 'StringPooling' : "false",
- 'SuppressStartupBanner' : "false",
- 'WarningAsError' : "false",
- 'WarningLevel' : 1, # warningLevel_1
- 'WholeProgramOptimization' : "false",
- }
-
- tool = MSVSTool('VCCLCompilerTool', **default_attrs)
- attrs = tool.attrs
-
- for arg in args:
- if arg in ('/c',):
- continue
- if arg.startswith('/Fo'):
- continue
- if arg.startswith('/D'):
- attrs['PreprocessorDefinitions'].append(arg[2:])
- elif arg == '/EH':
- attrs['ExceptionHandling'] = 0
- elif arg == '/GF':
- attrs['StringPooling'] = "true"
- elif arg == '/GL':
- attrs['WholeProgramOptimization'] = "true"
- elif arg == '/GM':
- attrs['MinimalRebuild'] = "true"
- elif arg == '/GR-':
- attrs['RuntimeTypeInfo'] = "true"
- elif arg == '/Gs':
- attrs['BufferSecurityCheck'] = "true"
- elif arg == '/Gs-':
- attrs['BufferSecurityCheck'] = "false"
- elif arg == '/GT':
- attrs['EnableFiberSafeOptimizations'] = "true"
- elif arg == '/Gy':
- attrs['EnableFunctionLevelLinking'] = "true"
- elif arg == '/MD':
- attrs['RuntimeLibrary'] = 1 # rtMultiThreadedDebug
- elif arg == '/MDd':
- attrs['RuntimeLibrary'] = 2 # rtMultiThreadedDebugDLL
- elif arg == '/MT':
- attrs['RuntimeLibrary'] = 0 # rtMultiThreaded
- elif arg == '/MTd':
- attrs['RuntimeLibrary'] = 3 # rtMultiThreadedDLL
- elif arg == '/nologo':
- attrs['SuppressStartupBanner'] = "true"
- elif arg == '/O1':
- attrs['InlineFunctionExpansion'] = 4 # optimizeMinSpace
- elif arg == '/O2':
- attrs['InlineFunctionExpansion'] = 3 # optimizeMaxSpeed
- elif arg == '/Ob1':
- attrs['InlineFunctionExpansion'] = 2 # expandOnlyInline
- elif arg == '/Ob2':
- attrs['InlineFunctionExpansion'] = 0 # expandAnySuitable
- elif arg == '/Od':
- attrs['Optimization'] = 0
- elif arg == '/Oi':
- attrs['EnableIntrinsicFunctions'] = "true"
- elif arg == '/Os':
- attrs['FavorSizeOrSpeed'] = 1 # favorSize
- elif arg == '/Ot':
- attrs['FavorSizeOrSpeed'] = 2 # favorSpeed
- elif arg == '/Ox':
- attrs['Optimization'] = 2 # optimizeFull
- elif arg == '/Oy':
- attrs['OmitFramePointers'] = "true"
- elif arg == '/Oy-':
- attrs['TODO'] = "true"
- elif arg in ('/Tc', '/TC'):
- attrs['CompileAs'] = 1 # compileAsC
- elif arg in ('/Tp', '/TP'):
- attrs['CompileAs'] = 2 # compileAsCPlusPlus
- elif arg == '/WX':
- attrs['WarnAsError'] = "true"
- elif arg.startswith('/W'):
- attrs['WarningLevel'] = int(arg[2:]) # 0 through 4
- elif arg.startswith('/wd'):
- attrs['DisableSpecificWarnings'].append(str(arg[3:]))
- elif arg == '/Z7':
- attrs['DebugInformationFormat'] = 3 # debugOldSytleInfo TODO(???)
- elif arg == '/Zd':
- attrs['DebugInformationFormat'] = 0 # debugDisabled
- elif arg == '/Zi':
- attrs['DebugInformationFormat'] = 2 # debugEnabled TODO(???)
- elif arg == '/ZI':
- attrs['DebugInformationFormat'] = 1 # debugEditAndContinue TODO(???)
-
- cppdefines = attrs['PreprocessorDefinitions']
- if cppdefines:
- attrs['PreprocessorDefinitions'] = ';'.join(cppdefines)
- warnings = attrs['DisableSpecificWarnings']
- if warnings:
- warnings = SCons.Util.uniquer(warnings)
- attrs['DisableSpecificWarnings'] = ';'.join(warnings)
-
- return tool
-
- def VCLibrarianTool(self, args):
- default_attrs = {
- 'LinkTimeCodeGeneration' : "false",
- 'SuppressStartupBanner' : "false",
- }
-
- tool = MSVSTool('VCLibrarianTool', **default_attrs)
- attrs = tool.attrs
-
- for arg in args:
- if arg.startswith('/OUT'):
- continue
- if arg == '/ltcg':
- attrs['LinkTimeCodeGeneration'] = "true"
- elif arg == '/nologo':
- attrs['SuppressStartupBanner'] = "true"
-
- return tool
-
- def VCLinkerTool(self, args):
- default_attrs = {
- 'LinkIncremental' : "false",
- 'LinkTimeCodeGeneration' : "false",
- 'EnableCOMDATFolding' : TODO,
- 'OptimizeForWindows98' : TODO,
- 'OptimizeReferences' : TODO,
- 'Profile' : "false",
- 'SuppressStartupBanner' : "false",
- }
-
- tool = MSVSTool('VCLinkerTool', **default_attrs)
- attrs = tool.attrs
-
- for arg in args:
- if arg == '':
- continue
- if arg == '/INCREMENTAL':
- attrs['LinkIncremental'] = "true"
- elif arg == '/INCREMENTAL:NO':
- attrs['LinkIncremental'] = "false"
- elif arg == '/LTCG':
- attrs['LinkTimeCodeGeneration'] = "true"
- elif arg == '/nologo':
- attrs['SuppressStartupBanner'] = "true"
- elif arg == '/OPT:NOICF':
- attrs['EnableCOMDATFolding'] = 2 #
- elif arg == '/OPT:NOWIN98':
- attrs['OptimizeForWindows98'] = 1 #
- elif arg == '/OPT:REF':
- attrs['OptimizeReferences'] = 2 #
- elif arg == '/PROFILE':
- attrs['Profile'] = "true"
-
- return tool
-
- command_to_tool_map = {
- 'cl' : 'VCCLCompilerTool',
- 'cl.exe' : 'VCCLCompilerTool',
- 'lib' : 'VCLibrarianTool',
- 'lib.exe' : 'VCLibrarianTool',
- 'link' : 'VCLinkerTool',
- 'link.exe' : 'VCLinkerTool',
- }
-
- def cl_to_tool(self, args):
- command = os.path.basename(args[0])
- method_name = self.command_to_tool_map.get(command)
- if not method_name:
- return None
- return getattr(self, method_name)(args[1:])
-
- def _AddFileConfigurationDifferences(self, target, source, base_env, file_env, name):
- """Adds a per-file configuration.
-
- Args:
- target: The target being built from the source.
- source: The source to which the file configuration is being added.
- base_env: The base construction environment for the project.
- Differences from this will go into the FileConfiguration
- in the project file.
- file_env: The construction environment for the target, containing
- the per-target settings.
- """
- executor = target.get_executor()
- base_cl = map(str, base_env.subst_list(executor)[0])
- file_cl = map(str, file_env.subst_list(executor)[0])
- if base_cl == file_cl:
- return
-
- base_tool = self.cl_to_tool(base_cl)
- file_tool = self.cl_to_tool(file_cl)
-
- if not base_tool or not_file_tool:
- return
-
- file_tool.diff(base_tool)
-
- self.AddFileConfig(source, name, tools=[file_tool])
-
- def _AddFileConfigurations(self, env):
- """Adds per-file configurations for the buildtarget's sources.
-
- Args:
- env: The base construction environment for the project.
- """
- if not self.buildtargets:
- return
-
- for bt in self.buildtargets:
- executor = bt.get_executor()
- build_env = bt.get_build_env()
- bt_cl = map(str, build_env.subst_list(executor)[0])
- tool = self.cl_to_tool(bt_cl)
- default_tool = self.cl_to_tool([bt_cl[0]])
- if default_tool:
- tool.diff(default_tool)
- else:
- # TODO(sgk): print a message unconditionally is too
- # verbose for things like Python function actions,
- # but doing nothing runs the risk of burying problems.
- # Work out a better solution.
- #print "no tool for %r" % bt_cl[0]
- pass
- for t in bt.sources:
- e = t.get_build_env()
- additional_files = SCons.Util.UniqueList()
- for s in t.sources:
- s = env.arg2nodes([s])[0].srcnode()
- if not self.FindFile(s):
- additional_files.append(s)
- if not build_env is e:
- # TODO(sgk): This test may be bogus, but it works for now.
- # We're trying to figure out if the file configuration
- # differences need to be added one per build target, or one
- # per configuration for the entire project. The assumption
- # is that if the number of buildtargets configured matches
- # the number of project configurations, that we use those
- # in preference to the project configurations.
- if len(self.buildtargets) == len(self.configurations):
- self._AddFileConfigurationDifferences(t, s, build_env, e, e.subst('$MSVSCONFIGURATIONNAME'))
- else:
- for config in self.configurations:
- self._AddFileConfigurationDifferences(t, s, build_env, e, config.Name)
- self._MergeFiles(self.files, additional_files)
-
- def Write(self, env):
- """Writes the project file."""
- for flist in self.file_lists:
- self._FilesToSourceFiles(flist)
- self._MergeFiles(self.files, flist)
- for k, v in self.file_configurations.items():
- self.file_configurations[str(k)] = v
- k = self.env.File(k).srcnode()
- self.file_configurations[k] = v
- self.file_configurations[str(k)] = v
- self._AddFileConfigurations(env)
-
- self.Create()
-
- f = open(str(self), 'wt')
- f.write(self.formatMSVSProjectXML(self.doc))
- f.close()
-
- # Methods for formatting XML as nearly identically to Microsoft's
- # .vcproj output as we can practically make it.
- #
- # The general design here is copied from:
- #
- # Bruce Eckels' MindView, Inc: 12-09-04 XML Oddyssey
- # http://www.mindview.net/WebLog/log-0068
- #
- # Eckels' implementation broke up long tag definitions for readability,
- # in much the same way that .vcproj does, but we've modified things
- # for .vcproj quirks (like some tags *always* terminating with </Tag>,
- # even when empty).
-
- encoding = 'Windows-1252'
-
- def formatMSVSProjectXML(self, xmldoc):
- xmldoc = xmldoc.toprettyxml("", "\n", encoding=self.encoding)
- # Remove trailing whitespace from each line:
- xmldoc = "\n".join(
- [line.rstrip() for line in xmldoc.split("\n")])
- # Remove all empty lines before opening '<':
- while xmldoc.find("\n\n<") != -1:
- xmldoc = xmldoc.replace("\n\n<", "\n<")
- dom = xml.dom.minidom.parseString(xmldoc)
- xmldoc = dom.toprettyxml("\t", "", encoding=self.encoding)
- xmldoc = xmldoc.replace('?><', '?>\n<')
- xmldoc = self.reformatLines(xmldoc)
- return xmldoc
-
- def reformatLines(self, xmldoc):
- result = []
- for line in [line.rstrip() for line in xmldoc.split("\n")]:
- if line.lstrip().startswith("<"):
- result.append(self.reformatLine(line) + "\n")
- else:
- result.append(line + "\n")
- return ''.join(result)
-
- # Keyword order for specific tags.
- #
- # Listed keywords will come first and in the specified order.
- # Any unlisted keywords come after, in whatever order they appear
- # in the input config. In theory this means we would only *have* to
- # list the keywords that we care about, but in practice we'll probably
- # want to nail down Visual Studio's order to make sure we match them
- # as nearly as possible.
-
- order = {
- 'Configuration' : [
- 'Name',
- 'ConfigurationType',
- 'InheritedPropertySheets',
- ],
- 'FileConfiguration' : [
- 'Name',
- 'ExcludedFromBuild',
- ],
- 'Tool' : [
- 'Name',
- 'DisableSpecificWarnings',
-
- 'AdditionalIncludeDirectories',
- 'Description',
- 'CommandLine',
- 'OutputFile',
- 'ImportLibrary',
- 'PreprocessorDefinitions',
- 'UsePrecompiledHeader',
- 'PrecompiledHeaderThrough',
- 'ForcedIncludeFiles',
- ],
- 'VisualStudioProject' : [
- 'ProjectType',
- 'Version',
- 'Name',
- 'ProjectGUID',
- 'RootNamespace',
- 'Keyword',
- ],
- }
-
- force_closing_tag = [
- 'File',
- 'Globals',
- 'References',
- 'ToolFiles'
- ]
-
- oneLiner = re.compile("(\s*)<(\w+)(.*)>")
- keyValuePair = re.compile('\w+="[^"]*?"')
- def reformatLine(self, line):
- """Reformat an xml tag to put each key-value
- element on a single indented line, for readability"""
- matchobj = self.oneLiner.match(line.rstrip())
- if not matchobj:
- return line
- baseIndent, tag, rest = matchobj.groups()
- slash = ''
- if rest[-1:] == '/':
- slash = '/'
- rest = rest[:-1]
- result = [baseIndent + '<' + tag]
- indent = baseIndent + "\t"
- pairs = self.keyValuePair.findall(rest)
- for key in self.order.get(tag, []):
- for p in [ p for p in pairs if p.startswith(key+'=') ]:
- result.append("\n" + indent + p)
- pairs.remove(p)
- for pair in pairs:
- result.append("\n" + indent + pair)
- result = [''.join(result).rstrip()]
-
- if tag in self.force_closing_tag:
- # These force termination with </Tag>, so translate slash.
- if rest:
- result.append("\n")
- result.append(indent)
- result.append(">")
- if slash:
- result.append("\n")
- result.append(baseIndent + "</" + tag + ">")
- else:
- if rest:
- result.append("\n")
- if slash:
- result.append(baseIndent)
- else:
- result.append(indent)
- result.append(slash + ">")
-
- return ''.join(result)
-
-def MSVSProject(env, item, *args, **kw):
- if not SCons.Util.is_String(item):
- return item
- item = env.subst(item)
- result = env.fs._lookup(item, None, _MSVSProject, create=1)
- result.initialize(env, item, *args, **kw)
- LookupAdd(item, result)
- return result
-
-#------------------------------------------------------------------------------
-
-MSVSSolutionAction = SCons.Script.Action(MSVSAction,
- "Generating Visual Studio solution `$TARGET' ...")
-
-class _MSVSSolution(SCons.Node.FS.File):
- """Visual Studio solution."""
-
- def initialize(self, env, path, entries=None, variants=None, websiteProperties=True):
- """Initializes the solution.
-
- Args:
- path: Path to solution file.
- entries: List of entries in solution. May contain Folder or Project
- objects. May be None, if the folder is empty.
- variants: List of build variant strings. If none, a default list will
- be used.
- """
- self.msvs_path = path
- self.websiteProperties = websiteProperties
-
- # Copy passed lists (or set to empty lists)
- self.entries = list(entries or [])
-
- if variants:
- # Copy passed list
- self.variants = variants[:]
- else:
- # Use default
- self.variants = ['Debug|Win32', 'Release|Win32']
- # TODO(rspangler): Need to be able to handle a mapping of solution config
- # to project config. Should we be able to handle variants being a dict,
- # or add a separate variant_map variable? If it's a dict, we can't
- # guarantee the order of variants since dict keys aren't ordered.
-
- env.Command(self, [], MSVSSolutionAction)
-
- def Write(self, env):
- """Writes the solution file to disk.
-
- Raises:
- IndexError: An entry appears multiple times.
- """
- r = []
- errors = []
-
- def lookup_subst(item, env=env, errors=errors):
- if SCons.Util.is_String(item):
- lookup_item = env.subst(item)
- else:
- lookup_item = item
- try:
- return Lookup(lookup_item)
- except SCons.Errors.UserError:
- raise LookupError(item, lookup_item)
-
- # Walk the entry tree and collect all the folders and projects.
- all_entries = []
- entries_to_check = self.entries[:]
- while entries_to_check:
- # Pop from the beginning of the list to preserve the user's order.
- entry = entries_to_check.pop(0)
- try:
- entry = lookup_subst(entry)
- except LookupError, e:
- errors.append("Could not look up entry `%s'." % e)
- continue
-
- # A project or folder can only appear once in the solution's folder tree.
- # This also protects from cycles.
- if entry in all_entries:
- #raise IndexError('Entry "%s" appears more than once in solution' %
- # e.name)
- continue
-
- all_entries.append(entry)
-
- # If this is a folder, check its entries too.
- if isinstance(entry, _MSVSFolder):
- entries_to_check += entry.entries
-
- # Header
- r.append('Microsoft Visual Studio Solution File, Format Version 9.00\n')
- r.append('# Visual Studio 2005\n')
-
- # Project entries
- for e in all_entries:
- r.append('Project("%s") = "%s", "%s", "%s"\n' % (
- e.entry_type_guid, # Entry type GUID
- e.msvs_name, # Folder name
- e.get_msvs_path(self), # Folder name (again)
- e.get_guid(), # Entry GUID
- ))
-
- # TODO(rspangler): Need a way to configure this stuff
- if self.websiteProperties:
- r.append('\tProjectSection(WebsiteProperties) = preProject\n'
- '\t\tDebug.AspNetCompiler.Debug = "True"\n'
- '\t\tRelease.AspNetCompiler.Debug = "False"\n'
- '\tEndProjectSection\n')
-
- if isinstance(e, _MSVSFolder):
- if e.items:
- r.append('\tProjectSection(SolutionItems) = preProject\n')
- for i in e.items:
- i = i.replace('/', '\\')
- r.append('\t\t%s = %s\n' % (i, i))
- r.append('\tEndProjectSection\n')
-
- if isinstance(e, _MSVSProject):
- if e.dependencies:
- r.append('\tProjectSection(ProjectDependencies) = postProject\n')
- for d in e.dependencies:
- try:
- d = lookup_subst(d)
- except LookupError, e:
- errors.append("Could not look up dependency `%s'." % e)
- else:
- r.append('\t\t%s = %s\n' % (d.get_guid(), d.get_guid()))
- r.append('\tEndProjectSection\n')
-
- r.append('EndProject\n')
-
- # Global section
- r.append('Global\n')
-
- # Configurations (variants)
- r.append('\tGlobalSection(SolutionConfigurationPlatforms) = preSolution\n')
- for v in self.variants:
- r.append('\t\t%s = %s\n' % (v, v))
- r.append('\tEndGlobalSection\n')
-
- # Sort config guids for easier diffing of solution changes.
- config_guids = []
- for e in all_entries:
- if isinstance(e, _MSVSProject):
- config_guids.append(e.get_guid())
- config_guids.sort()
-
- r.append('\tGlobalSection(ProjectConfigurationPlatforms) = postSolution\n')
- for g in config_guids:
- for v in self.variants:
- r.append('\t\t%s.%s.ActiveCfg = %s\n' % (
- g, # Project GUID
- v, # Solution build configuration
- v, # Project build config for that solution config
- ))
-
- # Enable project in this solution configuratation
- r.append('\t\t%s.%s.Build.0 = %s\n' % (
- g, # Project GUID
- v, # Solution build configuration
- v, # Project build config for that solution config
- ))
- r.append('\tEndGlobalSection\n')
-
- # TODO(rspangler): Should be able to configure this stuff too (though I've
- # never seen this be any different)
- r.append('\tGlobalSection(SolutionProperties) = preSolution\n')
- r.append('\t\tHideSolutionNode = FALSE\n')
- r.append('\tEndGlobalSection\n')
-
- # Folder mappings
- # TODO(rspangler): Should omit this section if there are no folders
- folder_mappings = []
- for e in all_entries:
- if not isinstance(e, _MSVSFolder):
- continue # Does not apply to projects, only folders
- for subentry in e.entries:
- try:
- subentry = lookup_subst(subentry)
- except LookupError, e:
- errors.append("Could not look up subentry `%s'." % subentry)
- else:
- folder_mappings.append((subentry.get_guid(), e.get_guid()))
- folder_mappings.sort()
- r.append('\tGlobalSection(NestedProjects) = preSolution\n')
- for fm in folder_mappings:
- r.append('\t\t%s = %s\n' % fm)
- r.append('\tEndGlobalSection\n')
-
- r.append('EndGlobal\n')
-
- if errors:
- errors = ['Errors while generating solution file:'] + errors
- raise SCons.Errors.UserError, '\n\t'.join(errors)
-
- f = open(self.path, 'wt')
- f.write(''.join(r))
- f.close()
-
-def MSVSSolution(env, item, *args, **kw):
- if not SCons.Util.is_String(item):
- return item
- item = env.subst(item)
- result = env.fs._lookup(item, None, _MSVSSolution, create=1)
- result.initialize(env, item, *args, **kw)
- LookupAdd(item, result)
- return result
-
-class Derived(SCons.Util.Proxy):
- def srcnode(self, *args, **kw):
- return self
- def __getattr__(self, name):
- if name == 'sources':
- return []
- return SCons.Util.Proxy.__getattr__(self, name)
- def __hash__(self, *args, **kw):
- return id(self)
-
-import __builtin__
-
-__builtin__.Derived = Derived
-__builtin__.MSVSConfig = MSVSConfig
-__builtin__.MSVSFilter = MSVSFilter
-__builtin__.MSVSProject = MSVSProject
-__builtin__.MSVSSolution = MSVSSolution
-__builtin__.MSVSTool = MSVSTool
« no previous file with comments | « site_scons/site_tools/MSVSNew.py ('k') | site_scons/site_tools/atlmfc_vc80.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698