cmGlobalVisualStudio7Generator.cxx 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file LICENSE.rst or https://cmake.org/licensing for details. */
  3. #include "cmGlobalVisualStudio7Generator.h"
  4. #include <algorithm>
  5. #include <cstdio>
  6. #include <ostream>
  7. #include <utility>
  8. #include <vector>
  9. #include <cm/memory>
  10. #include <cm/string_view>
  11. #include <windows.h>
  12. #include "cmGeneratedFileStream.h"
  13. #include "cmGeneratorExpression.h"
  14. #include "cmGeneratorTarget.h"
  15. #include "cmGlobalGenerator.h"
  16. #include "cmList.h"
  17. #include "cmLocalGenerator.h"
  18. #include "cmLocalVisualStudio7Generator.h"
  19. #include "cmMakefile.h"
  20. #include "cmMessageType.h"
  21. #include "cmState.h"
  22. #include "cmStateTypes.h"
  23. #include "cmStringAlgorithms.h"
  24. #include "cmSystemTools.h"
  25. #include "cmTarget.h"
  26. #include "cmTargetDepend.h"
  27. #include "cmUuid.h"
  28. #include "cmVisualStudioGeneratorOptions.h"
  29. #include "cmake.h"
  30. static cmVS7FlagTable cmVS7ExtraFlagTable[] = {
  31. // Precompiled header and related options. Note that the
  32. // UsePrecompiledHeader entries are marked as "Continue" so that the
  33. // corresponding PrecompiledHeaderThrough entry can be found.
  34. { "UsePrecompiledHeader", "YX", "Automatically Generate", "2",
  35. cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue },
  36. { "PrecompiledHeaderThrough", "YX", "Precompiled Header Name", "",
  37. cmVS7FlagTable::UserValueRequired },
  38. { "UsePrecompiledHeader", "Yu", "Use Precompiled Header", "3",
  39. cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue },
  40. { "PrecompiledHeaderThrough", "Yu", "Precompiled Header Name", "",
  41. cmVS7FlagTable::UserValueRequired },
  42. { "UsePrecompiledHeader", "Y-", "Don't use precompiled header", "0", 0 },
  43. { "WholeProgramOptimization", "LTCG", "WholeProgramOptimization", "true",
  44. 0 },
  45. // Exception handling mode. If no entries match, it will be FALSE.
  46. { "ExceptionHandling", "GX", "enable c++ exceptions", "true", 0 },
  47. { "ExceptionHandling", "EHsc", "enable c++ exceptions", "true", 0 },
  48. // The EHa option does not have an IDE setting. Let it go to false,
  49. // and have EHa passed on the command line by leaving out the table
  50. // entry.
  51. { "", "", "", "", 0 }
  52. };
  53. namespace {
  54. std::string GetSLNFile(cmLocalGenerator* root)
  55. {
  56. return cmStrCat(root->GetCurrentBinaryDirectory(), '/',
  57. root->GetProjectName(), ".sln");
  58. }
  59. }
  60. cmGlobalVisualStudio7Generator::cmGlobalVisualStudio7Generator(cmake* cm)
  61. : cmGlobalVisualStudioGenerator(cm)
  62. {
  63. this->DevEnvCommandInitialized = false;
  64. this->MarmasmEnabled = false;
  65. this->MasmEnabled = false;
  66. this->NasmEnabled = false;
  67. this->ExtraFlagTable = cmVS7ExtraFlagTable;
  68. }
  69. cmGlobalVisualStudio7Generator::~cmGlobalVisualStudio7Generator() = default;
  70. // Package GUID of Intel Visual Fortran plugin to VS IDE
  71. #define CM_INTEL_PLUGIN_GUID "{B68A201D-CB9B-47AF-A52F-7EEC72E217E4}"
  72. std::string const& cmGlobalVisualStudio7Generator::GetIntelProjectVersion()
  73. {
  74. if (this->IntelProjectVersion.empty()) {
  75. // Compute the version of the Intel plugin to the VS IDE.
  76. // If the key does not exist then use a default guess.
  77. std::string intelVersion;
  78. std::string vskey =
  79. cmStrCat(this->GetRegistryBase(),
  80. "\\Packages\\" CM_INTEL_PLUGIN_GUID ";ProductVersion");
  81. cmSystemTools::ReadRegistryValue(vskey, intelVersion,
  82. cmSystemTools::KeyWOW64_32);
  83. unsigned int intelVersionNumber = ~0u;
  84. if (sscanf(intelVersion.c_str(), "%u", &intelVersionNumber) != 1 ||
  85. intelVersionNumber >= 11) {
  86. // Default to latest known project file version.
  87. intelVersion = "11.0";
  88. } else if (intelVersionNumber == 10) {
  89. // Version 10.x actually uses 9.10 in project files!
  90. intelVersion = "9.10";
  91. } else {
  92. // Version <= 9: use ProductVersion from registry.
  93. }
  94. this->IntelProjectVersion = intelVersion;
  95. }
  96. return this->IntelProjectVersion;
  97. }
  98. void cmGlobalVisualStudio7Generator::EnableLanguage(
  99. std::vector<std::string> const& lang, cmMakefile* mf, bool optional)
  100. {
  101. mf->AddDefinition("CMAKE_GENERATOR_RC", "rc");
  102. mf->AddDefinition("CMAKE_GENERATOR_NO_COMPILER_ENV", "1");
  103. mf->InitCMAKE_CONFIGURATION_TYPES("Debug;Release;MinSizeRel;RelWithDebInfo");
  104. // Create list of configurations requested by user's cache, if any.
  105. this->cmGlobalVisualStudioGenerator::EnableLanguage(lang, mf, optional);
  106. // if this environment variable is set, then copy it to
  107. // a static cache entry. It will be used by
  108. // cmLocalGenerator::ConstructScript, to add an extra PATH
  109. // to all custom commands. This is because the VS IDE
  110. // does not use the environment it is run in, and this allows
  111. // for running commands and using dll's that the IDE environment
  112. // does not know about.
  113. std::string extraPath;
  114. if (cmSystemTools::GetEnv("CMAKE_MSVCIDE_RUN_PATH", extraPath)) {
  115. mf->AddCacheDefinition("CMAKE_MSVCIDE_RUN_PATH", extraPath,
  116. "Saved environment variable CMAKE_MSVCIDE_RUN_PATH",
  117. cmStateEnums::STATIC);
  118. }
  119. }
  120. bool cmGlobalVisualStudio7Generator::FindMakeProgram(cmMakefile* mf)
  121. {
  122. if (!this->cmGlobalVisualStudioGenerator::FindMakeProgram(mf)) {
  123. return false;
  124. }
  125. mf->AddDefinition("CMAKE_VS_DEVENV_COMMAND", this->GetDevEnvCommand());
  126. return true;
  127. }
  128. std::string const& cmGlobalVisualStudio7Generator::GetDevEnvCommand()
  129. {
  130. if (!this->DevEnvCommandInitialized) {
  131. this->DevEnvCommandInitialized = true;
  132. this->DevEnvCommand = this->FindDevEnvCommand();
  133. }
  134. return this->DevEnvCommand;
  135. }
  136. std::string cmGlobalVisualStudio7Generator::FindDevEnvCommand()
  137. {
  138. std::string vscmd;
  139. std::string vskey;
  140. // Search in standard location.
  141. vskey = cmStrCat(this->GetRegistryBase(), ";InstallDir");
  142. if (cmSystemTools::ReadRegistryValue(vskey, vscmd,
  143. cmSystemTools::KeyWOW64_32)) {
  144. cmSystemTools::ConvertToUnixSlashes(vscmd);
  145. vscmd += "/devenv.com";
  146. if (cmSystemTools::FileExists(vscmd, true)) {
  147. return vscmd;
  148. }
  149. }
  150. // Search where VS15Preview places it.
  151. vskey =
  152. cmStrCat(R"(HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\VisualStudio\SxS\VS7;)",
  153. this->GetIDEVersion());
  154. if (cmSystemTools::ReadRegistryValue(vskey, vscmd,
  155. cmSystemTools::KeyWOW64_32)) {
  156. cmSystemTools::ConvertToUnixSlashes(vscmd);
  157. vscmd += "/Common7/IDE/devenv.com";
  158. if (cmSystemTools::FileExists(vscmd, true)) {
  159. return vscmd;
  160. }
  161. }
  162. vscmd = "devenv.com";
  163. return vscmd;
  164. }
  165. char const* cmGlobalVisualStudio7Generator::ExternalProjectType(
  166. std::string const& location)
  167. {
  168. std::string extension = cmSystemTools::GetFilenameLastExtension(location);
  169. if (extension == ".vbproj"_s) {
  170. return "F184B08F-C81C-45F6-A57F-5ABD9991F28F";
  171. }
  172. if (extension == ".csproj"_s) {
  173. return "FAE04EC0-301F-11D3-BF4B-00C04F79EFBC";
  174. }
  175. if (extension == ".fsproj"_s) {
  176. return "F2A71F9B-5D33-465A-A702-920D77279786";
  177. }
  178. if (extension == ".vdproj"_s) {
  179. return "54435603-DBB4-11D2-8724-00A0C9A8B90C";
  180. }
  181. if (extension == ".dbproj"_s) {
  182. return "C8D11400-126E-41CD-887F-60BD40844F9E";
  183. }
  184. if (extension == ".wixproj"_s) {
  185. return "930C7802-8A8C-48F9-8165-68863BCCD9DD";
  186. }
  187. if (extension == ".pyproj"_s) {
  188. return "888888A0-9F3D-457C-B088-3A5042F75D52";
  189. }
  190. return "8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942";
  191. }
  192. std::vector<cmGlobalGenerator::GeneratedMakeCommand>
  193. cmGlobalVisualStudio7Generator::GenerateBuildCommand(
  194. std::string const& makeProgram, std::string const& projectName,
  195. std::string const& /*projectDir*/,
  196. std::vector<std::string> const& targetNames, std::string const& config,
  197. int /*jobs*/, bool /*verbose*/, cmBuildOptions const& /*buildOptions*/,
  198. std::vector<std::string> const& makeOptions)
  199. {
  200. // Select the caller- or user-preferred make program, else devenv.
  201. std::string makeProgramSelected =
  202. this->SelectMakeProgram(makeProgram, this->GetDevEnvCommand());
  203. // Ignore the above preference if it is msbuild.
  204. // Assume any other value is either a devenv or
  205. // command-line compatible with devenv.
  206. std::string makeProgramLower = makeProgramSelected;
  207. cmSystemTools::LowerCase(makeProgramLower);
  208. if (makeProgramLower.find("msbuild") != std::string::npos) {
  209. makeProgramSelected = this->GetDevEnvCommand();
  210. }
  211. // Workaround to convince VCExpress.exe to produce output.
  212. bool const requiresOutputForward =
  213. (makeProgramLower.find("vcexpress") != std::string::npos);
  214. std::vector<GeneratedMakeCommand> makeCommands;
  215. std::vector<std::string> realTargetNames = targetNames;
  216. if (targetNames.empty() ||
  217. ((targetNames.size() == 1) && targetNames.front().empty())) {
  218. realTargetNames = { "ALL_BUILD" };
  219. }
  220. for (auto const& tname : realTargetNames) {
  221. std::string realTarget;
  222. if (!tname.empty()) {
  223. realTarget = tname;
  224. } else {
  225. continue;
  226. }
  227. bool clean = false;
  228. if (realTarget == "clean"_s) {
  229. clean = true;
  230. realTarget = "ALL_BUILD";
  231. }
  232. GeneratedMakeCommand makeCommand;
  233. makeCommand.RequiresOutputForward = requiresOutputForward;
  234. makeCommand.Add(makeProgramSelected);
  235. makeCommand.Add(cmStrCat(projectName, ".sln"));
  236. makeCommand.Add((clean ? "/clean" : "/build"));
  237. makeCommand.Add((config.empty() ? "Debug" : config));
  238. makeCommand.Add("/project");
  239. makeCommand.Add(realTarget);
  240. makeCommand.Add(makeOptions.begin(), makeOptions.end());
  241. makeCommands.emplace_back(std::move(makeCommand));
  242. }
  243. return makeCommands;
  244. }
  245. //! Create a local generator appropriate to this Global Generator
  246. std::unique_ptr<cmLocalGenerator>
  247. cmGlobalVisualStudio7Generator::CreateLocalGenerator(cmMakefile* mf)
  248. {
  249. auto lg = cm::make_unique<cmLocalVisualStudio7Generator>(this, mf);
  250. return std::unique_ptr<cmLocalGenerator>(std::move(lg));
  251. }
  252. #if !defined(CMAKE_BOOTSTRAP)
  253. Json::Value cmGlobalVisualStudio7Generator::GetJson() const
  254. {
  255. Json::Value generator = this->cmGlobalVisualStudioGenerator::GetJson();
  256. generator["platform"] = this->GetPlatformName();
  257. return generator;
  258. }
  259. #endif
  260. bool cmGlobalVisualStudio7Generator::SetSystemName(std::string const& s,
  261. cmMakefile* mf)
  262. {
  263. mf->AddDefinition("CMAKE_VS_INTEL_Fortran_PROJECT_VERSION",
  264. this->GetIntelProjectVersion());
  265. return this->cmGlobalVisualStudioGenerator::SetSystemName(s, mf);
  266. }
  267. void cmGlobalVisualStudio7Generator::Generate()
  268. {
  269. // first do the superclass method
  270. this->cmGlobalVisualStudioGenerator::Generate();
  271. // Now write out the VS Solution files.
  272. for (auto& it : this->ProjectMap) {
  273. this->OutputSLNFile(it.second[0], it.second);
  274. }
  275. // If any solution or project files changed during the generation,
  276. // tell Visual Studio to reload them...
  277. if (!cmSystemTools::GetErrorOccurredFlag() &&
  278. !this->LocalGenerators.empty()) {
  279. this->CallVisualStudioMacro(MacroReload,
  280. GetSLNFile(this->LocalGenerators[0].get()));
  281. }
  282. }
  283. void cmGlobalVisualStudio7Generator::OutputSLNFile(
  284. cmLocalGenerator* root, std::vector<cmLocalGenerator*>& generators)
  285. {
  286. if (generators.empty()) {
  287. return;
  288. }
  289. // Collect all targets under this root generator and the transitive
  290. // closure of their dependencies.
  291. TargetDependSet projectTargets;
  292. TargetDependSet originalTargets;
  293. this->GetTargetSets(projectTargets, originalTargets, root, generators);
  294. OrderedTargetDependSet orderedProjectTargets(
  295. projectTargets, this->GetStartupProjectName(root));
  296. VSFolders vsFolders = this->CreateSolutionFolders(orderedProjectTargets);
  297. this->AddSolutionItems(root, vsFolders);
  298. std::string fname = GetSLNFile(root);
  299. cmGeneratedFileStream fout(fname);
  300. fout.SetCopyIfDifferent(true);
  301. if (!fout) {
  302. return;
  303. }
  304. this->WriteSLNFile(fout, root, orderedProjectTargets, vsFolders);
  305. if (fout.Close()) {
  306. this->FileReplacedDuringGenerate(fname);
  307. }
  308. }
  309. void cmGlobalVisualStudio7Generator::WriteTargetConfigurations(
  310. std::ostream& fout, std::vector<std::string> const& configs,
  311. OrderedTargetDependSet const& projectTargets) const
  312. {
  313. // loop over again and write out configurations for each target
  314. // in the solution
  315. for (cmGeneratorTarget const* target : projectTargets) {
  316. if (!this->IsInSolution(target)) {
  317. continue;
  318. }
  319. cmValue expath = target->GetProperty("EXTERNAL_MSPROJECT");
  320. if (expath) {
  321. std::set<std::string> allConfigurations(configs.begin(), configs.end());
  322. cmValue mapping = target->GetProperty("VS_PLATFORM_MAPPING");
  323. this->WriteProjectConfigurations(fout, target->GetName(), *target,
  324. configs, allConfigurations,
  325. mapping ? *mapping : "");
  326. } else {
  327. std::set<std::string> const& configsPartOfDefaultBuild =
  328. this->IsPartOfDefaultBuild(configs, projectTargets, target);
  329. cmValue vcprojName = target->GetProperty("GENERATOR_FILE_NAME");
  330. if (vcprojName) {
  331. std::string mapping;
  332. // On VS 19 and above, always map .NET SDK projects to "Any CPU".
  333. if (target->IsDotNetSdkTarget() && this->Version >= VSVersion::VS16 &&
  334. !cmGlobalVisualStudio7Generator::IsReservedTarget(
  335. target->GetName())) {
  336. mapping = "Any CPU";
  337. }
  338. this->WriteProjectConfigurations(fout, *vcprojName, *target, configs,
  339. configsPartOfDefaultBuild, mapping);
  340. }
  341. }
  342. }
  343. }
  344. cmGlobalVisualStudio7Generator::VSFolders
  345. cmGlobalVisualStudio7Generator::CreateSolutionFolders(
  346. OrderedTargetDependSet const& orderedProjectTargets)
  347. {
  348. VSFolders vsFolders;
  349. if (!this->UseFolderProperty()) {
  350. return vsFolders;
  351. }
  352. for (cmGeneratorTarget const* target : orderedProjectTargets) {
  353. if (this->IsInSolution(target) &&
  354. (target->GetProperty("EXTERNAL_MSPROJECT") ||
  355. target->GetProperty("GENERATOR_FILE_NAME"))) {
  356. // Create "solution folder" information from FOLDER target property
  357. if (cmVisualStudioFolder* folder =
  358. vsFolders.Create(target->GetEffectiveFolderName())) {
  359. folder->Projects.insert(target->GetName());
  360. }
  361. }
  362. }
  363. return vsFolders;
  364. }
  365. cmVisualStudioFolder* cmGlobalVisualStudio7Generator::VSFolders::Create(
  366. std::string const& path)
  367. {
  368. if (path.empty()) {
  369. return nullptr;
  370. }
  371. std::vector<std::string> tokens =
  372. cmSystemTools::SplitString(path, '/', false);
  373. std::string cumulativePath;
  374. for (std::string const& iter : tokens) {
  375. if (iter.empty()) {
  376. continue;
  377. }
  378. if (cumulativePath.empty()) {
  379. cumulativePath = cmStrCat("CMAKE_FOLDER_GUID_", iter);
  380. } else {
  381. this->Folders[cumulativePath].Projects.insert(
  382. cmStrCat(cumulativePath, '/', iter));
  383. cumulativePath = cmStrCat(cumulativePath, '/', iter);
  384. }
  385. }
  386. if (cumulativePath.empty()) {
  387. return nullptr;
  388. }
  389. return &this->Folders[cumulativePath];
  390. }
  391. void cmGlobalVisualStudio7Generator::WriteTargetsToSolution(
  392. std::ostream& fout, cmLocalGenerator* root,
  393. OrderedTargetDependSet const& projectTargets) const
  394. {
  395. std::vector<std::string> configs =
  396. root->GetMakefile()->GetGeneratorConfigs(cmMakefile::ExcludeEmptyConfig);
  397. for (cmGeneratorTarget const* target : projectTargets) {
  398. if (!this->IsInSolution(target)) {
  399. continue;
  400. }
  401. // handle external vc project files
  402. cmValue expath = target->GetProperty("EXTERNAL_MSPROJECT");
  403. if (expath) {
  404. std::string project = target->GetName();
  405. std::string const& location = *expath;
  406. this->WriteExternalProject(fout, project, location,
  407. target->GetProperty("VS_PROJECT_TYPE"),
  408. target->GetUtilities());
  409. } else {
  410. cmValue vcprojName = target->GetProperty("GENERATOR_FILE_NAME");
  411. if (vcprojName) {
  412. cmLocalGenerator* lg = target->GetLocalGenerator();
  413. std::string dir = lg->GetCurrentBinaryDirectory();
  414. dir = root->MaybeRelativeToCurBinDir(dir);
  415. if (dir == "."_s) {
  416. dir.clear(); // msbuild cannot handle ".\" prefix
  417. }
  418. this->WriteProject(fout, *vcprojName, dir, target);
  419. }
  420. }
  421. }
  422. }
  423. void cmGlobalVisualStudio7Generator::WriteFolders(
  424. std::ostream& fout, VSFolders const& vsFolders) const
  425. {
  426. cm::string_view const prefix = "CMAKE_FOLDER_GUID_";
  427. std::string guidProjectTypeFolder = "2150E333-8FDC-42A3-9474-1A3956D46DE8";
  428. for (auto const& iter : vsFolders.Folders) {
  429. std::string fullName = iter.first;
  430. std::string guid = this->GetGUID(fullName);
  431. std::replace(fullName.begin(), fullName.end(), '/', '\\');
  432. if (cmHasPrefix(fullName, prefix)) {
  433. fullName = fullName.substr(prefix.size());
  434. }
  435. std::string nameOnly = cmSystemTools::GetFilenameName(fullName);
  436. fout << "Project(\"{" << guidProjectTypeFolder << "}\") = \"" << nameOnly
  437. << "\", \"" << fullName << "\", \"{" << guid << "}\"\n";
  438. if (!iter.second.SolutionItems.empty()) {
  439. this->WriteFolderSolutionItems(fout, iter.second);
  440. }
  441. fout << "EndProject\n";
  442. }
  443. }
  444. void cmGlobalVisualStudio7Generator::WriteFoldersContent(
  445. std::ostream& fout, VSFolders const& vsFolders) const
  446. {
  447. for (auto const& iter : vsFolders.Folders) {
  448. std::string key(iter.first);
  449. std::string guidParent(this->GetGUID(key));
  450. for (std::string const& it : iter.second.Projects) {
  451. std::string const& value(it);
  452. std::string guid(this->GetGUID(value));
  453. fout << "\t\t{" << guid << "} = {" << guidParent << "}\n";
  454. }
  455. }
  456. }
  457. std::string cmGlobalVisualStudio7Generator::ConvertToSolutionPath(
  458. std::string const& path) const
  459. {
  460. // Convert to backslashes. Do not use ConvertToOutputPath because
  461. // we will add quoting ourselves, and we know these projects always
  462. // use windows slashes.
  463. std::string d = path;
  464. std::string::size_type pos = 0;
  465. while ((pos = d.find('/', pos)) != std::string::npos) {
  466. d[pos++] = '\\';
  467. }
  468. return d;
  469. }
  470. void cmGlobalVisualStudio7Generator::WriteSLNGlobalSections(
  471. std::ostream& fout, cmLocalGenerator* root) const
  472. {
  473. std::string const guid =
  474. this->GetGUID(cmStrCat(root->GetProjectName(), ".sln"));
  475. bool extensibilityGlobalsOverridden = false;
  476. bool extensibilityAddInsOverridden = false;
  477. std::vector<std::string> const propKeys =
  478. root->GetMakefile()->GetPropertyKeys();
  479. for (std::string const& it : propKeys) {
  480. if (cmHasLiteralPrefix(it, "VS_GLOBAL_SECTION_")) {
  481. std::string sectionType;
  482. std::string name = it.substr(18);
  483. if (cmHasLiteralPrefix(name, "PRE_")) {
  484. name = name.substr(4);
  485. sectionType = "preSolution";
  486. } else if (cmHasLiteralPrefix(name, "POST_")) {
  487. name = name.substr(5);
  488. sectionType = "postSolution";
  489. } else {
  490. continue;
  491. }
  492. if (!name.empty()) {
  493. bool addGuid = false;
  494. if (name == "ExtensibilityGlobals"_s &&
  495. sectionType == "postSolution"_s) {
  496. addGuid = true;
  497. extensibilityGlobalsOverridden = true;
  498. } else if (name == "ExtensibilityAddIns"_s &&
  499. sectionType == "postSolution"_s) {
  500. extensibilityAddInsOverridden = true;
  501. }
  502. fout << "\tGlobalSection(" << name << ") = " << sectionType << '\n';
  503. cmValue p = root->GetMakefile()->GetProperty(it);
  504. cmList keyValuePairs{ *p };
  505. for (std::string const& itPair : keyValuePairs) {
  506. std::string::size_type const posEqual = itPair.find('=');
  507. if (posEqual != std::string::npos) {
  508. std::string const key =
  509. cmTrimWhitespace(itPair.substr(0, posEqual));
  510. std::string const value =
  511. cmTrimWhitespace(itPair.substr(posEqual + 1));
  512. fout << "\t\t" << key << " = " << value << '\n';
  513. if (key == "SolutionGuid"_s) {
  514. addGuid = false;
  515. }
  516. }
  517. }
  518. if (addGuid) {
  519. fout << "\t\tSolutionGuid = {" << guid << "}\n";
  520. }
  521. fout << "\tEndGlobalSection\n";
  522. }
  523. }
  524. }
  525. if (!extensibilityGlobalsOverridden) {
  526. fout << "\tGlobalSection(ExtensibilityGlobals) = postSolution\n"
  527. << "\t\tSolutionGuid = {" << guid << "}\n"
  528. << "\tEndGlobalSection\n";
  529. }
  530. if (!extensibilityAddInsOverridden) {
  531. fout << "\tGlobalSection(ExtensibilityAddIns) = postSolution\n"
  532. << "\tEndGlobalSection\n";
  533. }
  534. }
  535. // Standard end of dsw file
  536. void cmGlobalVisualStudio7Generator::WriteSLNFooter(std::ostream& fout) const
  537. {
  538. fout << "EndGlobal\n";
  539. }
  540. std::string cmGlobalVisualStudio7Generator::WriteUtilityDepend(
  541. cmGeneratorTarget const* target)
  542. {
  543. std::vector<std::string> configs =
  544. target->Target->GetMakefile()->GetGeneratorConfigs(
  545. cmMakefile::ExcludeEmptyConfig);
  546. std::string pname = cmStrCat(target->GetName(), "_UTILITY");
  547. std::string fname =
  548. cmStrCat(target->GetLocalGenerator()->GetCurrentBinaryDirectory(), '/',
  549. pname, ".vcproj");
  550. cmGeneratedFileStream fout(fname);
  551. fout.SetCopyIfDifferent(true);
  552. std::string guid = this->GetGUID(pname);
  553. /* clang-format off */
  554. fout <<
  555. R"(<?xml version="1.0" encoding = ")"
  556. << this->Encoding() << "\"?>\n"
  557. "<VisualStudioProject\n"
  558. "\tProjectType=\"Visual C++\"\n"
  559. "\tVersion=\"" << this->GetIDEVersion() << "0\"\n"
  560. "\tName=\"" << pname << "\"\n"
  561. "\tProjectGUID=\"{" << guid << "}\"\n"
  562. "\tKeyword=\"Win32Proj\">\n"
  563. "\t<Platforms><Platform Name=\"Win32\"/></Platforms>\n"
  564. "\t<Configurations>\n"
  565. ;
  566. /* clang-format on */
  567. std::string intDirPrefix =
  568. target->GetLocalGenerator()->MaybeRelativeToCurBinDir(
  569. cmStrCat(target->GetSupportDirectory(), '\\'));
  570. for (std::string const& i : configs) {
  571. std::string intDir = cmStrCat(intDirPrefix, i);
  572. /* clang-format off */
  573. fout <<
  574. "\t\t<Configuration\n"
  575. "\t\t\tName=\"" << i << "|Win32\"\n"
  576. "\t\t\tOutputDirectory=\"" << i << "\"\n"
  577. "\t\t\tIntermediateDirectory=\"" << intDir << "\"\n"
  578. "\t\t\tConfigurationType=\"10\"\n"
  579. "\t\t\tUseOfMFC=\"0\"\n"
  580. "\t\t\tATLMinimizesCRunTimeLibraryUsage=\"FALSE\"\n"
  581. "\t\t\tCharacterSet=\"2\">\n"
  582. "\t\t</Configuration>\n"
  583. ;
  584. /* clang-format on */
  585. }
  586. /* clang-format off */
  587. fout <<
  588. "\t</Configurations>\n"
  589. "\t<Files></Files>\n"
  590. "\t<Globals></Globals>\n"
  591. "</VisualStudioProject>\n"
  592. ;
  593. /* clang-format on */
  594. if (fout.Close()) {
  595. this->FileReplacedDuringGenerate(fname);
  596. }
  597. return pname;
  598. }
  599. std::string cmGlobalVisualStudio7Generator::GetGUID(
  600. std::string const& name) const
  601. {
  602. std::string const& guidStoreName = cmStrCat(name, "_GUID_CMAKE");
  603. if (cmValue storedGUID =
  604. this->CMakeInstance->GetCacheDefinition(guidStoreName)) {
  605. return *storedGUID;
  606. }
  607. // Compute a GUID that is deterministic but unique to the build tree.
  608. std::string input =
  609. cmStrCat(this->CMakeInstance->GetState()->GetBinaryDirectory(), '|', name);
  610. cmUuid uuidGenerator;
  611. std::vector<unsigned char> uuidNamespace;
  612. uuidGenerator.StringToBinary("ee30c4be-5192-4fb0-b335-722a2dffe760",
  613. uuidNamespace);
  614. std::string guid = uuidGenerator.FromMd5(uuidNamespace, input);
  615. return cmSystemTools::UpperCase(guid);
  616. }
  617. void cmGlobalVisualStudio7Generator::AppendDirectoryForConfig(
  618. std::string const& prefix, std::string const& config,
  619. std::string const& suffix, std::string& dir)
  620. {
  621. if (!config.empty()) {
  622. dir += cmStrCat(prefix, config, suffix);
  623. }
  624. }
  625. std::set<std::string> cmGlobalVisualStudio7Generator::IsPartOfDefaultBuild(
  626. std::vector<std::string> const& configs,
  627. OrderedTargetDependSet const& projectTargets,
  628. cmGeneratorTarget const* target) const
  629. {
  630. std::set<std::string> activeConfigs;
  631. // if it is a utility target then only make it part of the
  632. // default build if another target depends on it
  633. int type = target->GetType();
  634. if (type == cmStateEnums::GLOBAL_TARGET) {
  635. std::vector<std::string> targetNames;
  636. targetNames.push_back("INSTALL");
  637. targetNames.push_back("PACKAGE");
  638. for (std::string const& t : targetNames) {
  639. // check if target <t> is part of default build
  640. if (target->GetName() == t) {
  641. std::string const propertyName =
  642. cmStrCat("CMAKE_VS_INCLUDE_", t, "_TO_DEFAULT_BUILD");
  643. // inspect CMAKE_VS_INCLUDE_<t>_TO_DEFAULT_BUILD properties
  644. for (std::string const& i : configs) {
  645. cmValue propertyValue =
  646. target->Target->GetMakefile()->GetDefinition(propertyName);
  647. if (propertyValue &&
  648. cmIsOn(cmGeneratorExpression::Evaluate(
  649. *propertyValue, target->GetLocalGenerator(), i))) {
  650. activeConfigs.insert(i);
  651. }
  652. }
  653. }
  654. }
  655. return activeConfigs;
  656. }
  657. if (type == cmStateEnums::UTILITY &&
  658. !this->IsDependedOn(projectTargets, target)) {
  659. return activeConfigs;
  660. }
  661. // inspect EXCLUDE_FROM_DEFAULT_BUILD[_<CONFIG>] properties
  662. for (std::string const& i : configs) {
  663. if (target->GetFeature("EXCLUDE_FROM_DEFAULT_BUILD", i).IsOff()) {
  664. activeConfigs.insert(i);
  665. }
  666. }
  667. return activeConfigs;
  668. }
  669. bool cmGlobalVisualStudio7Generator::IsDependedOn(
  670. OrderedTargetDependSet const& projectTargets,
  671. cmGeneratorTarget const* gtIn) const
  672. {
  673. return std::any_of(projectTargets.begin(), projectTargets.end(),
  674. [this, gtIn](cmTargetDepend const& l) {
  675. TargetDependSet const& tgtdeps =
  676. this->GetTargetDirectDepends(l);
  677. return tgtdeps.count(gtIn);
  678. });
  679. }
  680. std::string cmGlobalVisualStudio7Generator::Encoding()
  681. {
  682. return "UTF-8";
  683. }