diff options
Diffstat (limited to 'site_scons/site_tools/component_targets_msvs.py')
-rwxr-xr-x | site_scons/site_tools/component_targets_msvs.py | 663 |
1 files changed, 596 insertions, 67 deletions
diff --git a/site_scons/site_tools/component_targets_msvs.py b/site_scons/site_tools/component_targets_msvs.py index e86de89..b50f1a5 100755 --- a/site_scons/site_tools/component_targets_msvs.py +++ b/site_scons/site_tools/component_targets_msvs.py @@ -30,7 +30,9 @@ """Visual Studio solution output of component targets for SCons.""" +import copy import md5 +import os import sys import xml.dom import xml.dom.minidom @@ -65,7 +67,7 @@ def MakeGuid(name, seed='component_targets_msvs'): #------------------------------------------------------------------------------ -def GetGuidFromVSProject(project_path): +def GetGuidAndNameFromVSProject(project_path): """Reads the GUID from a Visual Studio project file. Args: @@ -73,6 +75,7 @@ def GetGuidFromVSProject(project_path): Returns: The GUID string from the file. + The project name from the file. """ doc = xml.dom.minidom.parse(project_path) try: @@ -80,7 +83,10 @@ def GetGuidFromVSProject(project_path): if n_root.nodeName != 'VisualStudioProject': raise SCons.Errors.UserError('%s is not a Visual Studio project.' % project_path) - return str(n_root.attributes['ProjectGUID'].nodeValue) + return ( + str(n_root.attributes['ProjectGUID'].nodeValue), + str(n_root.attributes['Name'].nodeValue), + ) finally: # Clean up doc doc.unlink() @@ -88,6 +94,146 @@ def GetGuidFromVSProject(project_path): #------------------------------------------------------------------------------ +class VSProjectWriter(object): + """Visual Studio XML project writer.""" + + def __init__(self, project_path): + """Initializes the project. + + Args: + project_path: Path to the project file. + """ + self.project_path = project_path + self.doc = None + + def Create(self, name): + """Creates the project document. + + Args: + name: Name of the project. + """ + self.name = name + self.guid = MakeGuid(name) + + # Create XML doc + xml_impl = xml.dom.getDOMImplementation() + self.doc = xml_impl.createDocument(None, 'VisualStudioProject', None) + + # Add attributes to root element + self.n_root = self.doc.documentElement + self.n_root.setAttribute('ProjectType', 'Visual C++') + self.n_root.setAttribute('Version', '8.00') + self.n_root.setAttribute('Name', self.name) + self.n_root.setAttribute('ProjectGUID', self.guid) + self.n_root.setAttribute('RootNamespace', self.name) + self.n_root.setAttribute('Keyword', 'MakeFileProj') + + # Add platform list + n_platform = self.doc.createElement('Platforms') + self.n_root.appendChild(n_platform) + n = self.doc.createElement('Platform') + n.setAttribute('Name', 'Win32') + n_platform.appendChild(n) + + # Add empty ToolFiles section + self.n_root.appendChild(self.doc.createElement('ToolFiles')) + + # Add configurations section + self.n_configs = self.doc.createElement('Configurations') + self.n_root.appendChild(self.n_configs) + + # Add files section + self.n_files = self.doc.createElement('Files') + self.n_root.appendChild(self.n_files) + + # Add empty Globals section + self.n_root.appendChild(self.doc.createElement('Globals')) + + def AddConfig(self, name, attrs, tool_attrs): + """Adds a configuration to the project. + + Args: + name: Configuration name. + attrs: Dict of configuration attributes. + tool_attrs: Dict of tool attributes. + """ + # Add configuration node + n_config = self.doc.createElement('Configuration') + n_config.setAttribute('Name', '%s|Win32' % name) + n_config.setAttribute('ConfigurationType', '0') + for k, v in attrs.items(): + n_config.setAttribute(k, v) + self.n_configs.appendChild(n_config) + + # Add tool node + n_tool = self.doc.createElement('Tool') + n_tool.setAttribute('Name', 'VCNMakeTool') + n_tool.setAttribute('IncludeSearchPath', '') + n_tool.setAttribute('ForcedIncludes', '') + n_tool.setAttribute('AssemblySearchPath', '') + n_tool.setAttribute('ForcedUsingAssemblies', '') + n_tool.setAttribute('CompileAsManaged', '') + n_tool.setAttribute('PreprocessorDefinitions', '') + for k, v in tool_attrs.items(): + n_tool.setAttribute(k, v) + n_config.appendChild(n_tool) + + def _WalkFolders(self, folder_dict, parent): + """Recursively walks the folder tree. + + Args: + folder_dict: Dict of folder entries. Entry is + either subdir_name:subdir_dict or relative_path_to_file:None. + parent: Parent node (folder node for that dict) + """ + entries = folder_dict.keys() + entries.sort() + for e in entries: + if folder_dict[e]: + # Folder + n_subfolder = self.doc.createElement('Filter') + n_subfolder.setAttribute('Name', e) + parent.appendChild(n_subfolder) + self._WalkFolders(folder_dict[e], n_subfolder) + else: + # File + n_file = self.doc.createElement('File') + n_file.setAttribute('RelativePath', e) + parent.appendChild(n_file) + + def AddFiles(self, name, files_dict): + """Adds files to the project. + + Args: + name: Name of the folder. If None, files/folders will be added directly + to the files list. + files_dict: A dict of files / folders. + + Within the files_dict: + * A file entry is relative_path:None + * A folder entry is folder_name:files_dict, where files_dict is another + dict of this form. + """ + # Create subfolder if necessary + if name: + n_folder = self.doc.createElement('Filter') + n_folder.setAttribute('Name', name) + self.n_files.appendChild(n_folder) + else: + n_folder = self.n_files + + # Recursively add files to the folder + self._WalkFolders(files_dict, n_folder) + + def Write(self): + """Writes the project file.""" + f = open(self.project_path, 'wt') + self.doc.writexml(f, encoding='Windows-1252', addindent=' ', newl='\n') + f.close() + +#------------------------------------------------------------------------------ + + def ComponentVSProjectBuilder(target, source, env): """Visual Studio project builder. @@ -109,70 +255,34 @@ def ComponentVSProjectBuilder(target, source, env): env_hammer_bat = env.Clone(VS_PROJECT_TO_MAIN_DIR=project_to_main) hammer_bat = env_hammer_bat.subst('$COMPONENT_VS_PROJECT_SCRIPT_PATH', raw=1) - # Project header - xml_impl = xml.dom.getDOMImplementation() - doc = xml_impl.createDocument(None, 'VisualStudioProject', None) - - n_root = doc.documentElement - n_root.setAttribute('ProjectType', 'Visual C++') - n_root.setAttribute('Version', '8.00') - n_root.setAttribute('Name', target_name) - n_root.setAttribute('ProjectGUID', MakeGuid(target_name)) - n_root.setAttribute('RootNamespace', target_name) - n_root.setAttribute('Keyword', 'MakeFileProj') - - n_platform = doc.createElement('Platforms') - n_root.appendChild(n_platform) - n = doc.createElement('Platform') - n.setAttribute('Name', 'Win32') - n_platform.appendChild(n) - - n_root.appendChild(doc.createElement('ToolFiles')) - - # One configuration per build mode supported by this target - n_configs = doc.createElement('Configurations') - n_root.appendChild(n_configs) + vsp = VSProjectWriter(project_file) + vsp.Create(target_name) + # Add configuration per build mode supported by this target target_path = env['TARGET_PATH'] for mode, path in target_path.items(): - n_config = doc.createElement('Configuration') - n_config.setAttribute('Name', '%s|Win32' % mode) - n_config.setAttribute('OutputDirectory', - '$(ProjectDir)/%s/%s/out' % (mode, target_name)) - n_config.setAttribute('IntermediateDirectory', - '$(ProjectDir)/%s/%s/tmp' % (mode, target_name)) - n_config.setAttribute('ConfigurationType', '0') - n_configs.appendChild(n_config) + attrs = {} + attrs['OutputDirectory'] = '$(ProjectDir)/%s/%s/out' % (mode, target_name) + attrs['IntermediateDirectory'] = ('$(ProjectDir)/%s/%s/tmp' % + (mode, target_name)) - n_tool = doc.createElement('Tool') - n_tool.setAttribute('Name', 'VCNMakeTool') - n_tool.setAttribute('IncludeSearchPath', '') - n_tool.setAttribute('ForcedIncludes', '') - n_tool.setAttribute('AssemblySearchPath', '') - n_tool.setAttribute('ForcedUsingAssemblies', '') - n_tool.setAttribute('CompileAsManaged', '') - n_tool.setAttribute('PreprocessorDefinitions', '') + tool_attrs = {} if path: - n_tool.setAttribute( - 'Output', env.RelativePath(target[0].dir, env.Entry(path), sep='/')) + tool_attrs['Output'] = env.RelativePath(target[0].dir, + env.Entry(path), sep='/') build_cmd = '%s --mode=%s %s' % (hammer_bat, mode, target_name) clean_cmd = '%s --mode=%s -c %s' % (hammer_bat, mode, target_name) - n_tool.setAttribute('BuildCommandLine', build_cmd) - n_tool.setAttribute('CleanCommandLine', clean_cmd) - n_tool.setAttribute('ReBuildCommandLine', clean_cmd + ' && ' + build_cmd) - n_config.appendChild(n_tool) + tool_attrs['BuildCommandLine'] = build_cmd + tool_attrs['CleanCommandLine'] = clean_cmd + tool_attrs['ReBuildCommandLine'] = clean_cmd + ' && ' + build_cmd + + vsp.AddConfig(mode, attrs, tool_attrs) - n_files = doc.createElement('Files') - n_root.appendChild(n_files) # TODO(rspangler): Fill in files - at least, the .scons file invoking the # target. - n_root.appendChild(doc.createElement('Globals')) - - f = open(project_file, 'wt') - doc.writexml(f, encoding='Windows-1252', addindent=' ', newl='\n') - f.close() - + # Write project + vsp.Write() return 0 @@ -225,6 +335,344 @@ def ComponentVSProject(self, target_name, **kwargs): #------------------------------------------------------------------------------ +class SourceWalker(object): + """Iterator for walking a node tree and collecting sources. + + This is depth-first, children are visited before the parent. The object + can be initialized with any node, and returns the next node on the descent + with each Next() call. + + This class does not get caught in node cycles caused, for example, by C + header file include loops. + + Based on SCons.Node.Walker. + """ + + def __init__(self, node, seen, print_interval = 1000): + """Initializes the source walker. + + Args: + node: Node to start walk from. + seen: Set to add seen nodes to, if not None. + print_interval: Interval in nodes examined at which to print a progress + indicator. + """ + self.interval = print_interval + # Put node on stack + self.stack = [node] + # Scan for node's children, then copy them to node.wkids + node.wkids = copy.copy(node.children(scan=1)) + # Keep track of nodes we've seen (returned) + if seen is None: + seen = set() + self.seen = seen + # Add node to history for cycle detection + self.history = set() + self.history.add(node) + # We've seen one node now + self.nodes_examined = 1 + self.unique_nodes = 1 + + + def Next(self): + """Get the next node for this walk of the tree. + + Returns: + The next node, or None if no more nodes. + + This function is intentionally iterative, not recursive, to sidestep any + issues of stack size limitations. + """ + while self.stack: + if self.stack[-1].wkids: + # Node has children we haven't examined, so iterate into the first + # child + node = self.stack[-1].wkids.pop(0) + if not self.stack[-1].wkids: + # No more children of this node + self.stack[-1].wkids = None + self.nodes_examined += 1 + if self.interval and not self.nodes_examined % self.interval: + self.PrintProgress() + if (node not in self.history) and (node not in self.seen): + # Haven't hit a cycle or a node we've already seen + node.wkids = copy.copy(node.children(scan=1)) + self.stack.append(node) + self.history.add(node) + else: + # Coming back from iterating, so return the next node on the stack. + node = self.stack.pop() + self.history.remove(node) + self.seen.add(node) + self.unique_nodes += 1 + return node + return None + + def PrintProgress(self): + """Prints a progress indicator.""" + print ' Examined %d nodes, found %d unique...' % ( + self.nodes_examined, self.unique_nodes) + + def WalkAll(self): + """Walks all nodes in the the tree.""" + while self.Next(): + pass + if self.interval: + self.PrintProgress() + + +def ComponentVSSourceProjectBuilder(target, source, env): + """Visual Studio source project builder. + + Args: + target: Destination file. + source: List of sources to be added to the target. + env: Environment context. + + Returns: + Zero if successful. + """ + source = source # Silence gpylint + + target_name = env['PROJECT_NAME'] + project_file = target[0].path + project_dir = target[0].dir + + # Get list of suffixes to include + suffixes = env.SubstList2('$COMPONENT_VS_SOURCE_SUFFIXES') + + # Convert source folders to absolute paths + folders = [] + for f in env['COMPONENT_VS_SOURCE_FOLDERS']: + # (folder name, folder abspath, dict of contents) + folders.append((f[0], env.Dir(f[1]).abspath, {})) + + # TODO(rspangler): Additional enhancements: + # * Should be able to specify paths in folder name (i.e., foo/bar) and + # create the nested folder nodes ('foo' and 'bar') + # * Should be tolerant of a folder being specified more than once with + # the same name (probably necessary to support nested folder nodes anyway) + # Can probably accomplish both of those by creating a parent fodler dict and + # calling WalkFolders() only once. + # Create a temporary solution alias to point to all the targets, so we can + # make a single call to SourceWalker() + tmp_alias = env.Alias('vs_source_project_' + target_name, + map(env.Alias, env['COMPONENT_VS_SOURCE_TARGETS'])) + + # Scan all targets and add unique nodes to set of sources + print ' Scanning dependency tree ...' + all_srcs = set() + walker = SourceWalker(tmp_alias[0], all_srcs) + walker.WalkAll() + + # Walk all sources and build directory trees + print ' Building source tree...' + for n in all_srcs: + if not hasattr(n, 'isfile') or not n.isfile(): + continue # Not a file + if n.has_builder(): + continue # Not a leaf node + if n.suffix not in suffixes: + continue # Not a file type we include + + path = n.rfile().abspath + for f in folders: + if path.startswith(f[1]): + if f[0] is None: + # Folder name of None is a filter + break + relpath = path[len(f[1]) + 1:].split(os.sep) + folder_dict = f[2] + # Recursively add subdirs + for pathseg in relpath[:-1]: + if pathseg not in folder_dict: + folder_dict[pathseg] = {} + folder_dict = folder_dict[pathseg] + # Add file to last subdir. No dict, since this isn't a subdir + folder_dict[env.RelativePath(project_dir, path)] = None + break + + print ' Writing project file...' + + vsp = VSProjectWriter(project_file) + vsp.Create(target_name) + + # One configuration for all build modes + vsp.AddConfig('all', {}, {}) + + # Add files + for f in folders: + if f[0] is None: + continue # Skip filters + vsp.AddFiles(f[0], f[2]) + + vsp.Write() + return 0 + + +def ComponentVSSourceProject(self, project_name, target_names, **kwargs): + """Visual Studio source project pseudo-builder. + + Args: + self: Environment context. + project_name: Name of the project. + target_names: List of target names to include source for. + kwargs: Optional keyword arguments override environment variables in the + derived environment used to create the project. + + Returns: + A list of output nodes. + """ + # Builder only works on Windows + if sys.platform not in ('win32', 'cygwin'): + return [] + + # Clone environment and add keyword args + env = self.Clone() + for k, v in kwargs.items(): + env[k] = v + + # Save the project name and targets + env['PROJECT_NAME'] = project_name + env['COMPONENT_VS_SOURCE_TARGETS'] = target_names + + # Call the real builder + return env.ComponentVSSourceProjectBuilder( + '$COMPONENT_VS_PROJECT_DIR/${PROJECT_NAME}', []) + +#------------------------------------------------------------------------------ + + +def FindSources(env, dest, source, suffixes=None): + """Recursively finds sources and adds them to the destination set. + + Args: + env: Environment context. + dest: Set to add source nodes to. + source: Source file(s) to find. May be a string, Node, or a list of + mixed strings or Nodes. Strings will be passed through env.Glob() to + evaluate wildcards. If a source evaluates to a directory, the entire + directory will be recursively added. + suffixes: List of suffixes to include. If not None, only files with these + suffixes will be added to dest. + """ + for source_entry in env.Flatten(source): + if type(source_entry) == str: + # Search for matches for each source entry + source_nodes = env.Glob(source_entry) + else: + # Source entry is already a file or directory node; no need to glob it + source_nodes = [source_entry] + for s in source_nodes: + if str(s.__class__) == 'SCons.Node.FS.Dir': + # Recursively search subdir. Since glob('*') doesn't match dot files, + # also glob('.*'). + FindSources(env, dest, [s.abspath + '/*', s.abspath + '/.*'], suffixes) + elif suffixes and s.suffix in suffixes: + dest.add(s) + + +def ComponentVSDirProjectBuilder(target, source, env): + """Visual Studio directory project builder. + + Args: + target: Destination file. + source: List of sources to be added to the target. + env: Environment context. + + Returns: + Zero if successful. + """ + source = source # Silence gpylint + + target_name = env['PROJECT_NAME'] + project_file = target[0].path + project_dir = target[0].dir + + # Convert source folders to absolute paths + folders = [] + for f in env['COMPONENT_VS_SOURCE_FOLDERS']: + # (folder name, folder abspath, dict of contents) + folders.append((f[0], env.Dir(f[1]).abspath, {})) + + # Recursively scan source directories + print ' Scanning directories for source...' + all_srcs = set() + FindSources(env, all_srcs, env['PROJECT_SOURCES'], + suffixes=env.SubstList2('$COMPONENT_VS_SOURCE_SUFFIXES')) + + # Walk all sources and build directory trees + print ' Building source tree...' + for n in all_srcs: + # Map addRepository'd source to its real location. + path = n.rfile().abspath + for f in folders: + if path.startswith(f[1]): + if f[0] is None: + # Folder name of None is a filter + break + relpath = path[len(f[1]) + 1:].split(os.sep) + folder_dict = f[2] + # Recursively add subdirs + for pathseg in relpath[:-1]: + if pathseg not in folder_dict: + folder_dict[pathseg] = {} + folder_dict = folder_dict[pathseg] + # Add file to last subdir. No dict, since this isn't a subdir + folder_dict[env.RelativePath(project_dir, path)] = None + break + + print ' Writing project file...' + + vsp = VSProjectWriter(project_file) + vsp.Create(target_name) + + # One configuration for all build modes + vsp.AddConfig('all', {}, {}) + + # Add files + for f in folders: + if f[0] is None: + continue # Skip filters + vsp.AddFiles(f[0], f[2]) + + vsp.Write() + return 0 + + +def ComponentVSDirProject(self, project_name, source, **kwargs): + """Visual Studio directory project pseudo-builder. + + Args: + self: Environment context. + project_name: Name of the project. + source: List of source files and/or directories. + kwargs: Optional keyword arguments override environment variables in the + derived environment used to create the project. + + Returns: + A list of output nodes. + """ + # Builder only works on Windows + if sys.platform not in ('win32', 'cygwin'): + return [] + + # Clone environment and add keyword args + env = self.Clone() + for k, v in kwargs.items(): + env[k] = v + + # Save the project name and sources + env['PROJECT_NAME'] = project_name + env['PROJECT_SOURCES'] = source + + # Call the real builder + return env.ComponentVSDirProjectBuilder( + '$COMPONENT_VS_PROJECT_DIR/${PROJECT_NAME}', []) + +#------------------------------------------------------------------------------ + + def ComponentVSSolutionBuilder(target, source, env): """Visual Studio solution builder. @@ -242,8 +690,13 @@ def ComponentVSSolutionBuilder(target, source, env): projects = env['SOLUTION_PROJECTS'] folders = env['SOLUTION_FOLDERS'] - f = open(solution_file, 'wt') + # Scan externally-generated projects + external_projects = [] + for p in source: + guid, name = GetGuidAndNameFromVSProject(p.abspath) + external_projects.append((p, name, guid)) + f = open(solution_file, 'wt') f.write('Microsoft Visual Studio Solution File, Format Version 9.00\n') f.write('# Visual Studio 2005\n') @@ -259,17 +712,19 @@ def ComponentVSSolutionBuilder(target, source, env): )) # Projects generated elsewhere - for p in source: + for p, name, guid in external_projects: f.write('Project("%s") = "%s", "%s", "%s"\nEndProject\n' % ( # TODO(rspangler): What if this project isn't type external makefile? # How to tell what type it is? '{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}', # External makefile GUID - p, # Project name + name, # Project name env.RelativePath(target[0].dir, p), # Path to project file - GetGuidFromVSProject(p.abspath), # Project GUID + guid, # Project GUID )) # Folders from build groups + # TODO(rspangler): Currently no way to add external project (specified in + # sources) to a solution folder. for folder in folders: f.write('Project("%s") = "%s", "%s", "%s"\nEndProject\n' % ( '{2150E333-8FDC-42A3-9474-1A3956D46DE8}', # Solution folder GUID @@ -285,7 +740,19 @@ def ComponentVSSolutionBuilder(target, source, env): f.write('\t\t%s|Win32 = %s|Win32\n' % (mode, mode)) f.write('\tEndGlobalSection\n') + # Determine which projects should be enabled + # TODO(rspangler): This is somewhat clunky. DEFAULT_TARGETS is global, and + # what we really need is something mode-specific. In theory we could make + # this a mode-specific dict rather than a list, but that'd also be a pain to + # populate. + # These variable names are also getting REALLY long. Perhaps we should + # define shorter ones (with the default value redirecting to the longer + # ones for legacy compatibility). + enable_projects = env.SubstList2('$COMPONENT_VS_ENABLED_PROJECTS') + f.write('\tGlobalSection(ProjectConfigurationPlatforms) = postSolution\n') + + # Projects generated by ComponentVSSolution() for p in projects: for mode in GetTargetModes(): f.write('\t\t%s.%s|Win32.ActiveCfg = %s|Win32\n' % ( @@ -293,14 +760,43 @@ def ComponentVSSolutionBuilder(target, source, env): mode, # Solution build configuration mode, # Project build config for that solution config )) + t = GetTargets().get(p) - if t and mode in t.mode_properties: + + # Determine if project should be enabled in this mode + enabled = t and mode in t.mode_properties + if enable_projects and p not in enable_projects: + # Enable list specified, but target isn't in it + # TODO(rspangler): Since we env.Default(scons-out) elsewhere, this + # likely causes all projects to be disabled by default. But that's + # realistically better than enabling them all... + enabled = False + + if enabled: # Target can be built in this mode f.write('\t\t%s.%s|Win32.Build.0 = %s|Win32\n' % ( MakeGuid(p), # Project GUID mode, # Solution build configuration mode, # Project build config for that solution config )) + + # Projects generated elsewhere + for p, name, guid in external_projects: + for mode in GetTargetModes(): + f.write('\t\t%s.%s|Win32.ActiveCfg = %s|Win32\n' % ( + guid, # Project GUID + mode, # Solution build configuration + mode, # Project build config for that solution config + )) + + if name in enable_projects or not enable_projects: + # Build target in this mode + f.write('\t\t%s.%s|Win32.Build.0 = %s|Win32\n' % ( + guid, # Project GUID + mode, # Solution build configuration + mode, # Project build config for that solution config + )) + f.write('\tEndGlobalSection\n') f.write('\tGlobalSection(SolutionProperties) = preSolution\n') @@ -361,10 +857,13 @@ def ComponentVSSolution(self, solution_name, target_names, projects=None, target_names = env.SubstList2(target_names) env['SOLUTION_TARGETS'] = target_names - project_names = {} - folders = [] + # Save the default targets list as an environment variable + env['COMPONENT_VS_SCONS_DEFAULT_TARGETS'] = SCons.Script.DEFAULT_TARGETS + # Expand target_names into project names, and create project-to-folder # mappings + project_names = {} + folders = [] if target_names: # Expand target_names into project names for target in target_names: @@ -420,8 +919,10 @@ def generate(env): # Add pseudo-builders to set up the project and solution builders. These # need to be available on all platforms so that SConscripts which reference # them will work. + env.AddMethod(ComponentVSDirProject) env.AddMethod(ComponentVSProject) env.AddMethod(ComponentVSSolution) + env.AddMethod(ComponentVSSourceProject) # If not on Windows, don't do anything else if sys.platform not in ('win32', 'cygwin'): @@ -431,38 +932,66 @@ def generate(env): env.Tool('gather_inputs') env.SetDefault( - COMPONENT_VS_SOLUTION_DIR='$DESTINATION_ROOT/solution', COMPONENT_VS_PROJECT_DIR='$COMPONENT_VS_SOLUTION_DIR/projects', - COMPONENT_VS_SOLUTION_SUFFIX='.sln', - COMPONENT_VS_PROJECT_SUFFIX='.vcproj', COMPONENT_VS_PROJECT_SCRIPT_NAME = 'hammer.bat', COMPONENT_VS_PROJECT_SCRIPT_PATH = ( '$$(ProjectDir)/$VS_PROJECT_TO_MAIN_DIR/' '$COMPONENT_VS_PROJECT_SCRIPT_NAME'), + COMPONENT_VS_PROJECT_SUFFIX='.vcproj', + + COMPONENT_VS_SOLUTION_DIR='$DESTINATION_ROOT/solution', + COMPONENT_VS_SOLUTION_SUFFIX='.sln', + COMPONENT_VS_ENABLED_PROJECTS=['$COMPONENT_VS_SCONS_DEFAULT_TARGETS'], + + COMPONENT_VS_SOURCE_SUFFIXES=['$CPPSUFFIXES', '.rc', '.scons'], + COMPONENT_VS_SOURCE_FOLDERS=[('source', '$MAIN_DIR')], ) AddTargetGroup('all_solutions', 'solutions can be built') # Add builders vcprojaction = SCons.Script.Action(ComponentVSProjectBuilder, varlist=[ + 'COMPONENT_VS_PROJECT_SCRIPT_PATH', 'TARGET_NAME', 'TARGET_PATH', - 'COMPONENT_VS_PROJECT_SCRIPT_PATH', ]) vcprojbuilder = SCons.Script.Builder( action=vcprojaction, suffix='$COMPONENT_VS_PROJECT_SUFFIX') + source_vcproj_action = SCons.Script.Action( + ComponentVSSourceProjectBuilder, varlist=[ + 'COMPONENT_VS_SOURCE_FOLDERS', + 'COMPONENT_VS_SOURCE_SUFFIXES', + 'COMPONENT_VS_SOURCE_TARGETS', + ]) + source_vcproj_builder = SCons.Script.Builder( + action=source_vcproj_action, + suffix='$COMPONENT_VS_PROJECT_SUFFIX') + + dir_vcproj_action = SCons.Script.Action( + ComponentVSDirProjectBuilder, varlist=[ + 'COMPONENT_VS_SOURCE_FOLDERS', + 'COMPONENT_VS_SOURCE_SUFFIXES', + 'PROJECT_SOURCES', + ]) + dir_vcproj_builder = SCons.Script.Builder( + action=dir_vcproj_action, + suffix='$COMPONENT_VS_PROJECT_SUFFIX') + slnaction = SCons.Script.Action(ComponentVSSolutionBuilder, varlist=[ - 'SOLUTION_TARGETS', + 'COMPONENT_VS_ENABLED_PROJECTS', 'SOLUTION_FOLDERS', 'SOLUTION_PROJECTS', + 'SOLUTION_TARGETS', ]) slnbuilder = SCons.Script.Builder( action=slnaction, suffix='$COMPONENT_VS_SOLUTION_SUFFIX') env.Append(BUILDERS={ + 'ComponentVSDirProjectBuilder': dir_vcproj_builder, 'ComponentVSProjectBuilder': vcprojbuilder, 'ComponentVSSolutionBuilder': slnbuilder, + 'ComponentVSSourceProjectBuilder': source_vcproj_builder, }) |