/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmExportLibraryDependenciesCommand.h" #include <map> #include <utility> #include <cm/memory> #include "cmsys/FStream.hxx" #include "cmExecutionStatus.h" #include "cmGeneratedFileStream.h" #include "cmGlobalGenerator.h" #include "cmLocalGenerator.h" #include "cmMakefile.h" #include "cmProperty.h" #include "cmStateTypes.h" #include "cmStringAlgorithms.h" #include "cmSystemTools.h" #include "cmTarget.h" #include "cmTargetLinkLibraryType.h" #include "cmake.h" class cmListFileBacktrace; static void FinalAction(cmMakefile& makefile, std::string const& filename, bool append) { // Use copy-if-different if not appending. std::unique_ptr<cmsys::ofstream> foutPtr; if (append) { const auto openmodeApp = std::ios::app; foutPtr = cm::make_unique<cmsys::ofstream>(filename.c_str(), openmodeApp); } else { std::unique_ptr<cmGeneratedFileStream> ap( new cmGeneratedFileStream(filename, true)); ap->SetCopyIfDifferent(true); foutPtr = std::move(ap); } std::ostream& fout = *foutPtr; if (!fout) { cmSystemTools::Error("Error Writing " + filename); cmSystemTools::ReportLastSystemError(""); return; } // Collect dependency information about all library targets built in // the project. cmake* cm = makefile.GetCMakeInstance(); cmGlobalGenerator* global = cm->GetGlobalGenerator(); const auto& locals = global->GetMakefiles(); std::map<std::string, std::string> libDepsOld; std::map<std::string, std::string> libDepsNew; std::map<std::string, std::string> libTypes; for (const auto& local : locals) { for (auto const& tgt : local->GetTargets()) { // Get the current target. cmTarget const& target = tgt.second; // Skip non-library targets. if (target.GetType() < cmStateEnums::STATIC_LIBRARY || target.GetType() > cmStateEnums::MODULE_LIBRARY) { continue; } // Construct the dependency variable name. std::string targetEntry = cmStrCat(target.GetName(), "_LIB_DEPENDS"); // Construct the dependency variable value with the direct link // dependencies. std::string valueOld; std::string valueNew; cmTarget::LinkLibraryVectorType const& libs = target.GetOriginalLinkLibraries(); for (cmTarget::LibraryID const& li : libs) { std::string ltVar = cmStrCat(li.first, "_LINK_TYPE"); std::string ltValue; switch (li.second) { case GENERAL_LibraryType: valueNew += "general;"; ltValue = "general"; break; case DEBUG_LibraryType: valueNew += "debug;"; ltValue = "debug"; break; case OPTIMIZED_LibraryType: valueNew += "optimized;"; ltValue = "optimized"; break; } std::string lib = li.first; if (cmTarget* libtgt = global->FindTarget(lib)) { // Handle simple output name changes. This command is // deprecated so we do not support full target name // translation (which requires per-configuration info). if (cmProp outname = libtgt->GetProperty("OUTPUT_NAME")) { lib = *outname; } } valueOld += lib; valueOld += ";"; valueNew += lib; valueNew += ";"; std::string& ltEntry = libTypes[ltVar]; if (ltEntry.empty()) { ltEntry = ltValue; } else if (ltEntry != ltValue) { ltEntry = "general"; } } libDepsNew[targetEntry] = valueNew; libDepsOld[targetEntry] = valueOld; } } // Generate dependency information for both old and new style CMake // versions. const char* vertest = "\"${CMAKE_MAJOR_VERSION}.${CMAKE_MINOR_VERSION}\" GREATER 2.4"; fout << "# Generated by CMake\n\n"; fout << "if(" << vertest << ")\n"; fout << " # Information for CMake 2.6 and above.\n"; for (auto const& i : libDepsNew) { if (!i.second.empty()) { fout << " set(\"" << i.first << "\" \"" << i.second << "\")\n"; } } fout << "else()\n"; fout << " # Information for CMake 2.4 and lower.\n"; for (auto const& i : libDepsOld) { if (!i.second.empty()) { fout << " set(\"" << i.first << "\" \"" << i.second << "\")\n"; } } for (auto const& i : libTypes) { if (i.second != "general") { fout << " set(\"" << i.first << "\" \"" << i.second << "\")\n"; } } fout << "endif()\n"; } bool cmExportLibraryDependenciesCommand(std::vector<std::string> const& args, cmExecutionStatus& status) { if (args.empty()) { status.SetError("called with incorrect number of arguments"); return false; } std::string const& filename = args[0]; bool const append = args.size() > 1 && args[1] == "APPEND"; status.GetMakefile().AddGeneratorAction( [filename, append](cmLocalGenerator& lg, const cmListFileBacktrace&) { FinalAction(*lg.GetMakefile(), filename, append); }); return true; }