cmGlobalVisualStudioVersionedGenerator.cxx 14 KB

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