|  | # Copyright 2016 The Chromium Authors. All rights reserved. | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  |  | 
|  | from __future__ import print_function | 
|  |  | 
|  | import os | 
|  | import glob | 
|  | import re | 
|  | import sys | 
|  | from shutil import copyfile | 
|  |  | 
|  | # Helpers | 
|  | def ensureExists(path): | 
|  | try: | 
|  | os.makedirs(path) | 
|  | except OSError: | 
|  | pass | 
|  |  | 
|  | def writeLinesToFile(lines, fileName): | 
|  | ensureExists(os.path.dirname(fileName)) | 
|  | with open(fileName, "w") as f: | 
|  | f.writelines(lines) | 
|  |  | 
|  | def extractIdg(projFileName): | 
|  | result = [] | 
|  | with open(projFileName) as projFile: | 
|  | lines = iter(projFile) | 
|  | for pLine in lines: | 
|  | if "<ItemDefinitionGroup" in pLine: | 
|  | while not "</ItemDefinitionGroup" in pLine: | 
|  | result.append(pLine) | 
|  | pLine = lines.next() | 
|  | result.append(pLine) | 
|  | return result | 
|  |  | 
|  | # [ (name, hasSln), ... ] | 
|  | configs = [] | 
|  |  | 
|  | # Find all directories that can be used as configs (and record if they have VS | 
|  | # files present) | 
|  | for root, dirs, files in os.walk("out"): | 
|  | for outDir in dirs: | 
|  | gnFile = os.path.join("out", outDir, "build.ninja.d") | 
|  | if os.path.exists(gnFile): | 
|  | slnFile = os.path.join("out", outDir, "all.sln") | 
|  | configs.append((outDir, os.path.exists(slnFile))) | 
|  | break | 
|  |  | 
|  | # Every project has a GUID that encodes the type. We only care about C++. | 
|  | cppTypeGuid = "8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942" | 
|  |  | 
|  | # name -> [ (config, pathToProject, GUID), ... ] | 
|  | allProjects = {} | 
|  | projectPattern = (r'Project\("\{' + cppTypeGuid + | 
|  | r'\}"\) = "([^"]*)", "([^"]*)", "\{([^\}]*)\}"') | 
|  | projectNamePattern = (r'obj/(.*)\.vcxproj') | 
|  |  | 
|  | for config in configs: | 
|  | if config[1]: | 
|  | slnLines = iter(open("out/" + config[0] + "/all.sln")) | 
|  | for slnLine in slnLines: | 
|  | matchObj = re.match(projectPattern, slnLine) | 
|  | if matchObj: | 
|  | projPath = matchObj.group(2) | 
|  | nameObj = re.match(projectNamePattern, projPath) | 
|  | if nameObj: | 
|  | projName = nameObj.group(1).replace('/', '.') | 
|  | if not allProjects.has_key(projName): | 
|  | allProjects[projName] = [] | 
|  | allProjects[projName].append((config[0], projPath, | 
|  | matchObj.group(3))) | 
|  |  | 
|  | # We need something to work with. Typically, this will fail if no GN folders | 
|  | # have IDE files | 
|  | if len(allProjects) == 0: | 
|  | print("ERROR: At least one GN directory must have been built with --ide=vs") | 
|  | sys.exit() | 
|  |  | 
|  | # Create a new solution. We arbitrarily use the first config as the GUID source | 
|  | # (but we need to match that behavior later, when we copy/generate the project | 
|  | # files). | 
|  | newSlnLines = [] | 
|  | newSlnLines.append( | 
|  | 'Microsoft Visual Studio Solution File, Format Version 12.00\n') | 
|  | newSlnLines.append('# Visual Studio 2015\n') | 
|  | for projName, projConfigs in allProjects.items(): | 
|  | newSlnLines.append('Project("{' + cppTypeGuid + '}") = "' + projName + | 
|  | '", "' + projConfigs[0][1] + '", "{' + projConfigs[0][2] | 
|  | + '}"\n') | 
|  | newSlnLines.append('EndProject\n') | 
|  |  | 
|  | newSlnLines.append('Global\n') | 
|  | newSlnLines.append( | 
|  | '\tGlobalSection(SolutionConfigurationPlatforms) = preSolution\n') | 
|  | for config in configs: | 
|  | newSlnLines.append('\t\t' + config[0] + '|x64 = ' + config[0] + '|x64\n') | 
|  | newSlnLines.append('\tEndGlobalSection\n') | 
|  | newSlnLines.append( | 
|  | '\tGlobalSection(ProjectConfigurationPlatforms) = postSolution\n') | 
|  | for projName, projConfigs in allProjects.items(): | 
|  | projGuid = projConfigs[0][2] | 
|  | for config in configs: | 
|  | newSlnLines.append('\t\t{' + projGuid + '}.' + config[0] + | 
|  | '|x64.ActiveCfg = ' + config[0] + '|x64\n') | 
|  | newSlnLines.append('\t\t{' + projGuid + '}.' + config[0] + | 
|  | '|x64.Build.0 = ' + config[0] + '|x64\n') | 
|  | newSlnLines.append('\tEndGlobalSection\n') | 
|  | newSlnLines.append('\tGlobalSection(SolutionProperties) = preSolution\n') | 
|  | newSlnLines.append('\t\tHideSolutionNode = FALSE\n') | 
|  | newSlnLines.append('\tEndGlobalSection\n') | 
|  | newSlnLines.append('\tGlobalSection(NestedProjects) = preSolution\n') | 
|  | newSlnLines.append('\tEndGlobalSection\n') | 
|  | newSlnLines.append('EndGlobal\n') | 
|  |  | 
|  | # Write solution file | 
|  | writeLinesToFile(newSlnLines, "out/sln/skia.sln") | 
|  |  | 
|  | idgHdr = "<ItemDefinitionGroup Condition=\"'$(Configuration)|$(Platform)'=='" | 
|  |  | 
|  | # Now, bring over the project files | 
|  | for projName, projConfigs in allProjects.items(): | 
|  | # Paths to project and filter file in src and dst locations | 
|  | srcProjPath = os.path.join("out", projConfigs[0][0], projConfigs[0][1]) | 
|  | dstProjPath = os.path.join("out", "sln", projConfigs[0][1]) | 
|  | srcFilterPath = srcProjPath + ".filters" | 
|  | dstFilterPath = dstProjPath + ".filters" | 
|  |  | 
|  | # Copy the filter file unmodified | 
|  | ensureExists(os.path.dirname(dstProjPath)) | 
|  | copyfile(srcFilterPath, dstFilterPath) | 
|  |  | 
|  | # Bring over the project file, modified with extra configs | 
|  | with open(srcProjPath) as srcProjFile: | 
|  | projLines = iter(srcProjFile) | 
|  | newProjLines = [] | 
|  | for line in projLines: | 
|  | if "<ItemDefinitionGroup" in line: | 
|  | # This is a large group that contains many settings. We need to | 
|  | # replicate it, with conditions so it varies per configuration. | 
|  | idgLines = [] | 
|  | while not "</ItemDefinitionGroup" in line: | 
|  | idgLines.append(line) | 
|  | line = projLines.next() | 
|  | idgLines.append(line) | 
|  | for projConfig in projConfigs: | 
|  | configIdgLines = extractIdg(os.path.join("out", | 
|  | projConfig[0], | 
|  | projConfig[1])) | 
|  | newProjLines.append(idgHdr + projConfig[0] + "|x64'\">\n") | 
|  | for idgLine in configIdgLines[1:]: | 
|  | newProjLines.append(idgLine) | 
|  | elif "ProjectConfigurations" in line: | 
|  | newProjLines.append(line) | 
|  | projConfigLines = [ | 
|  | projLines.next(), | 
|  | projLines.next(), | 
|  | projLines.next(), | 
|  | projLines.next() ] | 
|  | for config in configs: | 
|  | for projConfigLine in projConfigLines: | 
|  | newProjLines.append(projConfigLine.replace("GN", | 
|  | config[0])) | 
|  | elif "<OutDir" in line: | 
|  | newProjLines.append(line.replace(projConfigs[0][0], | 
|  | "$(Configuration)")) | 
|  | else: | 
|  | newProjLines.append(line) | 
|  | with open(dstProjPath, "w") as newProj: | 
|  | newProj.writelines(newProjLines) |