cmGlobalVisualStudio8Generator.cxx 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  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 "cmGlobalVisualStudio8Generator.h"
  4. #include "cmDocumentationEntry.h"
  5. #include "cmGeneratedFileStream.h"
  6. #include "cmGeneratorTarget.h"
  7. #include "cmLocalVisualStudio7Generator.h"
  8. #include "cmMakefile.h"
  9. #include "cmMessageType.h"
  10. #include "cmSourceFile.h"
  11. #include "cmVisualStudioWCEPlatformParser.h"
  12. #include "cmake.h"
  13. cmGlobalVisualStudio8Generator::cmGlobalVisualStudio8Generator(
  14. cmake* cm, const std::string& name,
  15. std::string const& platformInGeneratorName)
  16. : cmGlobalVisualStudio71Generator(cm, platformInGeneratorName)
  17. {
  18. this->ProjectConfigurationSectionName = "ProjectConfigurationPlatforms";
  19. this->Name = name;
  20. this->ExtraFlagTable = this->GetExtraFlagTableVS8();
  21. }
  22. std::string cmGlobalVisualStudio8Generator::FindDevEnvCommand()
  23. {
  24. // First look for VCExpress.
  25. std::string vsxcmd;
  26. std::string vsxkey = "HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\VCExpress\\";
  27. vsxkey += this->GetIDEVersion();
  28. vsxkey += ";InstallDir";
  29. if (cmSystemTools::ReadRegistryValue(vsxkey.c_str(), vsxcmd,
  30. cmSystemTools::KeyWOW64_32)) {
  31. cmSystemTools::ConvertToUnixSlashes(vsxcmd);
  32. vsxcmd += "/VCExpress.exe";
  33. return vsxcmd;
  34. }
  35. // Now look for devenv.
  36. return this->cmGlobalVisualStudio71Generator::FindDevEnvCommand();
  37. }
  38. void cmGlobalVisualStudio8Generator::EnableLanguage(
  39. std::vector<std::string> const& lang, cmMakefile* mf, bool optional)
  40. {
  41. for (std::string const& it : lang) {
  42. if (it == "ASM_MASM") {
  43. this->MasmEnabled = true;
  44. }
  45. }
  46. this->AddPlatformDefinitions(mf);
  47. cmGlobalVisualStudio7Generator::EnableLanguage(lang, mf, optional);
  48. }
  49. void cmGlobalVisualStudio8Generator::AddPlatformDefinitions(cmMakefile* mf)
  50. {
  51. if (this->TargetsWindowsCE()) {
  52. mf->AddDefinition("CMAKE_VS_WINCE_VERSION",
  53. this->WindowsCEVersion.c_str());
  54. }
  55. }
  56. bool cmGlobalVisualStudio8Generator::SetGeneratorPlatform(std::string const& p,
  57. cmMakefile* mf)
  58. {
  59. if (!this->PlatformInGeneratorName) {
  60. this->GeneratorPlatform = p;
  61. return this->cmGlobalVisualStudio7Generator::SetGeneratorPlatform("", mf);
  62. } else {
  63. return this->cmGlobalVisualStudio7Generator::SetGeneratorPlatform(p, mf);
  64. }
  65. }
  66. std::string cmGlobalVisualStudio8Generator::GetGenerateStampList()
  67. {
  68. return "generate.stamp.list";
  69. }
  70. void cmGlobalVisualStudio8Generator::Configure()
  71. {
  72. this->cmGlobalVisualStudio7Generator::Configure();
  73. }
  74. bool cmGlobalVisualStudio8Generator::UseFolderProperty() const
  75. {
  76. return IsExpressEdition() ? false : cmGlobalGenerator::UseFolderProperty();
  77. }
  78. bool cmGlobalVisualStudio8Generator::AddCheckTarget()
  79. {
  80. // Add a special target on which all other targets depend that
  81. // checks the build system and optionally re-runs CMake.
  82. // Skip the target if no regeneration is to be done.
  83. if (this->GlobalSettingIsOn("CMAKE_SUPPRESS_REGENERATION")) {
  84. return false;
  85. }
  86. const char* no_working_directory = nullptr;
  87. std::vector<std::string> no_depends;
  88. std::vector<cmLocalGenerator*> const& generators = this->LocalGenerators;
  89. cmLocalVisualStudio7Generator* lg =
  90. static_cast<cmLocalVisualStudio7Generator*>(generators[0]);
  91. cmMakefile* mf = lg->GetMakefile();
  92. cmCustomCommandLines noCommandLines;
  93. cmTarget* tgt = mf->AddUtilityCommand(
  94. CMAKE_CHECK_BUILD_SYSTEM_TARGET, cmMakefile::TargetOrigin::Generator,
  95. false, no_working_directory, no_depends, noCommandLines);
  96. cmGeneratorTarget* gt = new cmGeneratorTarget(tgt, lg);
  97. lg->AddGeneratorTarget(gt);
  98. // Organize in the "predefined targets" folder:
  99. //
  100. if (this->UseFolderProperty()) {
  101. tgt->SetProperty("FOLDER", this->GetPredefinedTargetsFolder());
  102. }
  103. // Create a list of all stamp files for this project.
  104. std::vector<std::string> stamps;
  105. std::string stampList = cmake::GetCMakeFilesDirectoryPostSlash();
  106. stampList += cmGlobalVisualStudio8Generator::GetGenerateStampList();
  107. {
  108. std::string stampListFile =
  109. generators[0]->GetMakefile()->GetCurrentBinaryDirectory();
  110. stampListFile += "/";
  111. stampListFile += stampList;
  112. std::string stampFile;
  113. cmGeneratedFileStream fout(stampListFile.c_str());
  114. for (cmLocalGenerator const* gi : generators) {
  115. stampFile = gi->GetMakefile()->GetCurrentBinaryDirectory();
  116. stampFile += "/";
  117. stampFile += cmake::GetCMakeFilesDirectoryPostSlash();
  118. stampFile += "generate.stamp";
  119. fout << stampFile << "\n";
  120. stamps.push_back(stampFile);
  121. }
  122. }
  123. // Add a custom rule to re-run CMake if any input files changed.
  124. {
  125. // Collect the input files used to generate all targets in this
  126. // project.
  127. std::vector<std::string> listFiles;
  128. for (unsigned int j = 0; j < generators.size(); ++j) {
  129. cmMakefile* lmf = generators[j]->GetMakefile();
  130. listFiles.insert(listFiles.end(), lmf->GetListFiles().begin(),
  131. lmf->GetListFiles().end());
  132. }
  133. // Add a custom prebuild target to run the VerifyGlobs script.
  134. cmake* cm = this->GetCMakeInstance();
  135. if (cm->DoWriteGlobVerifyTarget()) {
  136. cmCustomCommandLine verifyCommandLine;
  137. verifyCommandLine.push_back(cmSystemTools::GetCMakeCommand());
  138. verifyCommandLine.push_back("-P");
  139. verifyCommandLine.push_back(cm->GetGlobVerifyScript());
  140. cmCustomCommandLines verifyCommandLines;
  141. verifyCommandLines.push_back(verifyCommandLine);
  142. std::vector<std::string> byproducts;
  143. byproducts.push_back(cm->GetGlobVerifyStamp());
  144. mf->AddCustomCommandToTarget(CMAKE_CHECK_BUILD_SYSTEM_TARGET, byproducts,
  145. no_depends, verifyCommandLines,
  146. cmTarget::PRE_BUILD, "Checking File Globs",
  147. no_working_directory, false);
  148. // Ensure ZERO_CHECK always runs in Visual Studio using MSBuild,
  149. // otherwise the prebuild command will not be run.
  150. tgt->SetProperty("VS_GLOBAL_DisableFastUpToDateCheck", "true");
  151. listFiles.push_back(cm->GetGlobVerifyStamp());
  152. }
  153. // Sort the list of input files and remove duplicates.
  154. std::sort(listFiles.begin(), listFiles.end(), std::less<std::string>());
  155. std::vector<std::string>::iterator new_end =
  156. std::unique(listFiles.begin(), listFiles.end());
  157. listFiles.erase(new_end, listFiles.end());
  158. // Create a rule to re-run CMake.
  159. cmCustomCommandLine commandLine;
  160. commandLine.push_back(cmSystemTools::GetCMakeCommand());
  161. std::string argS = "-S";
  162. argS += lg->GetSourceDirectory();
  163. commandLine.push_back(argS);
  164. std::string argB = "-B";
  165. argB += lg->GetBinaryDirectory();
  166. commandLine.push_back(argB);
  167. commandLine.push_back("--check-stamp-list");
  168. commandLine.push_back(stampList.c_str());
  169. commandLine.push_back("--vs-solution-file");
  170. std::string const sln = std::string(lg->GetBinaryDirectory()) + "/" +
  171. lg->GetProjectName() + ".sln";
  172. commandLine.push_back(sln);
  173. cmCustomCommandLines commandLines;
  174. commandLines.push_back(commandLine);
  175. // Add the rule. Note that we cannot use the CMakeLists.txt
  176. // file as the main dependency because it would get
  177. // overwritten by the CreateVCProjBuildRule.
  178. // (this could be avoided with per-target source files)
  179. std::string no_main_dependency;
  180. std::vector<std::string> no_byproducts;
  181. if (cmSourceFile* file = mf->AddCustomCommandToOutput(
  182. stamps, no_byproducts, listFiles, no_main_dependency, commandLines,
  183. "Checking Build System", no_working_directory, true, false)) {
  184. gt->AddSource(file->GetFullPath());
  185. } else {
  186. cmSystemTools::Error("Error adding rule for ", stamps[0].c_str());
  187. }
  188. }
  189. return true;
  190. }
  191. void cmGlobalVisualStudio8Generator::AddExtraIDETargets()
  192. {
  193. cmGlobalVisualStudio7Generator::AddExtraIDETargets();
  194. if (this->AddCheckTarget()) {
  195. for (unsigned int i = 0; i < this->LocalGenerators.size(); ++i) {
  196. const std::vector<cmGeneratorTarget*>& tgts =
  197. this->LocalGenerators[i]->GetGeneratorTargets();
  198. // All targets depend on the build-system check target.
  199. for (cmGeneratorTarget const* ti : tgts) {
  200. if (ti->GetName() != CMAKE_CHECK_BUILD_SYSTEM_TARGET) {
  201. ti->Target->AddUtility(CMAKE_CHECK_BUILD_SYSTEM_TARGET);
  202. }
  203. }
  204. }
  205. }
  206. }
  207. void cmGlobalVisualStudio8Generator::WriteSolutionConfigurations(
  208. std::ostream& fout, std::vector<std::string> const& configs)
  209. {
  210. fout << "\tGlobalSection(SolutionConfigurationPlatforms) = preSolution\n";
  211. for (std::string const& i : configs) {
  212. fout << "\t\t" << i << "|" << this->GetPlatformName() << " = " << i << "|"
  213. << this->GetPlatformName() << "\n";
  214. }
  215. fout << "\tEndGlobalSection\n";
  216. }
  217. void cmGlobalVisualStudio8Generator::WriteProjectConfigurations(
  218. std::ostream& fout, const std::string& name, cmGeneratorTarget const& target,
  219. std::vector<std::string> const& configs,
  220. const std::set<std::string>& configsPartOfDefaultBuild,
  221. std::string const& platformMapping)
  222. {
  223. std::string guid = this->GetGUID(name);
  224. for (std::string const& i : configs) {
  225. std::vector<std::string> mapConfig;
  226. const char* dstConfig = i.c_str();
  227. if (target.GetProperty("EXTERNAL_MSPROJECT")) {
  228. if (const char* m = target.GetProperty("MAP_IMPORTED_CONFIG_" +
  229. cmSystemTools::UpperCase(i))) {
  230. cmSystemTools::ExpandListArgument(m, mapConfig);
  231. if (!mapConfig.empty()) {
  232. dstConfig = mapConfig[0].c_str();
  233. }
  234. }
  235. }
  236. fout << "\t\t{" << guid << "}." << i << "|" << this->GetPlatformName()
  237. << ".ActiveCfg = " << dstConfig << "|"
  238. << (!platformMapping.empty() ? platformMapping
  239. : this->GetPlatformName())
  240. << "\n";
  241. std::set<std::string>::const_iterator ci =
  242. configsPartOfDefaultBuild.find(i);
  243. if (!(ci == configsPartOfDefaultBuild.end())) {
  244. fout << "\t\t{" << guid << "}." << i << "|" << this->GetPlatformName()
  245. << ".Build.0 = " << dstConfig << "|"
  246. << (!platformMapping.empty() ? platformMapping
  247. : this->GetPlatformName())
  248. << "\n";
  249. }
  250. if (this->NeedsDeploy(target.GetType())) {
  251. fout << "\t\t{" << guid << "}." << i << "|" << this->GetPlatformName()
  252. << ".Deploy.0 = " << dstConfig << "|"
  253. << (!platformMapping.empty() ? platformMapping
  254. : this->GetPlatformName())
  255. << "\n";
  256. }
  257. }
  258. }
  259. bool cmGlobalVisualStudio8Generator::NeedsDeploy(
  260. cmStateEnums::TargetType type) const
  261. {
  262. bool needsDeploy =
  263. (type == cmStateEnums::EXECUTABLE || type == cmStateEnums::SHARED_LIBRARY);
  264. return this->TargetsWindowsCE() && needsDeploy;
  265. }
  266. bool cmGlobalVisualStudio8Generator::ComputeTargetDepends()
  267. {
  268. // Skip over the cmGlobalVisualStudioGenerator implementation!
  269. // We do not need the support that VS <= 7.1 needs.
  270. return this->cmGlobalGenerator::ComputeTargetDepends();
  271. }
  272. void cmGlobalVisualStudio8Generator::WriteProjectDepends(
  273. std::ostream& fout, const std::string&, const char*,
  274. cmGeneratorTarget const* gt)
  275. {
  276. TargetDependSet const& unordered = this->GetTargetDirectDepends(gt);
  277. OrderedTargetDependSet depends(unordered, std::string());
  278. for (cmTargetDepend const& i : depends) {
  279. if (i->GetType() == cmStateEnums::INTERFACE_LIBRARY) {
  280. continue;
  281. }
  282. std::string guid = this->GetGUID(i->GetName());
  283. fout << "\t\t{" << guid << "} = {" << guid << "}\n";
  284. }
  285. }
  286. bool cmGlobalVisualStudio8Generator::NeedLinkLibraryDependencies(
  287. cmGeneratorTarget* target)
  288. {
  289. // Look for utility dependencies that magically link.
  290. for (BT<std::string> const& ui : target->GetUtilities()) {
  291. if (cmGeneratorTarget* depTarget =
  292. target->GetLocalGenerator()->FindGeneratorTargetToUse(ui.Value)) {
  293. if (depTarget->GetType() != cmStateEnums::INTERFACE_LIBRARY &&
  294. depTarget->GetProperty("EXTERNAL_MSPROJECT")) {
  295. // This utility dependency names an external .vcproj target.
  296. // We use LinkLibraryDependencies="true" to link to it without
  297. // predicting the .lib file location or name.
  298. return true;
  299. }
  300. }
  301. }
  302. return false;
  303. }
  304. static cmVS7FlagTable cmVS8ExtraFlagTable[] = {
  305. { "CallingConvention", "Gd", "cdecl", "0", 0 },
  306. { "CallingConvention", "Gr", "fastcall", "1", 0 },
  307. { "CallingConvention", "Gz", "stdcall", "2", 0 },
  308. { "Detect64BitPortabilityProblems", "Wp64",
  309. "Detect 64Bit Portability Problems", "true", 0 },
  310. { "ErrorReporting", "errorReport:prompt", "Report immediately", "1", 0 },
  311. { "ErrorReporting", "errorReport:queue", "Queue for next login", "2", 0 },
  312. // Precompiled header and related options. Note that the
  313. // UsePrecompiledHeader entries are marked as "Continue" so that the
  314. // corresponding PrecompiledHeaderThrough entry can be found.
  315. { "UsePrecompiledHeader", "Yu", "Use Precompiled Header", "2",
  316. cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue },
  317. { "PrecompiledHeaderThrough", "Yu", "Precompiled Header Name", "",
  318. cmVS7FlagTable::UserValueRequired },
  319. // There is no YX option in the VS8 IDE.
  320. // Exception handling mode. If no entries match, it will be FALSE.
  321. { "ExceptionHandling", "GX", "enable c++ exceptions", "1", 0 },
  322. { "ExceptionHandling", "EHsc", "enable c++ exceptions", "1", 0 },
  323. { "ExceptionHandling", "EHa", "enable SEH exceptions", "2", 0 },
  324. { "EnablePREfast", "analyze", "", "true", 0 },
  325. { "EnablePREfast", "analyze-", "", "false", 0 },
  326. // Language options
  327. { "TreatWChar_tAsBuiltInType", "Zc:wchar_t", "wchar_t is a built-in type",
  328. "true", 0 },
  329. { "TreatWChar_tAsBuiltInType", "Zc:wchar_t-",
  330. "wchar_t is not a built-in type", "false", 0 },
  331. { "", "", "", "", 0 }
  332. };
  333. cmIDEFlagTable const* cmGlobalVisualStudio8Generator::GetExtraFlagTableVS8()
  334. {
  335. return cmVS8ExtraFlagTable;
  336. }