cmGlobalVisualStudioVersionedGenerator.cxx 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  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* VSVersionToAndroidToolset(
  97. cmGlobalVisualStudioGenerator::VSVersion v)
  98. {
  99. switch (v) {
  100. case cmGlobalVisualStudioGenerator::VS9:
  101. case cmGlobalVisualStudioGenerator::VS10:
  102. case cmGlobalVisualStudioGenerator::VS11:
  103. case cmGlobalVisualStudioGenerator::VS12:
  104. return "";
  105. case cmGlobalVisualStudioGenerator::VS14:
  106. return "Clang_3_8";
  107. case cmGlobalVisualStudioGenerator::VS15:
  108. case cmGlobalVisualStudioGenerator::VS16:
  109. return "Clang_5_0";
  110. }
  111. return "";
  112. }
  113. static const char vs15generatorName[] = "Visual Studio 15 2017";
  114. // Map generator name without year to name with year.
  115. static const char* cmVS15GenName(const std::string& name, std::string& genName)
  116. {
  117. if (strncmp(name.c_str(), vs15generatorName,
  118. sizeof(vs15generatorName) - 6) != 0) {
  119. return 0;
  120. }
  121. const char* p = name.c_str() + sizeof(vs15generatorName) - 6;
  122. if (cmHasLiteralPrefix(p, " 2017")) {
  123. p += 5;
  124. }
  125. genName = std::string(vs15generatorName) + p;
  126. return p;
  127. }
  128. class cmGlobalVisualStudioVersionedGenerator::Factory15
  129. : public cmGlobalGeneratorFactory
  130. {
  131. public:
  132. std::unique_ptr<cmGlobalGenerator> CreateGlobalGenerator(
  133. const std::string& name, cmake* cm) const override
  134. {
  135. std::string genName;
  136. const char* p = cmVS15GenName(name, genName);
  137. if (!p) {
  138. return std::unique_ptr<cmGlobalGenerator>();
  139. }
  140. if (!*p) {
  141. return std::unique_ptr<cmGlobalGenerator>(
  142. new cmGlobalVisualStudioVersionedGenerator(
  143. cmGlobalVisualStudioGenerator::VS15, cm, genName, ""));
  144. }
  145. if (*p++ != ' ') {
  146. return std::unique_ptr<cmGlobalGenerator>();
  147. }
  148. if (strcmp(p, "Win64") == 0) {
  149. return std::unique_ptr<cmGlobalGenerator>(
  150. new cmGlobalVisualStudioVersionedGenerator(
  151. cmGlobalVisualStudioGenerator::VS15, cm, genName, "x64"));
  152. }
  153. if (strcmp(p, "ARM") == 0) {
  154. return std::unique_ptr<cmGlobalGenerator>(
  155. new cmGlobalVisualStudioVersionedGenerator(
  156. cmGlobalVisualStudioGenerator::VS15, cm, genName, "ARM"));
  157. }
  158. return std::unique_ptr<cmGlobalGenerator>();
  159. }
  160. void GetDocumentation(cmDocumentationEntry& entry) const override
  161. {
  162. entry.Name = std::string(vs15generatorName) + " [arch]";
  163. entry.Brief = "Generates Visual Studio 2017 project files. "
  164. "Optional [arch] can be \"Win64\" or \"ARM\".";
  165. }
  166. std::vector<std::string> GetGeneratorNames() const override
  167. {
  168. std::vector<std::string> names;
  169. names.push_back(vs15generatorName);
  170. return names;
  171. }
  172. std::vector<std::string> GetGeneratorNamesWithPlatform() const override
  173. {
  174. std::vector<std::string> names;
  175. names.push_back(vs15generatorName + std::string(" ARM"));
  176. names.push_back(vs15generatorName + std::string(" Win64"));
  177. return names;
  178. }
  179. bool SupportsToolset() const override { return true; }
  180. bool SupportsPlatform() const override { return true; }
  181. std::vector<std::string> GetKnownPlatforms() const override
  182. {
  183. std::vector<std::string> platforms;
  184. platforms.emplace_back("x64");
  185. platforms.emplace_back("Win32");
  186. platforms.emplace_back("ARM");
  187. platforms.emplace_back("ARM64");
  188. return platforms;
  189. }
  190. std::string GetDefaultPlatformName() const override { return "Win32"; }
  191. };
  192. std::unique_ptr<cmGlobalGeneratorFactory>
  193. cmGlobalVisualStudioVersionedGenerator::NewFactory15()
  194. {
  195. return std::unique_ptr<cmGlobalGeneratorFactory>(new Factory15);
  196. }
  197. static const char vs16generatorName[] = "Visual Studio 16 2019";
  198. // Map generator name without year to name with year.
  199. static const char* cmVS16GenName(const std::string& name, std::string& genName)
  200. {
  201. if (strncmp(name.c_str(), vs16generatorName,
  202. sizeof(vs16generatorName) - 6) != 0) {
  203. return 0;
  204. }
  205. const char* p = name.c_str() + sizeof(vs16generatorName) - 6;
  206. if (cmHasLiteralPrefix(p, " 2019")) {
  207. p += 5;
  208. }
  209. genName = std::string(vs16generatorName) + p;
  210. return p;
  211. }
  212. class cmGlobalVisualStudioVersionedGenerator::Factory16
  213. : public cmGlobalGeneratorFactory
  214. {
  215. public:
  216. std::unique_ptr<cmGlobalGenerator> CreateGlobalGenerator(
  217. const std::string& name, cmake* cm) const override
  218. {
  219. std::string genName;
  220. const char* p = cmVS16GenName(name, genName);
  221. if (!p) {
  222. return std::unique_ptr<cmGlobalGenerator>();
  223. }
  224. if (!*p) {
  225. return std::unique_ptr<cmGlobalGenerator>(
  226. new cmGlobalVisualStudioVersionedGenerator(
  227. cmGlobalVisualStudioGenerator::VS16, cm, genName, ""));
  228. }
  229. return std::unique_ptr<cmGlobalGenerator>();
  230. }
  231. void GetDocumentation(cmDocumentationEntry& entry) const override
  232. {
  233. entry.Name = std::string(vs16generatorName);
  234. entry.Brief = "Generates Visual Studio 2019 project files. "
  235. "Use -A option to specify architecture.";
  236. }
  237. std::vector<std::string> GetGeneratorNames() const override
  238. {
  239. std::vector<std::string> names;
  240. names.push_back(vs16generatorName);
  241. return names;
  242. }
  243. std::vector<std::string> GetGeneratorNamesWithPlatform() const override
  244. {
  245. return std::vector<std::string>();
  246. }
  247. bool SupportsToolset() const override { return true; }
  248. bool SupportsPlatform() const override { return true; }
  249. std::vector<std::string> GetKnownPlatforms() const override
  250. {
  251. std::vector<std::string> platforms;
  252. platforms.emplace_back("x64");
  253. platforms.emplace_back("Win32");
  254. platforms.emplace_back("ARM");
  255. platforms.emplace_back("ARM64");
  256. return platforms;
  257. }
  258. std::string GetDefaultPlatformName() const override
  259. {
  260. return VSHostPlatformName();
  261. }
  262. };
  263. std::unique_ptr<cmGlobalGeneratorFactory>
  264. cmGlobalVisualStudioVersionedGenerator::NewFactory16()
  265. {
  266. return std::unique_ptr<cmGlobalGeneratorFactory>(new Factory16);
  267. }
  268. cmGlobalVisualStudioVersionedGenerator::cmGlobalVisualStudioVersionedGenerator(
  269. VSVersion version, cmake* cm, const std::string& name,
  270. std::string const& platformInGeneratorName)
  271. : cmGlobalVisualStudio14Generator(cm, name, platformInGeneratorName)
  272. , vsSetupAPIHelper(VSVersionToMajor(version))
  273. {
  274. this->Version = version;
  275. this->ExpressEdition = false;
  276. this->DefaultPlatformToolset = VSVersionToToolset(this->Version);
  277. this->DefaultAndroidToolset = VSVersionToAndroidToolset(this->Version);
  278. this->DefaultCLFlagTableName = VSVersionToToolset(this->Version);
  279. this->DefaultCSharpFlagTableName = VSVersionToToolset(this->Version);
  280. this->DefaultLinkFlagTableName = VSVersionToToolset(this->Version);
  281. if (this->Version >= cmGlobalVisualStudioGenerator::VS16) {
  282. this->DefaultPlatformName = VSHostPlatformName();
  283. this->DefaultPlatformToolsetHostArchitecture = VSHostArchitecture();
  284. }
  285. }
  286. bool cmGlobalVisualStudioVersionedGenerator::MatchesGeneratorName(
  287. const std::string& name) const
  288. {
  289. std::string genName;
  290. switch (this->Version) {
  291. case cmGlobalVisualStudioGenerator::VS9:
  292. case cmGlobalVisualStudioGenerator::VS10:
  293. case cmGlobalVisualStudioGenerator::VS11:
  294. case cmGlobalVisualStudioGenerator::VS12:
  295. case cmGlobalVisualStudioGenerator::VS14:
  296. break;
  297. case cmGlobalVisualStudioGenerator::VS15:
  298. if (cmVS15GenName(name, genName)) {
  299. return genName == this->GetName();
  300. }
  301. break;
  302. case cmGlobalVisualStudioGenerator::VS16:
  303. if (cmVS16GenName(name, genName)) {
  304. return genName == this->GetName();
  305. }
  306. break;
  307. }
  308. return false;
  309. }
  310. bool cmGlobalVisualStudioVersionedGenerator::SetGeneratorInstance(
  311. std::string const& i, cmMakefile* mf)
  312. {
  313. if (!i.empty()) {
  314. if (!this->vsSetupAPIHelper.SetVSInstance(i)) {
  315. std::ostringstream e;
  316. /* clang-format off */
  317. e <<
  318. "Generator\n"
  319. " " << this->GetName() << "\n"
  320. "could not find specified instance of Visual Studio:\n"
  321. " " << i;
  322. /* clang-format on */
  323. mf->IssueMessage(MessageType::FATAL_ERROR, e.str());
  324. return false;
  325. }
  326. }
  327. std::string vsInstance;
  328. if (!this->vsSetupAPIHelper.GetVSInstanceInfo(vsInstance)) {
  329. std::ostringstream e;
  330. /* clang-format off */
  331. e <<
  332. "Generator\n"
  333. " " << this->GetName() << "\n"
  334. "could not find any instance of Visual Studio.\n";
  335. /* clang-format on */
  336. mf->IssueMessage(MessageType::FATAL_ERROR, e.str());
  337. return false;
  338. }
  339. // Save the selected instance persistently.
  340. std::string genInstance = mf->GetSafeDefinition("CMAKE_GENERATOR_INSTANCE");
  341. if (vsInstance != genInstance) {
  342. this->CMakeInstance->AddCacheEntry(
  343. "CMAKE_GENERATOR_INSTANCE", vsInstance.c_str(),
  344. "Generator instance identifier.", cmStateEnums::INTERNAL);
  345. }
  346. return true;
  347. }
  348. bool cmGlobalVisualStudioVersionedGenerator::GetVSInstance(
  349. std::string& dir) const
  350. {
  351. return vsSetupAPIHelper.GetVSInstanceInfo(dir);
  352. }
  353. bool cmGlobalVisualStudioVersionedGenerator::GetVSInstanceVersion(
  354. unsigned long long& vsInstanceVersion) const
  355. {
  356. return vsSetupAPIHelper.GetVSInstanceVersion(vsInstanceVersion);
  357. }
  358. bool cmGlobalVisualStudioVersionedGenerator::IsDefaultToolset(
  359. const std::string& version) const
  360. {
  361. if (version.empty()) {
  362. return true;
  363. }
  364. std::string vcToolsetVersion;
  365. if (this->vsSetupAPIHelper.GetVCToolsetVersion(vcToolsetVersion)) {
  366. cmsys::RegularExpression regex("[0-9][0-9]\\.[0-9]+");
  367. if (regex.find(version) && regex.find(vcToolsetVersion)) {
  368. const auto majorMinorEnd = vcToolsetVersion.find('.', 3);
  369. const auto majorMinor = vcToolsetVersion.substr(0, majorMinorEnd);
  370. return version == majorMinor;
  371. }
  372. }
  373. return false;
  374. }
  375. bool cmGlobalVisualStudioVersionedGenerator::IsStdOutEncodingSupported() const
  376. {
  377. // Supported from Visual Studio 16.7 Preview 3.
  378. if (this->Version > cmGlobalVisualStudioGenerator::VSVersion::VS16) {
  379. return true;
  380. }
  381. if (this->Version < cmGlobalVisualStudioGenerator::VSVersion::VS16) {
  382. return false;
  383. }
  384. unsigned long long const vsInstanceVersion16_7_P2 = 4503631666610212;
  385. unsigned long long vsInstanceVersion;
  386. return (this->GetVSInstanceVersion(vsInstanceVersion) &&
  387. vsInstanceVersion > vsInstanceVersion16_7_P2);
  388. }
  389. const char*
  390. cmGlobalVisualStudioVersionedGenerator::GetAndroidApplicationTypeRevision()
  391. const
  392. {
  393. switch (this->Version) {
  394. case cmGlobalVisualStudioGenerator::VS9:
  395. case cmGlobalVisualStudioGenerator::VS10:
  396. case cmGlobalVisualStudioGenerator::VS11:
  397. case cmGlobalVisualStudioGenerator::VS12:
  398. return "";
  399. case cmGlobalVisualStudioGenerator::VS14:
  400. return "2.0";
  401. case cmGlobalVisualStudioGenerator::VS15:
  402. case cmGlobalVisualStudioGenerator::VS16:
  403. return "3.0";
  404. }
  405. return "";
  406. }
  407. std::string cmGlobalVisualStudioVersionedGenerator::GetAuxiliaryToolset() const
  408. {
  409. const char* version = this->GetPlatformToolsetVersion();
  410. if (version) {
  411. std::string instancePath;
  412. GetVSInstance(instancePath);
  413. std::string toolsetDir = instancePath + "/VC/Auxiliary/Build";
  414. char sep = '/';
  415. if (cmSystemTools::VersionCompareGreaterEq(version, "14.20")) {
  416. std::string toolsetDot =
  417. cmStrCat(toolsetDir, '.', version, "/Microsoft.VCToolsVersion.",
  418. version, ".props");
  419. if (cmSystemTools::PathExists(toolsetDot)) {
  420. sep = '.';
  421. }
  422. }
  423. std::string toolsetPath =
  424. cmStrCat(toolsetDir, sep, version, "/Microsoft.VCToolsVersion.", version,
  425. ".props");
  426. cmSystemTools::ConvertToUnixSlashes(toolsetPath);
  427. return toolsetPath;
  428. }
  429. return {};
  430. }
  431. bool cmGlobalVisualStudioVersionedGenerator::InitializeWindows(cmMakefile* mf)
  432. {
  433. // If the Win 8.1 SDK is installed then we can select a SDK matching
  434. // the target Windows version.
  435. if (this->IsWin81SDKInstalled()) {
  436. // VS 2019 does not default to 8.1 so specify it explicitly when needed.
  437. if (this->Version >= cmGlobalVisualStudioGenerator::VS16 &&
  438. !cmSystemTools::VersionCompareGreater(this->SystemVersion, "8.1")) {
  439. this->SetWindowsTargetPlatformVersion("8.1", mf);
  440. return true;
  441. }
  442. return cmGlobalVisualStudio14Generator::InitializeWindows(mf);
  443. }
  444. // Otherwise we must choose a Win 10 SDK even if we are not targeting
  445. // Windows 10.
  446. return this->SelectWindows10SDK(mf, false);
  447. }
  448. bool cmGlobalVisualStudioVersionedGenerator::SelectWindowsStoreToolset(
  449. std::string& toolset) const
  450. {
  451. if (cmHasLiteralPrefix(this->SystemVersion, "10.0")) {
  452. if (this->IsWindowsStoreToolsetInstalled() &&
  453. this->IsWindowsDesktopToolsetInstalled()) {
  454. toolset = VSVersionToToolset(this->Version);
  455. return true;
  456. } else {
  457. return false;
  458. }
  459. }
  460. return this->cmGlobalVisualStudio14Generator::SelectWindowsStoreToolset(
  461. toolset);
  462. }
  463. bool cmGlobalVisualStudioVersionedGenerator::IsWindowsDesktopToolsetInstalled()
  464. const
  465. {
  466. return vsSetupAPIHelper.IsVSInstalled();
  467. }
  468. bool cmGlobalVisualStudioVersionedGenerator::IsWindowsStoreToolsetInstalled()
  469. const
  470. {
  471. return vsSetupAPIHelper.IsWin10SDKInstalled();
  472. }
  473. bool cmGlobalVisualStudioVersionedGenerator::IsWin81SDKInstalled() const
  474. {
  475. // Does the VS installer tool know about one?
  476. if (vsSetupAPIHelper.IsWin81SDKInstalled()) {
  477. return true;
  478. }
  479. // Does the registry know about one (e.g. from VS 2015)?
  480. std::string win81Root;
  481. if (cmSystemTools::ReadRegistryValue(
  482. "HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\"
  483. "Windows Kits\\Installed Roots;KitsRoot81",
  484. win81Root, cmSystemTools::KeyWOW64_32) ||
  485. cmSystemTools::ReadRegistryValue(
  486. "HKEY_CURRENT_USER\\SOFTWARE\\Microsoft\\"
  487. "Windows Kits\\Installed Roots;KitsRoot81",
  488. win81Root, cmSystemTools::KeyWOW64_32)) {
  489. return cmSystemTools::FileExists(win81Root + "/include/um/windows.h",
  490. true);
  491. }
  492. return false;
  493. }
  494. std::string cmGlobalVisualStudioVersionedGenerator::GetWindows10SDKMaxVersion()
  495. const
  496. {
  497. return std::string();
  498. }
  499. std::string cmGlobalVisualStudioVersionedGenerator::FindMSBuildCommand()
  500. {
  501. std::string msbuild;
  502. // Ask Visual Studio Installer tool.
  503. std::string vs;
  504. if (vsSetupAPIHelper.GetVSInstanceInfo(vs)) {
  505. msbuild = vs + "/MSBuild/Current/Bin/MSBuild.exe";
  506. if (cmSystemTools::FileExists(msbuild)) {
  507. return msbuild;
  508. }
  509. msbuild = vs + "/MSBuild/15.0/Bin/MSBuild.exe";
  510. if (cmSystemTools::FileExists(msbuild)) {
  511. return msbuild;
  512. }
  513. }
  514. msbuild = "MSBuild.exe";
  515. return msbuild;
  516. }
  517. std::string cmGlobalVisualStudioVersionedGenerator::FindDevEnvCommand()
  518. {
  519. std::string devenv;
  520. // Ask Visual Studio Installer tool.
  521. std::string vs;
  522. if (vsSetupAPIHelper.GetVSInstanceInfo(vs)) {
  523. devenv = vs + "/Common7/IDE/devenv.com";
  524. if (cmSystemTools::FileExists(devenv)) {
  525. return devenv;
  526. }
  527. }
  528. devenv = "devenv.com";
  529. return devenv;
  530. }