cpack.cxx 23 KB

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