/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "visualstudio.h" #include "config.h" #include #include namespace CreateProjectTool { ////////////////////////////////////////////////////////////////////////// // Visual Studio Provider (Visual Studio 2008) ////////////////////////////////////////////////////////////////////////// VisualStudioProvider::VisualStudioProvider(StringList &global_warnings, std::map &project_warnings, const int version, const MSVCVersion &msvc) : MSVCProvider(global_warnings, project_warnings, version, msvc) { _archs.push_back(ARCH_X86); _archs.push_back(ARCH_AMD64); } const char *VisualStudioProvider::getProjectExtension() { return ".vcproj"; } const char *VisualStudioProvider::getPropertiesExtension() { return ".vsprops"; } void VisualStudioProvider::createProjectFile(const std::string &name, const std::string &uuid, const BuildSetup &setup, const std::string &moduleDir, const StringList &includeList, const StringList &excludeList) { const std::string projectFile = setup.outputDir + '/' + name + getProjectExtension(); std::ofstream project(projectFile.c_str()); if (!project || !project.is_open()) { error("Could not open \"" + projectFile + "\" for writing"); return; } project << "\n" << "\n" "\t\n"; for (std::list::const_iterator arch = _archs.begin(); arch != _archs.end(); ++arch) { project << "\t\t\n"; } project << "\t\n" << "\t\n"; // Check for project-specific warnings: std::map >::iterator warningsIterator = _projectWarnings.find(name); if (setup.devTools || setup.tests || name == setup.projectName) { for (std::list::const_iterator arch = _archs.begin(); arch != _archs.end(); ++arch) { outputConfiguration(project, setup, false, "Debug", *arch); outputConfiguration(project, setup, false, "Analysis", *arch); outputConfiguration(project, setup, false, "LLVM", *arch); outputConfiguration(project, setup, true, "Release", *arch); } } else { bool enableLanguageExtensions = find(_enableLanguageExtensions.begin(), _enableLanguageExtensions.end(), name) != _enableLanguageExtensions.end(); bool disableEditAndContinue = find(_disableEditAndContinue.begin(), _disableEditAndContinue.end(), name) != _disableEditAndContinue.end(); std::string warnings = ""; if (warningsIterator != _projectWarnings.end()) for (StringList::const_iterator i = warningsIterator->second.begin(); i != warningsIterator->second.end(); ++i) warnings += *i + ';'; std::string toolConfig; toolConfig = (!warnings.empty() ? "DisableSpecificWarnings=\"" + warnings + "\"" : ""); toolConfig += (disableEditAndContinue ? "DebugInformationFormat=\"3\" " : ""); toolConfig += (enableLanguageExtensions ? "DisableLanguageExtensions=\"false\" " : ""); for (std::list::const_iterator arch = _archs.begin(); arch != _archs.end(); ++arch) { outputConfiguration(setup, project, toolConfig, "Debug", *arch); outputConfiguration(setup, project, toolConfig, "Analysis", *arch); outputConfiguration(setup, project, toolConfig, "LLVM", *arch); outputConfiguration(setup, project, toolConfig, "Release", *arch); } } project << "\t\n" << "\t\n"; std::string modulePath; if (!moduleDir.compare(0, setup.srcDir.size(), setup.srcDir)) { modulePath = moduleDir.substr(setup.srcDir.size()); if (!modulePath.empty() && modulePath.at(0) == '/') modulePath.erase(0, 1); } if (modulePath.size()) addFilesToProject(moduleDir, project, includeList, excludeList, setup.filePrefix + '/' + modulePath); else addFilesToProject(moduleDir, project, includeList, excludeList, setup.filePrefix); // Output auto-generated test runner if (setup.tests) { project << "\t\t\n"; } project << "\t\n" << "\n"; } void VisualStudioProvider::outputConfiguration(std::ostream &project, const BuildSetup &setup, bool isRelease, const std::string &config, const MSVC_Architecture arch) { std::string libraries = outputLibraryDependencies(setup, isRelease); project << "\t\t\n" << "\t\t\t\n" << "\t\t\t\n"; outputBuildEvents(project, setup, arch); project << "\t\t\n"; } void VisualStudioProvider::outputConfiguration(const BuildSetup &setup, std::ostream &project, const std::string &toolConfig, const std::string &config, const MSVC_Architecture arch) { project << "\t\t\n" << "\t\t\t\n" << "\t\t\n"; } void VisualStudioProvider::outputBuildEvents(std::ostream &project, const BuildSetup &setup, const MSVC_Architecture arch) { if (!setup.devTools && !setup.tests && setup.runBuildEvents) { project << "\t\t\t\n" << "\t\t\t\n"; } // Generate runner file before build for tests if (setup.tests) { project << "\t\t\t\n"; project << "\t\t\t\n"; } } void VisualStudioProvider::writeReferences(const BuildSetup &setup, std::ofstream &output) { output << "\tProjectSection(ProjectDependencies) = postProject\n"; for (UUIDMap::const_iterator i = _uuidMap.begin(); i != _uuidMap.end(); ++i) { if (i->first == setup.projectName) continue; output << "\t\t{" << i->second << "} = {" << i->second << "}\n"; } output << "\tEndProjectSection\n"; } void VisualStudioProvider::outputGlobalPropFile(const BuildSetup &setup, std::ofstream &properties, MSVC_Architecture arch, const StringList &defines, const std::string &prefix, bool runBuildEvents) { std::string warnings; for (StringList::const_iterator i = _globalWarnings.begin(); i != _globalWarnings.end(); ++i) warnings += *i + ';'; std::string definesList; for (StringList::const_iterator i = defines.begin(); i != defines.end(); ++i) { if (i != defines.begin()) definesList += ';'; definesList += *i; } // Add define to include revision header if (runBuildEvents) definesList += REVISION_DEFINE ";"; properties << "\n" << "\n" << "\t\n" << "\t\n" << "\t\n" << "\t\n" << "\n"; properties.flush(); } void VisualStudioProvider::createBuildProp(const BuildSetup &setup, bool isRelease, MSVC_Architecture arch, const std::string &configuration) { std::ofstream properties((setup.outputDir + '/' + setup.projectDescription + "_" + configuration + getMSVCArchName(arch) + getPropertiesExtension()).c_str()); if (!properties || !properties.is_open()) { error("Could not open \"" + setup.outputDir + '/' + setup.projectDescription + "_" + configuration + getMSVCArchName(arch) + getPropertiesExtension() + "\" for writing"); return; } properties << "\n" << "\n" << "\t\n" << "\t\n" << "\t\n" << "\n"; properties.flush(); properties.close(); } void VisualStudioProvider::writeFileListToProject(const FileNode &dir, std::ofstream &projectFile, const int indentation, const StringList &duplicate, const std::string &objPrefix, const std::string &filePrefix) { const std::string indentString = getIndent(indentation + 2); if (indentation) projectFile << getIndent(indentation + 1) << "\n"; for (FileNode::NodeList::const_iterator i = dir.children.begin(); i != dir.children.end(); ++i) { const FileNode *node = *i; if (!node->children.empty()) { writeFileListToProject(*node, projectFile, indentation + 1, duplicate, objPrefix + node->name + '_', filePrefix + node->name + '/'); } else { if (producesObjectFile(node->name)) { std::string name, ext; splitFilename(node->name, name, ext); name += ".o"; std::transform(name.begin(), name.end(), name.begin(), tolower); const bool isDuplicate = (std::find(duplicate.begin(), duplicate.end(), name) != duplicate.end()); std::string filePath = convertPathToWin(filePrefix + node->name); if (ext == "asm") { std::string objFileName = "$(IntDir)\\"; if (isDuplicate) objFileName += objPrefix; objFileName += "$(InputName).obj"; const std::string toolLine = indentString + "\t\t\n"; // NASM is not supported for x64, thus we do not need to add additional entries here :-). writeFileToProject(projectFile, filePath, ARCH_X86, indentString, toolLine); } else { if (isDuplicate) { const std::string toolLine = indentString + "\t\t\n"; for (std::list::const_iterator arch = _archs.begin(); arch != _archs.end(); ++arch) { writeFileToProject(projectFile, filePath, *arch, indentString, toolLine); } } else { projectFile << indentString << "name) << "\" />\n"; } } } else { projectFile << indentString << "name) << "\" />\n"; } } } if (indentation) projectFile << getIndent(indentation + 1) << "\n"; } void VisualStudioProvider::writeFileToProject(std::ofstream &projectFile, const std::string &filePath, MSVC_Architecture arch, const std::string &indentString, const std::string &toolLine) { projectFile << indentString << "\n" << indentString << "\t\n" << toolLine << indentString << "\t\n" << indentString << "\t\n" << toolLine << indentString << "\t\n" << indentString << "\t\n" << toolLine << indentString << "\t\n" << indentString << "\t\n" << toolLine << indentString << "\t\n" << indentString << "\n"; } } // namespace CreateProjectTool