| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmExportInstallFileGenerator.h"#include "cmAlgorithms.h"#include "cmExportSet.h"#include "cmGeneratedFileStream.h"#include "cmGeneratorExpression.h"#include "cmGeneratorTarget.h"#include "cmGlobalGenerator.h"#include "cmInstallExportGenerator.h"#include "cmInstallTargetGenerator.h"#include "cmLocalGenerator.h"#include "cmMakefile.h"#include "cmPolicies.h"#include "cmStateTypes.h"#include "cmSystemTools.h"#include "cmTarget.h"#include "cmTargetExport.h"#include <sstream>#include <utility>class cmExportSetMap;cmExportInstallFileGenerator::cmExportInstallFileGenerator(  cmInstallExportGenerator* iegen)  : IEGen(iegen){}std::string cmExportInstallFileGenerator::GetConfigImportFileGlob(){  std::string glob = this->FileBase;  glob += "-*";  glob += this->FileExt;  return glob;}bool cmExportInstallFileGenerator::GenerateMainFile(std::ostream& os){  std::vector<cmTargetExport*> allTargets;  {    std::string expectedTargets;    std::string sep;    for (cmTargetExport* te :         *this->IEGen->GetExportSet()->GetTargetExports()) {      expectedTargets += sep + this->Namespace + te->Target->GetExportName();      sep = " ";      if (this->ExportedTargets.insert(te->Target).second) {        allTargets.push_back(te);      } else {        std::ostringstream e;        e << "install(EXPORT \"" << this->IEGen->GetExportSet()->GetName()          << "\" ...) "          << "includes target \"" << te->Target->GetName()          << "\" more than once in the export set.";        cmSystemTools::Error(e.str().c_str());        return false;      }    }    this->GenerateExpectedTargetsCode(os, expectedTargets);  }  // Compute the relative import prefix for the file  this->GenerateImportPrefix(os);  std::vector<std::string> missingTargets;  bool require2_8_12 = false;  bool require3_0_0 = false;  bool require3_1_0 = false;  bool requiresConfigFiles = false;  // Create all the imported targets.  for (cmTargetExport* te : allTargets) {    cmGeneratorTarget* gt = te->Target;    cmStateEnums::TargetType targetType = this->GetExportTargetType(te);    requiresConfigFiles =      requiresConfigFiles || targetType != cmStateEnums::INTERFACE_LIBRARY;    this->GenerateImportTargetCode(os, gt, targetType);    ImportPropertyMap properties;    this->PopulateIncludeDirectoriesInterface(      te, cmGeneratorExpression::InstallInterface, properties, missingTargets);    this->PopulateSourcesInterface(te, cmGeneratorExpression::InstallInterface,                                   properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_SYSTEM_INCLUDE_DIRECTORIES", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_COMPILE_DEFINITIONS", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_COMPILE_OPTIONS", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_AUTOUIC_OPTIONS", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_COMPILE_FEATURES", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    this->PopulateInterfaceProperty("INTERFACE_LINK_OPTIONS", gt,                                    cmGeneratorExpression::InstallInterface,                                    properties, missingTargets);    std::string errorMessage;    if (!this->PopulateExportProperties(gt, properties, errorMessage)) {      cmSystemTools::Error(errorMessage.c_str());      return false;    }    const bool newCMP0022Behavior =      gt->GetPolicyStatusCMP0022() != cmPolicies::WARN &&      gt->GetPolicyStatusCMP0022() != cmPolicies::OLD;    if (newCMP0022Behavior) {      if (this->PopulateInterfaceLinkLibrariesProperty(            gt, cmGeneratorExpression::InstallInterface, properties,            missingTargets) &&          !this->ExportOld) {        require2_8_12 = true;      }    }    if (targetType == cmStateEnums::INTERFACE_LIBRARY) {      require3_0_0 = true;    }    if (gt->GetProperty("INTERFACE_SOURCES")) {      // We can only generate INTERFACE_SOURCES in CMake 3.3, but CMake 3.1      // can consume them.      require3_1_0 = true;    }    this->PopulateInterfaceProperty("INTERFACE_POSITION_INDEPENDENT_CODE", gt,                                    properties);    this->PopulateCompatibleInterfaceProperties(gt, properties);    this->GenerateInterfaceProperties(gt, os, properties);  }  if (require3_1_0) {    this->GenerateRequiredCMakeVersion(os, "3.1.0");  } else if (require3_0_0) {    this->GenerateRequiredCMakeVersion(os, "3.0.0");  } else if (require2_8_12) {    this->GenerateRequiredCMakeVersion(os, "2.8.12");  }  this->LoadConfigFiles(os);  this->CleanupTemporaryVariables(os);  this->GenerateImportedFileCheckLoop(os);  bool result = true;  // Generate an import file for each configuration.  // Don't do this if we only export INTERFACE_LIBRARY targets.  if (requiresConfigFiles) {    for (std::string const& c : this->Configurations) {      if (!this->GenerateImportFileConfig(c, missingTargets)) {        result = false;      }    }  }  this->GenerateMissingTargetsCheckCode(os, missingTargets);  return result;}void cmExportInstallFileGenerator::GenerateImportPrefix(std::ostream& os){  // Set an _IMPORT_PREFIX variable for import location properties  // to reference if they are relative to the install prefix.  std::string installPrefix =    this->IEGen->GetLocalGenerator()->GetMakefile()->GetSafeDefinition(      "CMAKE_INSTALL_PREFIX");  std::string const& expDest = this->IEGen->GetDestination();  if (cmSystemTools::FileIsFullPath(expDest)) {    // The export file is being installed to an absolute path so the    // package is not relocatable.  Use the configured install prefix.    /* clang-format off */    os <<      "# The installation prefix configured by this project.\n"      "set(_IMPORT_PREFIX \"" << installPrefix << "\")\n"      "\n";    /* clang-format on */  } else {    // Add code to compute the installation prefix relative to the    // import file location.    std::string absDest = installPrefix + "/" + expDest;    std::string absDestS = absDest + "/";    os << "# Compute the installation prefix relative to this file.\n"       << "get_filename_component(_IMPORT_PREFIX"       << " \"${CMAKE_CURRENT_LIST_FILE}\" PATH)\n";    if (cmHasLiteralPrefix(absDestS, "/lib/") ||        cmHasLiteralPrefix(absDestS, "/lib64/") ||        cmHasLiteralPrefix(absDestS, "/libx32/") ||        cmHasLiteralPrefix(absDestS, "/usr/lib/") ||        cmHasLiteralPrefix(absDestS, "/usr/lib64/") ||        cmHasLiteralPrefix(absDestS, "/usr/libx32/")) {      // Handle "/usr move" symlinks created by some Linux distros.      /* clang-format off */      os <<        "# Use original install prefix when loaded through a\n"        "# cross-prefix symbolic link such as /lib -> /usr/lib.\n"        "get_filename_component(_realCurr \"${_IMPORT_PREFIX}\" REALPATH)\n"        "get_filename_component(_realOrig \"" << absDest << "\" REALPATH)\n"        "if(_realCurr STREQUAL _realOrig)\n"        "  set(_IMPORT_PREFIX \"" << absDest << "\")\n"        "endif()\n"        "unset(_realOrig)\n"        "unset(_realCurr)\n";      /* clang-format on */    }    std::string dest = expDest;    while (!dest.empty()) {      os << "get_filename_component(_IMPORT_PREFIX \"${_IMPORT_PREFIX}\" "            "PATH)\n";      dest = cmSystemTools::GetFilenamePath(dest);    }    os << "if(_IMPORT_PREFIX STREQUAL \"/\")\n"       << "  set(_IMPORT_PREFIX \"\")\n"       << "endif()\n"       << "\n";  }}void cmExportInstallFileGenerator::CleanupTemporaryVariables(std::ostream& os){  /* clang-format off */  os << "# Cleanup temporary variables.\n"     << "set(_IMPORT_PREFIX)\n"     << "\n";  /* clang-format on */}void cmExportInstallFileGenerator::LoadConfigFiles(std::ostream& os){  // Now load per-configuration properties for them.  /* clang-format off */  os << "# Load information for each installed configuration.\n"     << "get_filename_component(_DIR \"${CMAKE_CURRENT_LIST_FILE}\" PATH)\n"     << "file(GLOB CONFIG_FILES \"${_DIR}/"     << this->GetConfigImportFileGlob() << "\")\n"     << "foreach(f ${CONFIG_FILES})\n"     << "  include(${f})\n"     << "endforeach()\n"     << "\n";  /* clang-format on */}void cmExportInstallFileGenerator::ReplaceInstallPrefix(std::string& input){  std::string::size_type pos = 0;  std::string::size_type lastPos = pos;  while ((pos = input.find("$<INSTALL_PREFIX>", lastPos)) !=         std::string::npos) {    std::string::size_type endPos = pos + sizeof("$<INSTALL_PREFIX>") - 1;    input.replace(pos, endPos - pos, "${_IMPORT_PREFIX}");    lastPos = endPos;  }}bool cmExportInstallFileGenerator::GenerateImportFileConfig(  const std::string& config, std::vector<std::string>& missingTargets){  // Skip configurations not enabled for this export.  if (!this->IEGen->InstallsForConfig(config)) {    return true;  }  // Construct the name of the file to generate.  std::string fileName = this->FileDir;  fileName += "/";  fileName += this->FileBase;  fileName += "-";  if (!config.empty()) {    fileName += cmSystemTools::LowerCase(config);  } else {    fileName += "noconfig";  }  fileName += this->FileExt;  // Open the output file to generate it.  cmGeneratedFileStream exportFileStream(fileName.c_str(), true);  if (!exportFileStream) {    std::string se = cmSystemTools::GetLastSystemError();    std::ostringstream e;    e << "cannot write to file \"" << fileName << "\": " << se;    cmSystemTools::Error(e.str().c_str());    return false;  }  std::ostream& os = exportFileStream;  // Start with the import file header.  this->GenerateImportHeaderCode(os, config);  // Generate the per-config target information.  this->GenerateImportConfig(os, config, missingTargets);  // End with the import file footer.  this->GenerateImportFooterCode(os);  // Record this per-config import file.  this->ConfigImportFiles[config] = fileName;  return true;}void cmExportInstallFileGenerator::GenerateImportTargetsConfig(  std::ostream& os, const std::string& config, std::string const& suffix,  std::vector<std::string>& missingTargets){  // Add each target in the set to the export.  for (cmTargetExport* te : *this->IEGen->GetExportSet()->GetTargetExports()) {    // Collect import properties for this target.    if (this->GetExportTargetType(te) == cmStateEnums::INTERFACE_LIBRARY) {      continue;    }    ImportPropertyMap properties;    std::set<std::string> importedLocations;    this->SetImportLocationProperty(config, suffix, te->ArchiveGenerator,                                    properties, importedLocations);    this->SetImportLocationProperty(config, suffix, te->LibraryGenerator,                                    properties, importedLocations);    this->SetImportLocationProperty(config, suffix, te->RuntimeGenerator,                                    properties, importedLocations);    this->SetImportLocationProperty(config, suffix, te->ObjectsGenerator,                                    properties, importedLocations);    this->SetImportLocationProperty(config, suffix, te->FrameworkGenerator,                                    properties, importedLocations);    this->SetImportLocationProperty(config, suffix, te->BundleGenerator,                                    properties, importedLocations);    // If any file location was set for the target add it to the    // import file.    if (!properties.empty()) {      // Get the rest of the target details.      cmGeneratorTarget* gtgt = te->Target;      this->SetImportDetailProperties(config, suffix, gtgt, properties,                                      missingTargets);      this->SetImportLinkInterface(config, suffix,                                   cmGeneratorExpression::InstallInterface,                                   gtgt, properties, missingTargets);      // TODO: PUBLIC_HEADER_LOCATION      // This should wait until the build feature propagation stuff      // is done.  Then this can be a propagated include directory.      // this->GenerateImportProperty(config, te->HeaderGenerator,      //                              properties);      // Generate code in the export file.      this->GenerateImportPropertyCode(os, config, gtgt, properties);      this->GenerateImportedFileChecksCode(os, gtgt, properties,                                           importedLocations);    }  }}void cmExportInstallFileGenerator::SetImportLocationProperty(  const std::string& config, std::string const& suffix,  cmInstallTargetGenerator* itgen, ImportPropertyMap& properties,  std::set<std::string>& importedLocations){  // Skip rules that do not match this configuration.  if (!(itgen && itgen->InstallsForConfig(config))) {    return;  }  // Get the target to be installed.  cmGeneratorTarget* target = itgen->GetTarget();  // Construct the installed location of the target.  std::string dest = itgen->GetDestination(config);  std::string value;  if (!cmSystemTools::FileIsFullPath(dest)) {    // The target is installed relative to the installation prefix.    value = "${_IMPORT_PREFIX}/";  }  value += dest;  value += "/";  if (itgen->IsImportLibrary()) {    // Construct the property name.    std::string prop = "IMPORTED_IMPLIB";    prop += suffix;    // Append the installed file name.    value += itgen->GetInstallFilename(target, config,                                       cmInstallTargetGenerator::NameImplib);    // Store the property.    properties[prop] = value;    importedLocations.insert(prop);  } else if (itgen->GetTarget()->GetType() == cmStateEnums::OBJECT_LIBRARY) {    // Construct the property name.    std::string prop = "IMPORTED_OBJECTS";    prop += suffix;    // Compute all the object files inside this target and setup    // IMPORTED_OBJECTS as a list of object files    std::vector<std::string> objects;    itgen->GetInstallObjectNames(config, objects);    for (std::string& obj : objects) {      obj = value + obj;    }    // Store the property.    properties[prop] = cmJoin(objects, ";");    importedLocations.insert(prop);  } else {    // Construct the property name.    std::string prop = "IMPORTED_LOCATION";    prop += suffix;    // Append the installed file name.    if (target->IsAppBundleOnApple()) {      value += itgen->GetInstallFilename(target, config);      value += ".app/Contents/MacOS/";      value += itgen->GetInstallFilename(target, config);    } else {      value += itgen->GetInstallFilename(target, config,                                         cmInstallTargetGenerator::NameReal);    }    // Store the property.    properties[prop] = value;    importedLocations.insert(prop);  }}cmStateEnums::TargetType cmExportInstallFileGenerator::GetExportTargetType(  cmTargetExport const* targetExport) const{  cmStateEnums::TargetType targetType = targetExport->Target->GetType();  // An OBJECT library installed with no OBJECTS DESTINATION  // is transformed to an INTERFACE library.  if (targetType == cmStateEnums::OBJECT_LIBRARY &&      targetExport->ObjectsGenerator == nullptr) {    targetType = cmStateEnums::INTERFACE_LIBRARY;  }  return targetType;}void cmExportInstallFileGenerator::HandleMissingTarget(  std::string& link_libs, std::vector<std::string>& missingTargets,  cmGeneratorTarget* depender, cmGeneratorTarget* dependee){  const std::string name = dependee->GetName();  cmGlobalGenerator* gg = dependee->GetLocalGenerator()->GetGlobalGenerator();  std::vector<std::string> namespaces = this->FindNamespaces(gg, name);  int targetOccurrences = static_cast<int>(namespaces.size());  if (targetOccurrences == 1) {    std::string missingTarget = namespaces[0];    missingTarget += dependee->GetExportName();    link_libs += missingTarget;    missingTargets.push_back(std::move(missingTarget));  } else {    // All exported targets should be known here and should be unique.    // This is probably user-error.    this->ComplainAboutMissingTarget(depender, dependee, targetOccurrences);  }}std::vector<std::string> cmExportInstallFileGenerator::FindNamespaces(  cmGlobalGenerator* gg, const std::string& name){  std::vector<std::string> namespaces;  const cmExportSetMap& exportSets = gg->GetExportSets();  for (auto const& expIt : exportSets) {    const cmExportSet* exportSet = expIt.second;    std::vector<cmTargetExport*> const* targets =      exportSet->GetTargetExports();    bool containsTarget = false;    for (cmTargetExport* target : *targets) {      if (name == target->TargetName) {        containsTarget = true;        break;      }    }    if (containsTarget) {      std::vector<cmInstallExportGenerator const*> const* installs =        exportSet->GetInstallations();      for (cmInstallExportGenerator const* install : *installs) {        namespaces.push_back(install->GetNamespace());      }    }  }  return namespaces;}void cmExportInstallFileGenerator::ComplainAboutMissingTarget(  cmGeneratorTarget* depender, cmGeneratorTarget* dependee, int occurrences){  std::ostringstream e;  e << "install(EXPORT \"" << this->IEGen->GetExportSet()->GetName()    << "\" ...) "    << "includes target \"" << depender->GetName()    << "\" which requires target \"" << dependee->GetName() << "\" ";  if (occurrences == 0) {    e << "that is not in the export set.";  } else {    e << "that is not in this export set, but " << occurrences      << " times in others.";  }  cmSystemTools::Error(e.str().c_str());}std::string cmExportInstallFileGenerator::InstallNameDir(  cmGeneratorTarget* target, const std::string& /*config*/){  std::string install_name_dir;  cmMakefile* mf = target->Target->GetMakefile();  if (mf->IsOn("CMAKE_PLATFORM_HAS_INSTALLNAME")) {    install_name_dir = target->GetInstallNameDirForInstallTree();  }  return install_name_dir;}
 |