cmGlobalVisualStudio7Generator.cxx 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783
  1. /*=========================================================================
  2. Program: CMake - Cross-Platform Makefile Generator
  3. Module: $RCSfile$
  4. Language: C++
  5. Date: $Date$
  6. Version: $Revision$
  7. Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
  8. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
  9. This software is distributed WITHOUT ANY WARRANTY; without even
  10. the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
  11. PURPOSE. See the above copyright notices for more information.
  12. =========================================================================*/
  13. #include "windows.h" // this must be first to define GetCurrentDirectory
  14. #include "cmGlobalVisualStudio7Generator.h"
  15. #include "cmGeneratedFileStream.h"
  16. #include "cmLocalVisualStudio7Generator.h"
  17. #include "cmMakefile.h"
  18. #include "cmake.h"
  19. cmGlobalVisualStudio7Generator::cmGlobalVisualStudio7Generator()
  20. {
  21. this->FindMakeProgramFile = "CMakeVS7FindMake.cmake";
  22. }
  23. void cmGlobalVisualStudio7Generator
  24. ::EnableLanguage(std::vector<std::string>const & lang,
  25. cmMakefile *mf, bool optional)
  26. {
  27. mf->AddDefinition("CMAKE_GENERATOR_CC", "cl");
  28. mf->AddDefinition("CMAKE_GENERATOR_CXX", "cl");
  29. mf->AddDefinition("CMAKE_GENERATOR_RC", "rc");
  30. mf->AddDefinition("CMAKE_GENERATOR_NO_COMPILER_ENV", "1");
  31. mf->AddDefinition("CMAKE_GENERATOR_FC", "ifort");
  32. this->AddPlatformDefinitions(mf);
  33. // Create list of configurations requested by user's cache, if any.
  34. this->cmGlobalGenerator::EnableLanguage(lang, mf, optional);
  35. this->GenerateConfigurations(mf);
  36. // if this environment variable is set, then copy it to
  37. // a static cache entry. It will be used by
  38. // cmLocalGenerator::ConstructScript, to add an extra PATH
  39. // to all custom commands. This is because the VS IDE
  40. // does not use the environment it is run in, and this allows
  41. // for running commands and using dll's that the IDE environment
  42. // does not know about.
  43. const char* extraPath = cmSystemTools::GetEnv("CMAKE_MSVCIDE_RUN_PATH");
  44. if(extraPath)
  45. {
  46. mf->AddCacheDefinition
  47. ("CMAKE_MSVCIDE_RUN_PATH", extraPath,
  48. "Saved environment variable CMAKE_MSVCIDE_RUN_PATH",
  49. cmCacheManager::STATIC);
  50. }
  51. }
  52. void cmGlobalVisualStudio7Generator::AddPlatformDefinitions(cmMakefile* mf)
  53. {
  54. mf->AddDefinition("MSVC70", "1");
  55. }
  56. std::string cmGlobalVisualStudio7Generator
  57. ::GenerateBuildCommand(const char* makeProgram,
  58. const char *projectName,
  59. const char* additionalOptions, const char *targetName,
  60. const char* config, bool ignoreErrors, bool)
  61. {
  62. // Ingoring errors is not implemented in visual studio 6
  63. (void) ignoreErrors;
  64. // now build the test
  65. std::string makeCommand =
  66. cmSystemTools::ConvertToOutputPath(makeProgram);
  67. std::string lowerCaseCommand = makeCommand;
  68. cmSystemTools::LowerCase(lowerCaseCommand);
  69. // if there are spaces in the makeCommand, assume a full path
  70. // and convert it to a path with no spaces in it as the
  71. // RunSingleCommand does not like spaces
  72. #if defined(_WIN32) && !defined(__CYGWIN__)
  73. if(makeCommand.find(' ') != std::string::npos)
  74. {
  75. cmSystemTools::GetShortPath(makeCommand.c_str(), makeCommand);
  76. }
  77. #endif
  78. makeCommand += " ";
  79. makeCommand += projectName;
  80. makeCommand += ".sln ";
  81. bool clean = false;
  82. if ( targetName && strcmp(targetName, "clean") == 0 )
  83. {
  84. clean = true;
  85. targetName = "ALL_BUILD";
  86. }
  87. if(clean)
  88. {
  89. makeCommand += "/clean ";
  90. }
  91. else
  92. {
  93. makeCommand += "/build ";
  94. }
  95. if(config && strlen(config))
  96. {
  97. makeCommand += config;
  98. }
  99. else
  100. {
  101. makeCommand += "Debug";
  102. }
  103. makeCommand += " /project ";
  104. if (targetName && strlen(targetName))
  105. {
  106. makeCommand += targetName;
  107. }
  108. else
  109. {
  110. makeCommand += "ALL_BUILD";
  111. }
  112. if ( additionalOptions )
  113. {
  114. makeCommand += " ";
  115. makeCommand += additionalOptions;
  116. }
  117. return makeCommand;
  118. }
  119. ///! Create a local generator appropriate to this Global Generator
  120. cmLocalGenerator *cmGlobalVisualStudio7Generator::CreateLocalGenerator()
  121. {
  122. cmLocalVisualStudio7Generator *lg = new cmLocalVisualStudio7Generator;
  123. lg->SetExtraFlagTable(this->GetExtraFlagTableVS7());
  124. lg->SetGlobalGenerator(this);
  125. return lg;
  126. }
  127. void cmGlobalVisualStudio7Generator::GenerateConfigurations(cmMakefile* mf)
  128. {
  129. // process the configurations
  130. const char* ct
  131. = this->CMakeInstance->GetCacheDefinition("CMAKE_CONFIGURATION_TYPES");
  132. if ( ct )
  133. {
  134. std::vector<std::string> argsOut;
  135. cmSystemTools::ExpandListArgument(ct, argsOut);
  136. for(std::vector<std::string>::iterator i = argsOut.begin();
  137. i != argsOut.end(); ++i)
  138. {
  139. if(std::find(this->Configurations.begin(),
  140. this->Configurations.end(),
  141. *i) == this->Configurations.end())
  142. {
  143. this->Configurations.push_back(*i);
  144. }
  145. }
  146. }
  147. // default to at least Debug and Release
  148. if(this->Configurations.size() == 0)
  149. {
  150. this->Configurations.push_back("Debug");
  151. this->Configurations.push_back("Release");
  152. }
  153. // Reset the entry to have a semi-colon separated list.
  154. std::string configs = this->Configurations[0];
  155. for(unsigned int i=1; i < this->Configurations.size(); ++i)
  156. {
  157. configs += ";";
  158. configs += this->Configurations[i];
  159. }
  160. mf->AddCacheDefinition(
  161. "CMAKE_CONFIGURATION_TYPES",
  162. configs.c_str(),
  163. "Semicolon separated list of supported configuration types, "
  164. "only supports Debug, Release, MinSizeRel, and RelWithDebInfo, "
  165. "anything else will be ignored.",
  166. cmCacheManager::STRING);
  167. }
  168. void cmGlobalVisualStudio7Generator::Generate()
  169. {
  170. // first do the superclass method
  171. this->cmGlobalVisualStudioGenerator::Generate();
  172. // Now write out the DSW
  173. this->OutputSLNFile();
  174. // If any solution or project files changed during the generation,
  175. // tell Visual Studio to reload them...
  176. if(!cmSystemTools::GetErrorOccuredFlag())
  177. {
  178. this->CallVisualStudioMacro(MacroReload);
  179. }
  180. }
  181. void cmGlobalVisualStudio7Generator
  182. ::OutputSLNFile(cmLocalGenerator* root,
  183. std::vector<cmLocalGenerator*>& generators)
  184. {
  185. if(generators.size() == 0)
  186. {
  187. return;
  188. }
  189. this->CurrentProject = root->GetMakefile()->GetProjectName();
  190. std::string fname = root->GetMakefile()->GetStartOutputDirectory();
  191. fname += "/";
  192. fname += root->GetMakefile()->GetProjectName();
  193. fname += ".sln";
  194. cmGeneratedFileStream fout(fname.c_str());
  195. fout.SetCopyIfDifferent(true);
  196. if(!fout)
  197. {
  198. return;
  199. }
  200. this->WriteSLNFile(fout, root, generators);
  201. if (fout.Close())
  202. {
  203. this->FileReplacedDuringGenerate(fname);
  204. }
  205. }
  206. // output the SLN file
  207. void cmGlobalVisualStudio7Generator::OutputSLNFile()
  208. {
  209. std::map<cmStdString, std::vector<cmLocalGenerator*> >::iterator it;
  210. for(it = this->ProjectMap.begin(); it!= this->ProjectMap.end(); ++it)
  211. {
  212. this->OutputSLNFile(it->second[0], it->second);
  213. }
  214. }
  215. void cmGlobalVisualStudio7Generator::AddAllBuildDepends(
  216. cmLocalGenerator* root,
  217. cmTarget* target,
  218. cmGlobalGenerator::TargetDependSet& originalTargets)
  219. {
  220. // if this is the special ALL_BUILD utility, then
  221. // make it depend on every other non UTILITY project.
  222. for(cmGlobalGenerator::TargetDependSet::iterator ot =
  223. originalTargets.begin(); ot != originalTargets.end(); ++ot)
  224. {
  225. cmTarget* t = const_cast<cmTarget*>(*ot);
  226. if(!this->IsExcluded(root, *t))
  227. {
  228. target->AddUtility(t->GetName());
  229. }
  230. }
  231. }
  232. void cmGlobalVisualStudio7Generator::WriteTargetConfigurations(
  233. std::ostream& fout,
  234. cmLocalGenerator* root,
  235. OrderedTargetDependSet const& projectTargets)
  236. {
  237. // loop over again and write out configurations for each target
  238. // in the solution
  239. for(OrderedTargetDependSet::const_iterator tt =
  240. projectTargets.begin(); tt != projectTargets.end(); ++tt)
  241. {
  242. cmTarget* target = *tt;
  243. if (strncmp(target->GetName(), "INCLUDE_EXTERNAL_MSPROJECT", 26) == 0)
  244. {
  245. cmCustomCommand cc = target->GetPostBuildCommands()[0];
  246. const cmCustomCommandLines& cmds = cc.GetCommandLines();
  247. std::string project = cmds[0][0];
  248. this->WriteProjectConfigurations(fout, project.c_str(),
  249. true);
  250. }
  251. else
  252. {
  253. bool partOfDefaultBuild = this->IsPartOfDefaultBuild(
  254. root->GetMakefile()->GetProjectName(), target);
  255. const char *vcprojName =
  256. target->GetProperty("GENERATOR_FILE_NAME");
  257. if (vcprojName)
  258. {
  259. this->WriteProjectConfigurations(fout, vcprojName,
  260. partOfDefaultBuild);
  261. }
  262. }
  263. }
  264. }
  265. void cmGlobalVisualStudio7Generator::WriteTargetsToSolution(
  266. std::ostream& fout,
  267. cmLocalGenerator* root,
  268. OrderedTargetDependSet const& projectTargets,
  269. cmGlobalGenerator::TargetDependSet& originalTargets
  270. )
  271. {
  272. std::string rootdir = root->GetMakefile()->GetStartOutputDirectory();
  273. rootdir += "/";
  274. for(OrderedTargetDependSet::const_iterator tt =
  275. projectTargets.begin(); tt != projectTargets.end(); ++tt)
  276. {
  277. cmTarget* target = *tt;
  278. cmMakefile* mf = target->GetMakefile();
  279. // look for the all_build rule and add depends to all
  280. // of the original targets (none that were "pulled" into this project)
  281. if(mf == root->GetMakefile() &&
  282. strcmp(target->GetName(), "ALL_BUILD") == 0)
  283. {
  284. this->AddAllBuildDepends(root, target, originalTargets);
  285. }
  286. // handle external vc project files
  287. if (strncmp(target->GetName(), "INCLUDE_EXTERNAL_MSPROJECT", 26) == 0)
  288. {
  289. cmCustomCommand cc = target->GetPostBuildCommands()[0];
  290. const cmCustomCommandLines& cmds = cc.GetCommandLines();
  291. std::string project = cmds[0][0];
  292. std::string location = cmds[0][1];
  293. this->WriteExternalProject(fout, project.c_str(),
  294. location.c_str(), cc.GetDepends());
  295. }
  296. else
  297. {
  298. bool skip = false;
  299. // if it is a global target or the check build system target
  300. // or the all_build target
  301. // then only use the one that is for the root
  302. if(target->GetType() == cmTarget::GLOBAL_TARGET
  303. || !strcmp(target->GetName(), CMAKE_CHECK_BUILD_SYSTEM_TARGET)
  304. || !strcmp(target->GetName(), this->GetAllTargetName()))
  305. {
  306. if(target->GetMakefile() != root->GetMakefile())
  307. {
  308. skip = true;
  309. }
  310. }
  311. // if not skipping the project then write it into the
  312. // solution
  313. if(!skip)
  314. {
  315. const char *vcprojName =
  316. target->GetProperty("GENERATOR_FILE_NAME");
  317. if(vcprojName)
  318. {
  319. cmMakefile* tmf = target->GetMakefile();
  320. std::string dir = tmf->GetStartOutputDirectory();
  321. dir = root->Convert(dir.c_str(),
  322. cmLocalGenerator::START_OUTPUT);
  323. this->WriteProject(fout, vcprojName, dir.c_str(),
  324. *target);
  325. }
  326. }
  327. }
  328. }
  329. }
  330. void cmGlobalVisualStudio7Generator::WriteTargetDepends(
  331. std::ostream& fout,
  332. OrderedTargetDependSet const& projectTargets
  333. )
  334. {
  335. for(OrderedTargetDependSet::const_iterator tt =
  336. projectTargets.begin(); tt != projectTargets.end(); ++tt)
  337. {
  338. cmTarget* target = *tt;
  339. cmMakefile* mf = target->GetMakefile();
  340. if (strncmp(target->GetName(), "INCLUDE_EXTERNAL_MSPROJECT", 26) == 0)
  341. {
  342. cmCustomCommand cc = target->GetPostBuildCommands()[0];
  343. const cmCustomCommandLines& cmds = cc.GetCommandLines();
  344. std::string name = cmds[0][0];
  345. std::vector<std::string> depends = cc.GetDepends();
  346. std::vector<std::string>::iterator iter;
  347. int depcount = 0;
  348. for(iter = depends.begin(); iter != depends.end(); ++iter)
  349. {
  350. std::string guid = this->GetGUID(iter->c_str());
  351. if(guid.size() == 0)
  352. {
  353. std::string m = "Target: ";
  354. m += target->GetName();
  355. m += " depends on unknown target: ";
  356. m += iter->c_str();
  357. cmSystemTools::Error(m.c_str());
  358. }
  359. fout << "\t\t{" << this->GetGUID(name.c_str())
  360. << "}." << depcount << " = {" << guid.c_str() << "}\n";
  361. depcount++;
  362. }
  363. }
  364. else
  365. {
  366. const char *vcprojName =
  367. target->GetProperty("GENERATOR_FILE_NAME");
  368. if (vcprojName)
  369. {
  370. std::string dir = mf->GetStartDirectory();
  371. this->WriteProjectDepends(fout, vcprojName,
  372. dir.c_str(), *target);
  373. }
  374. }
  375. }
  376. }
  377. // Write a SLN file to the stream
  378. void cmGlobalVisualStudio7Generator
  379. ::WriteSLNFile(std::ostream& fout,
  380. cmLocalGenerator* root,
  381. std::vector<cmLocalGenerator*>& generators)
  382. {
  383. // Write out the header for a SLN file
  384. this->WriteSLNHeader(fout);
  385. // collect the set of targets for this project by
  386. // tracing depends of all targets.
  387. // also collect the set of targets that are explicitly
  388. // in this project.
  389. cmGlobalGenerator::TargetDependSet projectTargets;
  390. cmGlobalGenerator::TargetDependSet originalTargets;
  391. this->GetTargetSets(projectTargets,
  392. originalTargets,
  393. root, generators);
  394. OrderedTargetDependSet orderedProjectTargets(projectTargets);
  395. this->WriteTargetsToSolution(fout, root, orderedProjectTargets,
  396. originalTargets);
  397. // Write out the configurations information for the solution
  398. fout << "Global\n"
  399. << "\tGlobalSection(SolutionConfiguration) = preSolution\n";
  400. int c = 0;
  401. for(std::vector<std::string>::iterator i = this->Configurations.begin();
  402. i != this->Configurations.end(); ++i)
  403. {
  404. fout << "\t\tConfigName." << c << " = " << *i << "\n";
  405. c++;
  406. }
  407. fout << "\tEndGlobalSection\n";
  408. // Write out project(target) depends
  409. fout << "\tGlobalSection(ProjectDependencies) = postSolution\n";
  410. this->WriteTargetDepends(fout, orderedProjectTargets);
  411. fout << "\tEndGlobalSection\n";
  412. // Write out the configurations for all the targets in the project
  413. fout << "\tGlobalSection(ProjectConfiguration) = postSolution\n";
  414. this->WriteTargetConfigurations(fout, root, orderedProjectTargets);
  415. fout << "\tEndGlobalSection\n";
  416. // Write the footer for the SLN file
  417. this->WriteSLNFooter(fout);
  418. }
  419. //----------------------------------------------------------------------------
  420. std::string
  421. cmGlobalVisualStudio7Generator::ConvertToSolutionPath(const char* path)
  422. {
  423. // Convert to backslashes. Do not use ConvertToOutputPath because
  424. // we will add quoting ourselves, and we know these projects always
  425. // use windows slashes.
  426. std::string d = path;
  427. std::string::size_type pos = 0;
  428. while((pos = d.find('/', pos)) != d.npos)
  429. {
  430. d[pos++] = '\\';
  431. }
  432. return d;
  433. }
  434. // Write a dsp file into the SLN file,
  435. // Note, that dependencies from executables to
  436. // the libraries it uses are also done here
  437. void cmGlobalVisualStudio7Generator::WriteProject(std::ostream& fout,
  438. const char* dspname,
  439. const char* dir, cmTarget& target)
  440. {
  441. // check to see if this is a fortran build
  442. const char* ext = ".vcproj";
  443. const char* project =
  444. "Project(\"{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}\") = \"";
  445. if(this->TargetIsFortranOnly(target))
  446. {
  447. ext = ".vfproj";
  448. project = "Project(\"{6989167D-11E4-40FE-8C1A-2192A86A7E90}\") = \"";
  449. }
  450. fout << project
  451. << dspname << "\", \""
  452. << this->ConvertToSolutionPath(dir)
  453. << "\\" << dspname << ext << "\", \"{"
  454. << this->GetGUID(dspname) << "}\"\nEndProject\n";
  455. }
  456. // Write a dsp file into the SLN file,
  457. // Note, that dependencies from executables to
  458. // the libraries it uses are also done here
  459. void
  460. cmGlobalVisualStudio7Generator
  461. ::WriteProjectDepends(std::ostream& fout,
  462. const char* dspname,
  463. const char*, cmTarget& target)
  464. {
  465. int depcount = 0;
  466. // insert Begin Project Dependency Project_Dep_Name project stuff here
  467. if (target.GetType() != cmTarget::STATIC_LIBRARY)
  468. {
  469. cmTarget::LinkLibraryVectorType::const_iterator j, jend;
  470. j = target.GetLinkLibraries().begin();
  471. jend = target.GetLinkLibraries().end();
  472. for(;j!= jend; ++j)
  473. {
  474. if(j->first != dspname)
  475. {
  476. // is the library part of this SLN ? If so add dependency
  477. if(this->FindTarget(0, j->first.c_str()))
  478. {
  479. std::string guid = this->GetGUID(j->first.c_str());
  480. if(guid.size() == 0)
  481. {
  482. std::string m = "Target: ";
  483. m += dspname;
  484. m += " depends on unknown target: ";
  485. m += j->first.c_str();
  486. cmSystemTools::Error(m.c_str());
  487. }
  488. fout << "\t\t{" << this->GetGUID(dspname) << "}."
  489. << depcount << " = {" << guid << "}\n";
  490. depcount++;
  491. }
  492. }
  493. }
  494. }
  495. std::set<cmStdString>::const_iterator i, end;
  496. // write utility dependencies.
  497. i = target.GetUtilities().begin();
  498. end = target.GetUtilities().end();
  499. for(;i!= end; ++i)
  500. {
  501. if(*i != dspname)
  502. {
  503. std::string name = this->GetUtilityForTarget(target, i->c_str());
  504. std::string guid = this->GetGUID(name.c_str());
  505. if(guid.size() == 0)
  506. {
  507. std::string m = "Target: ";
  508. m += dspname;
  509. m += " depends on unknown target: ";
  510. m += name.c_str();
  511. cmSystemTools::Error(m.c_str());
  512. }
  513. fout << "\t\t{" << this->GetGUID(dspname) << "}." << depcount << " = {"
  514. << guid << "}\n";
  515. depcount++;
  516. }
  517. }
  518. }
  519. // Write a dsp file into the SLN file, Note, that dependencies from
  520. // executables to the libraries it uses are also done here
  521. void cmGlobalVisualStudio7Generator
  522. ::WriteProjectConfigurations(std::ostream& fout, const char* name,
  523. bool partOfDefaultBuild)
  524. {
  525. std::string guid = this->GetGUID(name);
  526. for(std::vector<std::string>::iterator i = this->Configurations.begin();
  527. i != this->Configurations.end(); ++i)
  528. {
  529. fout << "\t\t{" << guid << "}." << *i
  530. << ".ActiveCfg = " << *i << "|Win32\n";
  531. if(partOfDefaultBuild)
  532. {
  533. fout << "\t\t{" << guid << "}." << *i
  534. << ".Build.0 = " << *i << "|Win32\n";
  535. }
  536. }
  537. }
  538. // Write a dsp file into the SLN file,
  539. // Note, that dependencies from executables to
  540. // the libraries it uses are also done here
  541. void cmGlobalVisualStudio7Generator::WriteExternalProject(std::ostream& fout,
  542. const char* name,
  543. const char* location,
  544. const std::vector<std::string>&)
  545. {
  546. std::cout << "WriteExternalProject vs7\n";
  547. std::string d = cmSystemTools::ConvertToOutputPath(location);
  548. fout << "Project(\"{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}\") = \""
  549. << name << "\", \""
  550. << this->ConvertToSolutionPath(location) << "\", \"{"
  551. << this->GetGUID(name)
  552. << "}\"\n";
  553. fout << "EndProject\n";
  554. }
  555. // Standard end of dsw file
  556. void cmGlobalVisualStudio7Generator::WriteSLNFooter(std::ostream& fout)
  557. {
  558. fout << "\tGlobalSection(ExtensibilityGlobals) = postSolution\n"
  559. << "\tEndGlobalSection\n"
  560. << "\tGlobalSection(ExtensibilityAddIns) = postSolution\n"
  561. << "\tEndGlobalSection\n"
  562. << "EndGlobal\n";
  563. }
  564. // ouput standard header for dsw file
  565. void cmGlobalVisualStudio7Generator::WriteSLNHeader(std::ostream& fout)
  566. {
  567. fout << "Microsoft Visual Studio Solution File, Format Version 7.00\n";
  568. }
  569. std::string cmGlobalVisualStudio7Generator::GetGUID(const char* name)
  570. {
  571. std::string guidStoreName = name;
  572. guidStoreName += "_GUID_CMAKE";
  573. const char* storedGUID =
  574. this->CMakeInstance->GetCacheDefinition(guidStoreName.c_str());
  575. if(storedGUID)
  576. {
  577. return std::string(storedGUID);
  578. }
  579. cmSystemTools::Error("Unknown Target referenced : ",
  580. name);
  581. return "";
  582. }
  583. void cmGlobalVisualStudio7Generator::CreateGUID(const char* name)
  584. {
  585. std::string guidStoreName = name;
  586. guidStoreName += "_GUID_CMAKE";
  587. if(this->CMakeInstance->GetCacheDefinition(guidStoreName.c_str()))
  588. {
  589. return;
  590. }
  591. std::string ret;
  592. UUID uid;
  593. unsigned char *uidstr;
  594. UuidCreate(&uid);
  595. UuidToString(&uid,&uidstr);
  596. ret = reinterpret_cast<char*>(uidstr);
  597. RpcStringFree(&uidstr);
  598. ret = cmSystemTools::UpperCase(ret);
  599. this->CMakeInstance->AddCacheEntry(guidStoreName.c_str(),
  600. ret.c_str(), "Stored GUID",
  601. cmCacheManager::INTERNAL);
  602. }
  603. std::vector<std::string> *cmGlobalVisualStudio7Generator::GetConfigurations()
  604. {
  605. return &this->Configurations;
  606. };
  607. //----------------------------------------------------------------------------
  608. void cmGlobalVisualStudio7Generator
  609. ::GetDocumentation(cmDocumentationEntry& entry) const
  610. {
  611. entry.Name = this->GetName();
  612. entry.Brief = "Generates Visual Studio .NET 2002 project files.";
  613. entry.Full = "";
  614. }
  615. // make sure "special" targets have GUID's
  616. void cmGlobalVisualStudio7Generator::Configure()
  617. {
  618. cmGlobalGenerator::Configure();
  619. this->CreateGUID("ALL_BUILD");
  620. this->CreateGUID("INSTALL");
  621. this->CreateGUID("RUN_TESTS");
  622. this->CreateGUID("EDIT_CACHE");
  623. this->CreateGUID("REBUILD_CACHE");
  624. this->CreateGUID("PACKAGE");
  625. }
  626. //----------------------------------------------------------------------------
  627. void
  628. cmGlobalVisualStudio7Generator
  629. ::AppendDirectoryForConfig(const char* prefix,
  630. const char* config,
  631. const char* suffix,
  632. std::string& dir)
  633. {
  634. if(config)
  635. {
  636. dir += prefix;
  637. dir += config;
  638. dir += suffix;
  639. }
  640. }
  641. bool cmGlobalVisualStudio7Generator::IsPartOfDefaultBuild(const char* project,
  642. cmTarget* target)
  643. {
  644. if(target->GetPropertyAsBool("EXCLUDE_FROM_DEFAULT_BUILD"))
  645. {
  646. return false;
  647. }
  648. // if it is a utilitiy target then only make it part of the
  649. // default build if another target depends on it
  650. int type = target->GetType();
  651. if (type == cmTarget::GLOBAL_TARGET)
  652. {
  653. return false;
  654. }
  655. if(type == cmTarget::UTILITY)
  656. {
  657. return this->IsDependedOn(project, target);
  658. }
  659. // default is to be part of the build
  660. return true;
  661. }
  662. //----------------------------------------------------------------------------
  663. bool
  664. cmGlobalVisualStudio7Generator::TargetCompare
  665. ::operator()(cmTarget const* l, cmTarget const* r)
  666. {
  667. // Make sure ALL_BUILD is first so it is the default active project.
  668. if(strcmp(r->GetName(), "ALL_BUILD") == 0)
  669. {
  670. return false;
  671. }
  672. if(strcmp(l->GetName(), "ALL_BUILD") == 0)
  673. {
  674. return true;
  675. }
  676. return strcmp(l->GetName(), r->GetName()) < 0;
  677. }
  678. //----------------------------------------------------------------------------
  679. cmGlobalVisualStudio7Generator::OrderedTargetDependSet
  680. ::OrderedTargetDependSet(cmGlobalGenerator::TargetDependSet const& targets)
  681. {
  682. for(cmGlobalGenerator::TargetDependSet::const_iterator ti =
  683. targets.begin(); ti != targets.end(); ++ti)
  684. {
  685. this->insert(*ti);
  686. }
  687. }
  688. //----------------------------------------------------------------------------
  689. static cmVS7FlagTable cmVS7ExtraFlagTable[] =
  690. {
  691. // Precompiled header and related options. Note that the
  692. // UsePrecompiledHeader entries are marked as "Continue" so that the
  693. // corresponding PrecompiledHeaderThrough entry can be found.
  694. {"UsePrecompiledHeader", "YX", "Automatically Generate", "2",
  695. cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue},
  696. {"PrecompiledHeaderThrough", "YX", "Precompiled Header Name", "",
  697. cmVS7FlagTable::UserValueRequired},
  698. {"UsePrecompiledHeader", "Yu", "Use Precompiled Header", "3",
  699. cmVS7FlagTable::UserValueIgnored | cmVS7FlagTable::Continue},
  700. {"PrecompiledHeaderThrough", "Yu", "Precompiled Header Name", "",
  701. cmVS7FlagTable::UserValueRequired},
  702. {"WholeProgramOptimization", "LTCG", "WholeProgramOptimization", "TRUE", 0},
  703. // Exception handling mode. If no entries match, it will be FALSE.
  704. {"ExceptionHandling", "GX", "enable c++ exceptions", "TRUE", 0},
  705. {"ExceptionHandling", "EHsc", "enable c++ exceptions", "TRUE", 0},
  706. // The EHa option does not have an IDE setting. Let it go to false,
  707. // and have EHa passed on the command line by leaving out the table
  708. // entry.
  709. {0,0,0,0,0}
  710. };
  711. cmVS7FlagTable const* cmGlobalVisualStudio7Generator::GetExtraFlagTableVS7()
  712. {
  713. return cmVS7ExtraFlagTable;
  714. }