cmGlobalVisualStudioVersionedGenerator.cxx 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file Copyright.txt or https://cmake.org/licensing for details. */
  3. #include "cmGlobalVisualStudioVersionedGenerator.h"
  4. #include "cmAlgorithms.h"
  5. #include "cmDocumentationEntry.h"
  6. #include "cmLocalVisualStudio10Generator.h"
  7. #include "cmMakefile.h"
  8. #include "cmStringAlgorithms.h"
  9. #include "cmVSSetupHelper.h"
  10. #include "cmake.h"
  11. #if defined(_M_ARM64)
  12. # define HOST_PLATFORM_NAME "ARM64"
  13. # define HOST_TOOLS_ARCH ""
  14. #elif defined(_M_ARM)
  15. # define HOST_PLATFORM_NAME "ARM"
  16. # define HOST_TOOLS_ARCH ""
  17. #elif defined(_M_IA64)
  18. # define HOST_PLATFORM_NAME "Itanium"
  19. # define HOST_TOOLS_ARCH ""
  20. #elif defined(_WIN64)
  21. # define HOST_PLATFORM_NAME "x64"
  22. # define HOST_TOOLS_ARCH "x64"
  23. #else
  24. static bool VSIsWow64()
  25. {
  26. BOOL isWow64 = false;
  27. return IsWow64Process(GetCurrentProcess(), &isWow64) && isWow64;
  28. }
  29. #endif
  30. static std::string VSHostPlatformName()
  31. {
  32. #ifdef HOST_PLATFORM_NAME
  33. return HOST_PLATFORM_NAME;
  34. #else
  35. if (VSIsWow64()) {
  36. return "x64";
  37. } else {
  38. return "Win32";
  39. }
  40. #endif
  41. }
  42. static std::string VSHostArchitecture()
  43. {
  44. #ifdef HOST_TOOLS_ARCH
  45. return HOST_TOOLS_ARCH;
  46. #else
  47. if (VSIsWow64()) {
  48. return "x64";
  49. } else {
  50. return "x86";
  51. }
  52. #endif
  53. }
  54. static unsigned int VSVersionToMajor(
  55. cmGlobalVisualStudioGenerator::VSVersion v)
  56. {
  57. switch (v) {
  58. case cmGlobalVisualStudioGenerator::VS9:
  59. return 9;
  60. case cmGlobalVisualStudioGenerator::VS10:
  61. return 10;
  62. case cmGlobalVisualStudioGenerator::VS11:
  63. return 11;
  64. case cmGlobalVisualStudioGenerator::VS12:
  65. return 12;
  66. case cmGlobalVisualStudioGenerator::VS14:
  67. return 14;
  68. case cmGlobalVisualStudioGenerator::VS15:
  69. return 15;
  70. case cmGlobalVisualStudioGenerator::VS16:
  71. return 16;
  72. }
  73. return 0;
  74. }
  75. static const char* VSVersionToToolset(
  76. cmGlobalVisualStudioGenerator::VSVersion v)
  77. {
  78. switch (v) {
  79. case cmGlobalVisualStudioGenerator::VS9:
  80. return "v90";
  81. case cmGlobalVisualStudioGenerator::VS10:
  82. return "v100";
  83. case cmGlobalVisualStudioGenerator::VS11:
  84. return "v110";
  85. case cmGlobalVisualStudioGenerator::VS12:
  86. return "v120";
  87. case cmGlobalVisualStudioGenerator::VS14:
  88. return "v140";
  89. case cmGlobalVisualStudioGenerator::VS15:
  90. return "v141";
  91. case cmGlobalVisualStudioGenerator::VS16:
  92. return "v142";
  93. }
  94. return "";
  95. }
  96. static const char vs15generatorName[] = "Visual Studio 15 2017";
  97. // Map generator name without year to name with year.
  98. static const char* cmVS15GenName(const std::string& name, std::string& genName)
  99. {
  100. if (strncmp(name.c_str(), vs15generatorName,
  101. sizeof(vs15generatorName) - 6) != 0) {
  102. return 0;
  103. }
  104. const char* p = name.c_str() + sizeof(vs15generatorName) - 6;
  105. if (cmHasLiteralPrefix(p, " 2017")) {
  106. p += 5;
  107. }
  108. genName = std::string(vs15generatorName) + p;
  109. return p;
  110. }
  111. class cmGlobalVisualStudioVersionedGenerator::Factory15
  112. : public cmGlobalGeneratorFactory
  113. {
  114. public:
  115. std::unique_ptr<cmGlobalGenerator> CreateGlobalGenerator(
  116. const std::string& name, cmake* cm) const override
  117. {
  118. std::string genName;
  119. const char* p = cmVS15GenName(name, genName);
  120. if (!p) {
  121. return std::unique_ptr<cmGlobalGenerator>();
  122. }
  123. if (!*p) {
  124. return std::unique_ptr<cmGlobalGenerator>(
  125. new cmGlobalVisualStudioVersionedGenerator(
  126. cmGlobalVisualStudioGenerator::VS15, cm, genName, ""));
  127. }
  128. if (*p++ != ' ') {
  129. return std::unique_ptr<cmGlobalGenerator>();
  130. }
  131. if (strcmp(p, "Win64") == 0) {
  132. return std::unique_ptr<cmGlobalGenerator>(
  133. new cmGlobalVisualStudioVersionedGenerator(
  134. cmGlobalVisualStudioGenerator::VS15, cm, genName, "x64"));
  135. }
  136. if (strcmp(p, "ARM") == 0) {
  137. return std::unique_ptr<cmGlobalGenerator>(
  138. new cmGlobalVisualStudioVersionedGenerator(
  139. cmGlobalVisualStudioGenerator::VS15, cm, genName, "ARM"));
  140. }
  141. return std::unique_ptr<cmGlobalGenerator>();
  142. }
  143. void GetDocumentation(cmDocumentationEntry& entry) const override
  144. {
  145. entry.Name = std::string(vs15generatorName) + " [arch]";
  146. entry.Brief = "Generates Visual Studio 2017 project files. "
  147. "Optional [arch] can be \"Win64\" or \"ARM\".";
  148. }
  149. std::vector<std::string> GetGeneratorNames() const override
  150. {
  151. std::vector<std::string> names;
  152. names.push_back(vs15generatorName);
  153. return names;
  154. }
  155. std::vector<std::string> GetGeneratorNamesWithPlatform() const override
  156. {
  157. std::vector<std::string> names;
  158. names.push_back(vs15generatorName + std::string(" ARM"));
  159. names.push_back(vs15generatorName + std::string(" Win64"));
  160. return names;
  161. }
  162. bool SupportsToolset() const override { return true; }
  163. bool SupportsPlatform() const override { return true; }
  164. std::vector<std::string> GetKnownPlatforms() const override
  165. {
  166. std::vector<std::string> platforms;
  167. platforms.emplace_back("x64");
  168. platforms.emplace_back("Win32");
  169. platforms.emplace_back("ARM");
  170. platforms.emplace_back("ARM64");
  171. return platforms;
  172. }
  173. std::string GetDefaultPlatformName() const override { return "Win32"; }
  174. };
  175. std::unique_ptr<cmGlobalGeneratorFactory>
  176. cmGlobalVisualStudioVersionedGenerator::NewFactory15()
  177. {
  178. return std::unique_ptr<cmGlobalGeneratorFactory>(new Factory15);
  179. }
  180. static const char vs16generatorName[] = "Visual Studio 16 2019";
  181. // Map generator name without year to name with year.
  182. static const char* cmVS16GenName(const std::string& name, std::string& genName)
  183. {
  184. if (strncmp(name.c_str(), vs16generatorName,
  185. sizeof(vs16generatorName) - 6) != 0) {
  186. return 0;
  187. }
  188. const char* p = name.c_str() + sizeof(vs16generatorName) - 6;
  189. if (cmHasLiteralPrefix(p, " 2019")) {
  190. p += 5;
  191. }
  192. genName = std::string(vs16generatorName) + p;
  193. return p;
  194. }
  195. class cmGlobalVisualStudioVersionedGenerator::Factory16
  196. : public cmGlobalGeneratorFactory
  197. {
  198. public:
  199. std::unique_ptr<cmGlobalGenerator> CreateGlobalGenerator(
  200. const std::string& name, cmake* cm) const override
  201. {
  202. std::string genName;
  203. const char* p = cmVS16GenName(name, genName);
  204. if (!p) {
  205. return std::unique_ptr<cmGlobalGenerator>();
  206. }
  207. if (!*p) {
  208. return std::unique_ptr<cmGlobalGenerator>(
  209. new cmGlobalVisualStudioVersionedGenerator(
  210. cmGlobalVisualStudioGenerator::VS16, cm, genName, ""));
  211. }
  212. return std::unique_ptr<cmGlobalGenerator>();
  213. }
  214. void GetDocumentation(cmDocumentationEntry& entry) const override
  215. {
  216. entry.Name = std::string(vs16generatorName);
  217. entry.Brief = "Generates Visual Studio 2019 project files. "
  218. "Use -A option to specify architecture.";
  219. }
  220. std::vector<std::string> GetGeneratorNames() const override
  221. {
  222. std::vector<std::string> names;
  223. names.push_back(vs16generatorName);
  224. return names;
  225. }
  226. std::vector<std::string> GetGeneratorNamesWithPlatform() const override
  227. {
  228. return std::vector<std::string>();
  229. }
  230. bool SupportsToolset() const override { return true; }
  231. bool SupportsPlatform() const override { return true; }
  232. std::vector<std::string> GetKnownPlatforms() const override
  233. {
  234. std::vector<std::string> platforms;
  235. platforms.emplace_back("x64");
  236. platforms.emplace_back("Win32");
  237. platforms.emplace_back("ARM");
  238. platforms.emplace_back("ARM64");
  239. return platforms;
  240. }
  241. std::string GetDefaultPlatformName() const override
  242. {
  243. return VSHostPlatformName();
  244. }
  245. };
  246. std::unique_ptr<cmGlobalGeneratorFactory>
  247. cmGlobalVisualStudioVersionedGenerator::NewFactory16()
  248. {
  249. return std::unique_ptr<cmGlobalGeneratorFactory>(new Factory16);
  250. }
  251. cmGlobalVisualStudioVersionedGenerator::cmGlobalVisualStudioVersionedGenerator(
  252. VSVersion version, cmake* cm, const std::string& name,
  253. std::string const& platformInGeneratorName)
  254. : cmGlobalVisualStudio14Generator(cm, name, platformInGeneratorName)
  255. , vsSetupAPIHelper(VSVersionToMajor(version))
  256. {
  257. this->Version = version;
  258. this->ExpressEdition = false;
  259. this->DefaultPlatformToolset = VSVersionToToolset(this->Version);
  260. this->DefaultCLFlagTableName = VSVersionToToolset(this->Version);
  261. this->DefaultCSharpFlagTableName = VSVersionToToolset(this->Version);
  262. this->DefaultLinkFlagTableName = VSVersionToToolset(this->Version);
  263. if (this->Version >= cmGlobalVisualStudioGenerator::VS16) {
  264. this->DefaultPlatformName = VSHostPlatformName();
  265. this->DefaultPlatformToolsetHostArchitecture = VSHostArchitecture();
  266. }
  267. }
  268. bool cmGlobalVisualStudioVersionedGenerator::MatchesGeneratorName(
  269. const std::string& name) const
  270. {
  271. std::string genName;
  272. switch (this->Version) {
  273. case cmGlobalVisualStudioGenerator::VS9:
  274. case cmGlobalVisualStudioGenerator::VS10:
  275. case cmGlobalVisualStudioGenerator::VS11:
  276. case cmGlobalVisualStudioGenerator::VS12:
  277. case cmGlobalVisualStudioGenerator::VS14:
  278. break;
  279. case cmGlobalVisualStudioGenerator::VS15:
  280. if (cmVS15GenName(name, genName)) {
  281. return genName == this->GetName();
  282. }
  283. break;
  284. case cmGlobalVisualStudioGenerator::VS16:
  285. if (cmVS16GenName(name, genName)) {
  286. return genName == this->GetName();
  287. }
  288. break;
  289. }
  290. return false;
  291. }
  292. bool cmGlobalVisualStudioVersionedGenerator::SetGeneratorInstance(
  293. std::string const& i, cmMakefile* mf)
  294. {
  295. if (!i.empty()) {
  296. if (!this->vsSetupAPIHelper.SetVSInstance(i)) {
  297. std::ostringstream e;
  298. /* clang-format off */
  299. e <<
  300. "Generator\n"
  301. " " << this->GetName() << "\n"
  302. "could not find specified instance of Visual Studio:\n"
  303. " " << i;
  304. /* clang-format on */
  305. mf->IssueMessage(MessageType::FATAL_ERROR, e.str());
  306. return false;
  307. }
  308. }
  309. std::string vsInstance;
  310. if (!this->vsSetupAPIHelper.GetVSInstanceInfo(vsInstance)) {
  311. std::ostringstream e;
  312. /* clang-format off */
  313. e <<
  314. "Generator\n"
  315. " " << this->GetName() << "\n"
  316. "could not find any instance of Visual Studio.\n";
  317. /* clang-format on */
  318. mf->IssueMessage(MessageType::FATAL_ERROR, e.str());
  319. return false;
  320. }
  321. // Save the selected instance persistently.
  322. std::string genInstance = mf->GetSafeDefinition("CMAKE_GENERATOR_INSTANCE");
  323. if (vsInstance != genInstance) {
  324. this->CMakeInstance->AddCacheEntry(
  325. "CMAKE_GENERATOR_INSTANCE", vsInstance.c_str(),
  326. "Generator instance identifier.", cmStateEnums::INTERNAL);
  327. }
  328. return true;
  329. }
  330. bool cmGlobalVisualStudioVersionedGenerator::GetVSInstance(
  331. std::string& dir) const
  332. {
  333. return vsSetupAPIHelper.GetVSInstanceInfo(dir);
  334. }
  335. bool cmGlobalVisualStudioVersionedGenerator::GetVSInstanceVersion(
  336. unsigned long long& vsInstanceVersion) const
  337. {
  338. return vsSetupAPIHelper.GetVSInstanceVersion(vsInstanceVersion);
  339. }
  340. bool cmGlobalVisualStudioVersionedGenerator::IsDefaultToolset(
  341. const std::string& version) const
  342. {
  343. if (version.empty()) {
  344. return true;
  345. }
  346. std::string vcToolsetVersion;
  347. if (this->vsSetupAPIHelper.GetVCToolsetVersion(vcToolsetVersion)) {
  348. cmsys::RegularExpression regex("[0-9][0-9]\\.[0-9]+");
  349. if (regex.find(version) && regex.find(vcToolsetVersion)) {
  350. const auto majorMinorEnd = vcToolsetVersion.find('.', 3);
  351. const auto majorMinor = vcToolsetVersion.substr(0, majorMinorEnd);
  352. return version == majorMinor;
  353. }
  354. }
  355. return false;
  356. }
  357. bool cmGlobalVisualStudioVersionedGenerator::IsStdOutEncodingSupported() const
  358. {
  359. // Supported from Visual Studio 16.7 Preview 3.
  360. if (this->Version > cmGlobalVisualStudioGenerator::VSVersion::VS16) {
  361. return true;
  362. }
  363. if (this->Version < cmGlobalVisualStudioGenerator::VSVersion::VS16) {
  364. return false;
  365. }
  366. unsigned long long const vsInstanceVersion16_7_P2 = 4503631666610212;
  367. unsigned long long vsInstanceVersion;
  368. return (this->GetVSInstanceVersion(vsInstanceVersion) &&
  369. vsInstanceVersion > vsInstanceVersion16_7_P2);
  370. }
  371. std::string cmGlobalVisualStudioVersionedGenerator::GetAuxiliaryToolset() const
  372. {
  373. const char* version = this->GetPlatformToolsetVersion();
  374. if (version) {
  375. std::string instancePath;
  376. GetVSInstance(instancePath);
  377. std::string toolsetDir = instancePath + "/VC/Auxiliary/Build";
  378. char sep = '/';
  379. if (cmSystemTools::VersionCompareGreaterEq(version, "14.20")) {
  380. std::string toolsetDot =
  381. cmStrCat(toolsetDir, '.', version, "/Microsoft.VCToolsVersion.",
  382. version, ".props");
  383. if (cmSystemTools::PathExists(toolsetDot)) {
  384. sep = '.';
  385. }
  386. }
  387. std::string toolsetPath =
  388. cmStrCat(toolsetDir, sep, version, "/Microsoft.VCToolsVersion.", version,
  389. ".props");
  390. cmSystemTools::ConvertToUnixSlashes(toolsetPath);
  391. return toolsetPath;
  392. }
  393. return {};
  394. }
  395. bool cmGlobalVisualStudioVersionedGenerator::InitializeWindows(cmMakefile* mf)
  396. {
  397. // If the Win 8.1 SDK is installed then we can select a SDK matching
  398. // the target Windows version.
  399. if (this->IsWin81SDKInstalled()) {
  400. // VS 2019 does not default to 8.1 so specify it explicitly when needed.
  401. if (this->Version >= cmGlobalVisualStudioGenerator::VS16 &&
  402. !cmSystemTools::VersionCompareGreater(this->SystemVersion, "8.1")) {
  403. this->SetWindowsTargetPlatformVersion("8.1", mf);
  404. return true;
  405. }
  406. return cmGlobalVisualStudio14Generator::InitializeWindows(mf);
  407. }
  408. // Otherwise we must choose a Win 10 SDK even if we are not targeting
  409. // Windows 10.
  410. return this->SelectWindows10SDK(mf, false);
  411. }
  412. bool cmGlobalVisualStudioVersionedGenerator::SelectWindowsStoreToolset(
  413. std::string& toolset) const
  414. {
  415. if (cmHasLiteralPrefix(this->SystemVersion, "10.0")) {
  416. if (this->IsWindowsStoreToolsetInstalled() &&
  417. this->IsWindowsDesktopToolsetInstalled()) {
  418. toolset = VSVersionToToolset(this->Version);
  419. return true;
  420. } else {
  421. return false;
  422. }
  423. }
  424. return this->cmGlobalVisualStudio14Generator::SelectWindowsStoreToolset(
  425. toolset);
  426. }
  427. bool cmGlobalVisualStudioVersionedGenerator::IsWindowsDesktopToolsetInstalled()
  428. const
  429. {
  430. return vsSetupAPIHelper.IsVSInstalled();
  431. }
  432. bool cmGlobalVisualStudioVersionedGenerator::IsWindowsStoreToolsetInstalled()
  433. const
  434. {
  435. return vsSetupAPIHelper.IsWin10SDKInstalled();
  436. }
  437. bool cmGlobalVisualStudioVersionedGenerator::IsWin81SDKInstalled() const
  438. {
  439. // Does the VS installer tool know about one?
  440. if (vsSetupAPIHelper.IsWin81SDKInstalled()) {
  441. return true;
  442. }
  443. // Does the registry know about one (e.g. from VS 2015)?
  444. std::string win81Root;
  445. if (cmSystemTools::ReadRegistryValue(
  446. "HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\"
  447. "Windows Kits\\Installed Roots;KitsRoot81",
  448. win81Root, cmSystemTools::KeyWOW64_32) ||
  449. cmSystemTools::ReadRegistryValue(
  450. "HKEY_CURRENT_USER\\SOFTWARE\\Microsoft\\"
  451. "Windows Kits\\Installed Roots;KitsRoot81",
  452. win81Root, cmSystemTools::KeyWOW64_32)) {
  453. return cmSystemTools::FileExists(win81Root + "/include/um/windows.h",
  454. true);
  455. }
  456. return false;
  457. }
  458. std::string cmGlobalVisualStudioVersionedGenerator::GetWindows10SDKMaxVersion()
  459. const
  460. {
  461. return std::string();
  462. }
  463. std::string cmGlobalVisualStudioVersionedGenerator::FindMSBuildCommand()
  464. {
  465. std::string msbuild;
  466. // Ask Visual Studio Installer tool.
  467. std::string vs;
  468. if (vsSetupAPIHelper.GetVSInstanceInfo(vs)) {
  469. msbuild = vs + "/MSBuild/Current/Bin/MSBuild.exe";
  470. if (cmSystemTools::FileExists(msbuild)) {
  471. return msbuild;
  472. }
  473. msbuild = vs + "/MSBuild/15.0/Bin/MSBuild.exe";
  474. if (cmSystemTools::FileExists(msbuild)) {
  475. return msbuild;
  476. }
  477. }
  478. msbuild = "MSBuild.exe";
  479. return msbuild;
  480. }
  481. std::string cmGlobalVisualStudioVersionedGenerator::FindDevEnvCommand()
  482. {
  483. std::string devenv;
  484. // Ask Visual Studio Installer tool.
  485. std::string vs;
  486. if (vsSetupAPIHelper.GetVSInstanceInfo(vs)) {
  487. devenv = vs + "/Common7/IDE/devenv.com";
  488. if (cmSystemTools::FileExists(devenv)) {
  489. return devenv;
  490. }
  491. }
  492. devenv = "devenv.com";
  493. return devenv;
  494. }