cpack.cxx 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620
  1. /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. file LICENSE.rst or https://cmake.org/licensing for details. */
  3. #include <algorithm>
  4. #include <cstddef>
  5. #include <functional>
  6. #include <iostream>
  7. #include <iterator>
  8. #include <map>
  9. #include <memory>
  10. #include <sstream>
  11. #include <string>
  12. #include <type_traits>
  13. #include <utility>
  14. #include <vector>
  15. #include <cm/optional>
  16. #include <cmext/algorithm>
  17. #include "cmsys/Encoding.hxx"
  18. #include "cmCMakePresetsGraph.h"
  19. #include "cmCPackGenerator.h"
  20. #include "cmCPackGeneratorFactory.h"
  21. #include "cmCPackLog.h"
  22. #include "cmCommandLineArgument.h"
  23. #include "cmDocumentation.h"
  24. #include "cmDocumentationEntry.h"
  25. #include "cmGlobalGenerator.h"
  26. #include "cmJSONState.h"
  27. #include "cmList.h"
  28. #include "cmMakefile.h"
  29. #include "cmState.h"
  30. #include "cmStateSnapshot.h"
  31. #include "cmStdIoConsole.h"
  32. #include "cmStdIoInit.h"
  33. #include "cmStringAlgorithms.h"
  34. #include "cmSystemTools.h"
  35. #include "cmValue.h"
  36. #include "cmake.h"
  37. namespace {
  38. cmDocumentationEntry const cmDocumentationName = {
  39. {},
  40. " cpack - Packaging driver provided by CMake."
  41. };
  42. cmDocumentationEntry const cmDocumentationUsage = { {}, " cpack [options]" };
  43. cmDocumentationEntry const cmDocumentationOptions[14] = {
  44. { "-G <generators>", "Override/define CPACK_GENERATOR" },
  45. { "-C <Configurations>", "Specify the project configuration(s)" },
  46. { "-D <var>=<value>", "Set a CPack variable." },
  47. { "--config <configFile>", "Specify the config file." },
  48. { "-V,--verbose", "Enable verbose output" },
  49. { "--trace", "Put underlying cmake scripts in trace mode." },
  50. { "--trace-expand", "Put underlying cmake scripts in expanded trace mode." },
  51. { "--debug", "Enable debug output (for CPack developers)" },
  52. { "-P <packageName>", "Override/define CPACK_PACKAGE_NAME" },
  53. { "-R <packageVersion>", "Override/define CPACK_PACKAGE_VERSION" },
  54. { "-B <packageDirectory>", "Override/define CPACK_PACKAGE_DIRECTORY" },
  55. { "--vendor <vendorName>", "Override/define CPACK_PACKAGE_VENDOR" },
  56. { "--preset", "Read arguments from a package preset" },
  57. { "--list-presets", "List available package presets" }
  58. };
  59. void cpackProgressCallback(std::string const& message, float /*unused*/)
  60. {
  61. std::cout << "-- " << message << '\n';
  62. }
  63. std::vector<cmDocumentationEntry> makeGeneratorDocs(
  64. cmCPackGeneratorFactory const& gf)
  65. {
  66. auto const& generators = gf.GetGeneratorsList();
  67. std::vector<cmDocumentationEntry> docs;
  68. docs.reserve(generators.size());
  69. std::transform(
  70. generators.cbegin(), generators.cend(), std::back_inserter(docs),
  71. [](std::decay<decltype(generators)>::type::value_type const& gen) {
  72. return cmDocumentationEntry{ gen.first, gen.second };
  73. });
  74. return docs;
  75. }
  76. } // namespace
  77. // this is CPack.
  78. int main(int argc, char const* const* argv)
  79. {
  80. cm::StdIo::Init();
  81. cm::StdIo::Console console;
  82. cmsys::Encoding::CommandLineArguments args =
  83. cmsys::Encoding::CommandLineArguments::Main(argc, argv);
  84. argc = args.argc();
  85. argv = args.argv();
  86. std::vector<std::string> inputArgs;
  87. inputArgs.reserve(argc - 1);
  88. cm::append(inputArgs, argv + 1, argv + argc);
  89. cmSystemTools::InitializeLibUV();
  90. cmSystemTools::FindCMakeResources(argv[0]);
  91. cmCPackLog log;
  92. log.SetErrorPrefix("CPack Error: ");
  93. log.SetWarningPrefix("CPack Warning: ");
  94. log.SetOutputPrefix("CPack: ");
  95. log.SetVerbosePrefix("CPack Verbose: ");
  96. if (cmSystemTools::GetLogicalWorkingDirectory().empty()) {
  97. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  98. "Current working directory cannot be established.\n");
  99. return 1;
  100. }
  101. std::string generator;
  102. bool help = false;
  103. bool helpVersion = false;
  104. std::string helpFull;
  105. std::string helpMAN;
  106. std::string helpHTML;
  107. std::string cpackProjectName;
  108. std::string cpackProjectDirectory;
  109. std::string cpackBuildConfig;
  110. std::string cpackProjectVersion;
  111. std::string cpackProjectPatch;
  112. std::string cpackProjectVendor;
  113. std::string cpackConfigFile;
  114. std::string preset;
  115. bool listPresets = false;
  116. std::map<std::string, std::string> definitions;
  117. auto const verboseLambda = [&log](std::string const&, cmake*,
  118. cmMakefile*) -> bool {
  119. log.SetVerbose(true);
  120. cmCPack_Log(&log, cmCPackLog::LOG_OUTPUT, "Enable Verbose\n");
  121. return true;
  122. };
  123. auto const debugLambda = [&log](std::string const&, cmake*,
  124. cmMakefile*) -> bool {
  125. log.SetDebug(true);
  126. cmCPack_Log(&log, cmCPackLog::LOG_OUTPUT, "Enable Debug\n");
  127. return true;
  128. };
  129. auto const traceLambda = [](std::string const&, cmake* state,
  130. cmMakefile*) -> bool {
  131. state->SetTrace(true);
  132. return true;
  133. };
  134. auto const traceExpandLambda = [](std::string const&, cmake* state,
  135. cmMakefile*) -> bool {
  136. state->SetTrace(true);
  137. state->SetTraceExpand(true);
  138. return true;
  139. };
  140. using CommandArgument =
  141. cmCommandLineArgument<bool(std::string const&, cmake*, cmMakefile*)>;
  142. std::vector<CommandArgument> arguments = {
  143. CommandArgument{ "--help", CommandArgument::Values::Zero,
  144. CommandArgument::setToTrue(help) },
  145. CommandArgument{ "--help-full", CommandArgument::Values::Zero,
  146. CommandArgument::setToValue(helpFull) },
  147. CommandArgument{ "--help-html", CommandArgument::Values::Zero,
  148. CommandArgument::setToValue(helpHTML) },
  149. CommandArgument{ "--help-man", CommandArgument::Values::Zero,
  150. CommandArgument::setToValue(helpMAN) },
  151. CommandArgument{ "--version", CommandArgument::Values::Zero,
  152. CommandArgument::setToTrue(helpVersion) },
  153. CommandArgument{ "-V", CommandArgument::Values::Zero, verboseLambda },
  154. CommandArgument{ "--verbose", CommandArgument::Values::Zero,
  155. verboseLambda },
  156. CommandArgument{ "--debug", CommandArgument::Values::Zero, debugLambda },
  157. CommandArgument{ "--config", CommandArgument::Values::One,
  158. CommandArgument::setToValue(cpackConfigFile) },
  159. CommandArgument{ "--trace", CommandArgument::Values::Zero, traceLambda },
  160. CommandArgument{ "--trace-expand", CommandArgument::Values::Zero,
  161. traceExpandLambda },
  162. CommandArgument{ "-C", CommandArgument::Values::One,
  163. CommandArgument::setToValue(cpackBuildConfig) },
  164. CommandArgument{ "-G", CommandArgument::Values::One,
  165. CommandArgument::setToValue(generator) },
  166. CommandArgument{ "-P", CommandArgument::Values::One,
  167. CommandArgument::setToValue(cpackProjectName) },
  168. CommandArgument{ "-R", CommandArgument::Values::One,
  169. CommandArgument::setToValue(cpackProjectVersion) },
  170. CommandArgument{ "-B", CommandArgument::Values::One,
  171. CommandArgument::setToValue(cpackProjectDirectory) },
  172. CommandArgument{ "--patch", CommandArgument::Values::One,
  173. CommandArgument::setToValue(cpackProjectPatch) },
  174. CommandArgument{ "--vendor", CommandArgument::Values::One,
  175. CommandArgument::setToValue(cpackProjectVendor) },
  176. CommandArgument{ "--preset", CommandArgument::Values::One,
  177. CommandArgument::setToValue(preset) },
  178. CommandArgument{ "--list-presets", CommandArgument::Values::Zero,
  179. CommandArgument::setToTrue(listPresets) },
  180. CommandArgument{ "-D", CommandArgument::Values::One,
  181. CommandArgument::RequiresSeparator::No,
  182. [&log, &definitions](std::string const& arg, cmake*,
  183. cmMakefile*) -> bool {
  184. std::string value = arg;
  185. size_t pos = value.find_first_of('=');
  186. if (pos == std::string::npos) {
  187. cmCPack_Log(
  188. &log, cmCPackLog::LOG_ERROR,
  189. "Please specify CPack definitions as: KEY=VALUE\n");
  190. return false;
  191. }
  192. std::string key = value.substr(0, pos);
  193. value.erase(0, pos + 1);
  194. definitions[key] = value;
  195. cmCPack_Log(&log, cmCPackLog::LOG_DEBUG,
  196. "Set CPack variable: " << key << " to \""
  197. << value << "\"\n");
  198. return true;
  199. } },
  200. };
  201. cmake cminst(cmake::RoleScript, cmState::CPack);
  202. cminst.SetHomeDirectory("");
  203. cminst.SetHomeOutputDirectory("");
  204. cminst.SetProgressCallback(cpackProgressCallback);
  205. cminst.GetCurrentSnapshot().SetDefaultDefinitions();
  206. cmGlobalGenerator cmgg(&cminst);
  207. cmMakefile globalMF(&cmgg, cminst.GetCurrentSnapshot());
  208. bool parsed = true;
  209. for (std::size_t i = 0; i < inputArgs.size(); i++) {
  210. auto const& arg = inputArgs[i];
  211. for (auto const& m : arguments) {
  212. if (m.matches(arg)) {
  213. if (!m.parse(arg, i, inputArgs, &cminst, &globalMF)) {
  214. parsed = false;
  215. }
  216. break;
  217. }
  218. }
  219. }
  220. cmCPackGeneratorFactory generators;
  221. generators.SetLogger(&log);
  222. // Set up presets
  223. if (!preset.empty() || listPresets) {
  224. auto const workingDirectory = cmSystemTools::GetLogicalWorkingDirectory();
  225. auto const presetGeneratorsPresent =
  226. [&generators](cmCMakePresetsGraph::PackagePreset const& p) {
  227. return std::all_of(p.Generators.begin(), p.Generators.end(),
  228. [&generators](std::string const& gen) {
  229. return generators.GetGeneratorsList().count(
  230. gen) != 0;
  231. });
  232. };
  233. cmCMakePresetsGraph presetsGraph;
  234. auto result = presetsGraph.ReadProjectPresets(workingDirectory);
  235. if (result != true) {
  236. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  237. "Could not read presets from "
  238. << workingDirectory << ":"
  239. << presetsGraph.parseState.GetErrorMessage() << '\n');
  240. return 1;
  241. }
  242. if (listPresets) {
  243. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  244. return 0;
  245. }
  246. auto presetPair = presetsGraph.PackagePresets.find(preset);
  247. if (presetPair == presetsGraph.PackagePresets.end()) {
  248. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  249. "No such package preset in " << workingDirectory << ": \""
  250. << preset << "\"\n");
  251. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  252. return 1;
  253. }
  254. if (presetPair->second.Unexpanded.Hidden) {
  255. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  256. "Cannot use hidden package preset in "
  257. << workingDirectory << ": \"" << preset << "\"\n");
  258. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  259. return 1;
  260. }
  261. auto const& expandedPreset = presetPair->second.Expanded;
  262. if (!expandedPreset) {
  263. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  264. "Could not evaluate package preset \""
  265. << preset << "\": Invalid macro expansion\n");
  266. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  267. return 1;
  268. }
  269. if (!expandedPreset->ConditionResult) {
  270. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  271. "Cannot use disabled package preset in "
  272. << workingDirectory << ": \"" << preset << "\"\n");
  273. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  274. return 1;
  275. }
  276. if (!presetGeneratorsPresent(presetPair->second.Unexpanded)) {
  277. cmCPack_Log(&log, cmCPackLog::LOG_ERROR, "Cannot use preset");
  278. presetsGraph.PrintPackagePresetList(presetGeneratorsPresent);
  279. return 1;
  280. }
  281. auto configurePresetPair =
  282. presetsGraph.ConfigurePresets.find(expandedPreset->ConfigurePreset);
  283. if (configurePresetPair == presetsGraph.ConfigurePresets.end()) {
  284. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  285. "No such configure preset in "
  286. << workingDirectory << ": \""
  287. << expandedPreset->ConfigurePreset << "\"\n");
  288. presetsGraph.PrintConfigurePresetList();
  289. return 1;
  290. }
  291. if (configurePresetPair->second.Unexpanded.Hidden) {
  292. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  293. "Cannot use hidden configure preset in "
  294. << workingDirectory << ": \""
  295. << expandedPreset->ConfigurePreset << "\"\n");
  296. presetsGraph.PrintConfigurePresetList();
  297. return 1;
  298. }
  299. auto const& expandedConfigurePreset = configurePresetPair->second.Expanded;
  300. if (!expandedConfigurePreset) {
  301. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  302. "Could not evaluate configure preset \""
  303. << expandedPreset->ConfigurePreset
  304. << "\": Invalid macro expansion\n");
  305. return 1;
  306. }
  307. cmSystemTools::SetLogicalWorkingDirectory(
  308. expandedConfigurePreset->BinaryDir);
  309. auto presetEnvironment = expandedPreset->Environment;
  310. for (auto const& var : presetEnvironment) {
  311. if (var.second) {
  312. cmSystemTools::PutEnv(cmStrCat(var.first, '=', *var.second));
  313. }
  314. }
  315. if (!expandedPreset->ConfigFile.empty() && cpackConfigFile.empty()) {
  316. cpackConfigFile = expandedPreset->ConfigFile;
  317. }
  318. if (!expandedPreset->Generators.empty() && generator.empty()) {
  319. generator = cmList::to_string(expandedPreset->Generators);
  320. }
  321. if (!expandedPreset->Configurations.empty() && cpackBuildConfig.empty()) {
  322. cpackBuildConfig = cmList::to_string(expandedPreset->Configurations);
  323. }
  324. definitions.insert(expandedPreset->Variables.begin(),
  325. expandedPreset->Variables.end());
  326. if (expandedPreset->DebugOutput == true) {
  327. debugLambda("", &cminst, &globalMF);
  328. }
  329. if (expandedPreset->VerboseOutput == true) {
  330. verboseLambda("", &cminst, &globalMF);
  331. }
  332. if (!expandedPreset->PackageName.empty() && cpackProjectName.empty()) {
  333. cpackProjectName = expandedPreset->PackageName;
  334. }
  335. if (!expandedPreset->PackageVersion.empty() &&
  336. cpackProjectVersion.empty()) {
  337. cpackProjectVersion = expandedPreset->PackageVersion;
  338. }
  339. if (!expandedPreset->PackageDirectory.empty() &&
  340. cpackProjectDirectory.empty()) {
  341. cpackProjectDirectory = expandedPreset->PackageDirectory;
  342. }
  343. if (!expandedPreset->VendorName.empty() && cpackProjectVendor.empty()) {
  344. cpackProjectVendor = expandedPreset->VendorName;
  345. }
  346. }
  347. cmCPack_Log(&log, cmCPackLog::LOG_VERBOSE,
  348. "Read CPack config file: " << cpackConfigFile << '\n');
  349. bool cpackConfigFileSpecified = true;
  350. if (!cpackConfigFile.empty()) {
  351. cpackConfigFile = cmSystemTools::ToNormalizedPathOnDisk(cpackConfigFile);
  352. } else {
  353. cpackConfigFile = cmStrCat(cmSystemTools::GetLogicalWorkingDirectory(),
  354. "/CPackConfig.cmake");
  355. cpackConfigFileSpecified = false;
  356. }
  357. cmDocumentation doc;
  358. doc.addCPackStandardDocSections();
  359. /* Were we invoked to display doc or to do some work ?
  360. * Unlike cmake launching cpack with zero argument
  361. * should launch cpack using "cpackConfigFile" if it exists
  362. * in the current directory.
  363. */
  364. help = doc.CheckOptions(argc, argv, "-G") && argc != 1;
  365. // This part is used for cpack documentation lookup as well.
  366. cminst.AddCMakePaths();
  367. if (parsed && !help) {
  368. // find out which system cpack is running on, so it can setup the search
  369. // paths, so FIND_XXX() commands can be used in scripts
  370. std::string systemFile =
  371. globalMF.GetModulesFile("CMakeDetermineSystem.cmake");
  372. if (!globalMF.ReadListFile(systemFile)) {
  373. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  374. "Error reading CMakeDetermineSystem.cmake\n");
  375. return 1;
  376. }
  377. systemFile =
  378. globalMF.GetModulesFile("CMakeSystemSpecificInformation.cmake");
  379. if (!globalMF.ReadListFile(systemFile)) {
  380. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  381. "Error reading CMakeSystemSpecificInformation.cmake\n");
  382. return 1;
  383. }
  384. if (!cpackBuildConfig.empty()) {
  385. globalMF.AddDefinition("CPACK_BUILD_CONFIG", cpackBuildConfig);
  386. }
  387. if (cmSystemTools::FileExists(cpackConfigFile)) {
  388. cmCPack_Log(&log, cmCPackLog::LOG_VERBOSE,
  389. "Read CPack configuration file: " << cpackConfigFile
  390. << '\n');
  391. if (!globalMF.ReadListFile(cpackConfigFile)) {
  392. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  393. "Problem reading CPack config file: \"" << cpackConfigFile
  394. << "\"\n");
  395. return 1;
  396. }
  397. } else if (cpackConfigFileSpecified) {
  398. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  399. "Cannot find CPack config file: \"" << cpackConfigFile
  400. << "\"\n");
  401. return 1;
  402. }
  403. if (!generator.empty()) {
  404. globalMF.AddDefinition("CPACK_GENERATOR", generator);
  405. }
  406. if (!cpackProjectName.empty()) {
  407. globalMF.AddDefinition("CPACK_PACKAGE_NAME", cpackProjectName);
  408. }
  409. if (!cpackProjectVersion.empty()) {
  410. globalMF.AddDefinition("CPACK_PACKAGE_VERSION", cpackProjectVersion);
  411. }
  412. if (!cpackProjectVendor.empty()) {
  413. globalMF.AddDefinition("CPACK_PACKAGE_VENDOR", cpackProjectVendor);
  414. }
  415. if (!cpackProjectDirectory.empty()) {
  416. // The value has been set on the command line. Ensure it is absolute.
  417. cpackProjectDirectory =
  418. cmSystemTools::ToNormalizedPathOnDisk(cpackProjectDirectory);
  419. } else {
  420. // The value has not been set on the command line. Check config file.
  421. if (cmValue pd = globalMF.GetDefinition("CPACK_PACKAGE_DIRECTORY")) {
  422. // The value has been set in the config file. Ensure it is absolute.
  423. cpackProjectDirectory = cmSystemTools::CollapseFullPath(*pd);
  424. } else {
  425. // Default to the current working directory.
  426. cpackProjectDirectory = cmSystemTools::GetLogicalWorkingDirectory();
  427. }
  428. }
  429. globalMF.AddDefinition("CPACK_PACKAGE_DIRECTORY", cpackProjectDirectory);
  430. for (auto const& cd : definitions) {
  431. globalMF.AddDefinition(cd.first, cd.second);
  432. }
  433. cmValue cpackModulesPath = globalMF.GetDefinition("CPACK_MODULE_PATH");
  434. if (cpackModulesPath) {
  435. globalMF.AddDefinition("CMAKE_MODULE_PATH", *cpackModulesPath);
  436. }
  437. cmValue genList = globalMF.GetDefinition("CPACK_GENERATOR");
  438. if (!genList) {
  439. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  440. "CPack generator not specified\n");
  441. } else {
  442. cmList generatorsList{ *genList };
  443. for (std::string const& gen : generatorsList) {
  444. cmMakefile::ScopePushPop raii(&globalMF);
  445. cmMakefile* mf = &globalMF;
  446. cmCPack_Log(&log, cmCPackLog::LOG_VERBOSE,
  447. "Specified generator: " << gen << '\n');
  448. if (!mf->GetDefinition("CPACK_PACKAGE_NAME")) {
  449. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  450. "CPack project name not specified" << '\n');
  451. parsed = false;
  452. }
  453. if (parsed &&
  454. !(mf->GetDefinition("CPACK_PACKAGE_VERSION") ||
  455. (mf->GetDefinition("CPACK_PACKAGE_VERSION_MAJOR") &&
  456. mf->GetDefinition("CPACK_PACKAGE_VERSION_MINOR") &&
  457. mf->GetDefinition("CPACK_PACKAGE_VERSION_PATCH")))) {
  458. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  459. "CPack project version not specified\n"
  460. "Specify CPACK_PACKAGE_VERSION, or "
  461. "CPACK_PACKAGE_VERSION_MAJOR, "
  462. "CPACK_PACKAGE_VERSION_MINOR, and "
  463. "CPACK_PACKAGE_VERSION_PATCH.\n");
  464. parsed = false;
  465. }
  466. if (parsed) {
  467. std::unique_ptr<cmCPackGenerator> cpackGenerator =
  468. generators.NewGenerator(gen);
  469. if (cpackGenerator) {
  470. cpackGenerator->SetTrace(cminst.GetTrace());
  471. cpackGenerator->SetTraceExpand(cminst.GetTraceExpand());
  472. } else {
  473. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  474. "Could not create CPack generator: " << gen << '\n');
  475. // Print out all the valid generators
  476. cmDocumentation generatorDocs;
  477. generatorDocs.SetSection("Generators",
  478. makeGeneratorDocs(generators));
  479. std::cerr << '\n';
  480. generatorDocs.PrintDocumentation(cmDocumentation::ListGenerators,
  481. std::cerr);
  482. parsed = false;
  483. }
  484. if (parsed && !cpackGenerator->Initialize(gen, mf)) {
  485. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  486. "Cannot initialize the generator " << gen << '\n');
  487. parsed = false;
  488. }
  489. if (!mf->GetDefinition("CPACK_INSTALL_COMMANDS") &&
  490. !mf->GetDefinition("CPACK_INSTALL_SCRIPT") &&
  491. !mf->GetDefinition("CPACK_INSTALLED_DIRECTORIES") &&
  492. !mf->GetDefinition("CPACK_INSTALL_CMAKE_PROJECTS")) {
  493. cmCPack_Log(
  494. &log, cmCPackLog::LOG_ERROR,
  495. "Please specify build tree of the project that uses CMake "
  496. "using CPACK_INSTALL_CMAKE_PROJECTS, specify "
  497. "CPACK_INSTALL_COMMANDS, CPACK_INSTALL_SCRIPT, or "
  498. "CPACK_INSTALLED_DIRECTORIES.\n");
  499. parsed = false;
  500. }
  501. if (parsed) {
  502. cmValue projName = mf->GetDefinition("CPACK_PACKAGE_NAME");
  503. cmCPack_Log(&log, cmCPackLog::LOG_VERBOSE,
  504. "Use generator: " << cpackGenerator->GetNameOfClass()
  505. << '\n');
  506. cmCPack_Log(&log, cmCPackLog::LOG_VERBOSE,
  507. "For project: " << *projName << '\n');
  508. cmValue projVersion = mf->GetDefinition("CPACK_PACKAGE_VERSION");
  509. if (!projVersion) {
  510. cmValue projVersionMajor =
  511. mf->GetDefinition("CPACK_PACKAGE_VERSION_MAJOR");
  512. cmValue projVersionMinor =
  513. mf->GetDefinition("CPACK_PACKAGE_VERSION_MINOR");
  514. cmValue projVersionPatch =
  515. mf->GetDefinition("CPACK_PACKAGE_VERSION_PATCH");
  516. std::ostringstream ostr;
  517. ostr << *projVersionMajor << '.' << *projVersionMinor << '.'
  518. << *projVersionPatch;
  519. mf->AddDefinition("CPACK_PACKAGE_VERSION", ostr.str());
  520. }
  521. int res = cpackGenerator->DoPackage();
  522. if (!res) {
  523. cmCPack_Log(&log, cmCPackLog::LOG_ERROR,
  524. "Error when generating package: " << *projName
  525. << '\n');
  526. return 1;
  527. }
  528. }
  529. }
  530. }
  531. }
  532. }
  533. /* In this case we are building the documentation object
  534. * instance in order to create appropriate structure
  535. * in order to satisfy the appropriate --help-xxx request
  536. */
  537. if (help) {
  538. // Construct and print requested documentation.
  539. doc.SetName("cpack");
  540. doc.SetSection("Name", cmDocumentationName);
  541. doc.SetSection("Usage", cmDocumentationUsage);
  542. doc.PrependSection("Options", cmDocumentationOptions);
  543. doc.SetSection("Generators", makeGeneratorDocs(generators));
  544. return !doc.PrintRequestedDocumentation(std::cout);
  545. }
  546. return int(cmSystemTools::GetErrorOccurredFlag());
  547. }