| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmQtAutoGenGlobalInitializer.h"#include "cmCustomCommandLines.h"#include "cmDuration.h"#include "cmGeneratorTarget.h"#include "cmLocalGenerator.h"#include "cmMakefile.h"#include "cmMessageType.h"#include "cmProcessOutput.h"#include "cmQtAutoGen.h"#include "cmQtAutoGenInitializer.h"#include "cmState.h"#include "cmStateTypes.h"#include "cmStringAlgorithms.h"#include "cmSystemTools.h"#include "cmTarget.h"#include "cm_memory.hxx"#include <utility>cmQtAutoGenGlobalInitializer::Keywords::Keywords()  : AUTOMOC("AUTOMOC")  , AUTOUIC("AUTOUIC")  , AUTORCC("AUTORCC")  , AUTOMOC_EXECUTABLE("AUTOMOC_EXECUTABLE")  , AUTOUIC_EXECUTABLE("AUTOUIC_EXECUTABLE")  , AUTORCC_EXECUTABLE("AUTORCC_EXECUTABLE")  , SKIP_AUTOGEN("SKIP_AUTOGEN")  , SKIP_AUTOMOC("SKIP_AUTOMOC")  , SKIP_AUTOUIC("SKIP_AUTOUIC")  , SKIP_AUTORCC("SKIP_AUTORCC")  , AUTOUIC_OPTIONS("AUTOUIC_OPTIONS")  , AUTORCC_OPTIONS("AUTORCC_OPTIONS")  , qrc("qrc")  , ui("ui"){}cmQtAutoGenGlobalInitializer::cmQtAutoGenGlobalInitializer(  std::vector<cmLocalGenerator*> const& localGenerators){  for (cmLocalGenerator* localGen : localGenerators) {    // Detect global autogen and autorcc target names    bool globalAutoGenTarget = false;    bool globalAutoRccTarget = false;    {      cmMakefile* makefile = localGen->GetMakefile();      // Detect global autogen target name      if (cmSystemTools::IsOn(            makefile->GetSafeDefinition("CMAKE_GLOBAL_AUTOGEN_TARGET"))) {        std::string targetName =          makefile->GetSafeDefinition("CMAKE_GLOBAL_AUTOGEN_TARGET_NAME");        if (targetName.empty()) {          targetName = "autogen";        }        GlobalAutoGenTargets_.emplace(localGen, std::move(targetName));        globalAutoGenTarget = true;      }      // Detect global autorcc target name      if (cmSystemTools::IsOn(            makefile->GetSafeDefinition("CMAKE_GLOBAL_AUTORCC_TARGET"))) {        std::string targetName =          makefile->GetSafeDefinition("CMAKE_GLOBAL_AUTORCC_TARGET_NAME");        if (targetName.empty()) {          targetName = "autorcc";        }        GlobalAutoRccTargets_.emplace(localGen, std::move(targetName));        globalAutoRccTarget = true;      }    }    // Find targets that require AUTOMOC/UIC/RCC processing    for (cmGeneratorTarget* target : localGen->GetGeneratorTargets()) {      // Process only certain target types      switch (target->GetType()) {        case cmStateEnums::EXECUTABLE:        case cmStateEnums::STATIC_LIBRARY:        case cmStateEnums::SHARED_LIBRARY:        case cmStateEnums::MODULE_LIBRARY:        case cmStateEnums::OBJECT_LIBRARY:          // Process target          break;        default:          // Don't process target          continue;      }      if (target->IsImported()) {        // Don't process target        continue;      }      bool const moc = target->GetPropertyAsBool(kw().AUTOMOC);      bool const uic = target->GetPropertyAsBool(kw().AUTOUIC);      bool const rcc = target->GetPropertyAsBool(kw().AUTORCC);      if (moc || uic || rcc) {        std::string const mocExec =          target->GetSafeProperty(kw().AUTOMOC_EXECUTABLE);        std::string const uicExec =          target->GetSafeProperty(kw().AUTOUIC_EXECUTABLE);        std::string const rccExec =          target->GetSafeProperty(kw().AUTORCC_EXECUTABLE);        // We support Qt4, Qt5 and Qt6        auto qtVersion = cmQtAutoGenInitializer::GetQtVersion(target);        bool const validQt = (qtVersion.first.Major == 4) ||          (qtVersion.first.Major == 5) || (qtVersion.first.Major == 6);        bool const mocAvailable = (validQt || !mocExec.empty());        bool const uicAvailable = (validQt || !uicExec.empty());        bool const rccAvailable = (validQt || !rccExec.empty());        bool const mocIsValid = (moc && mocAvailable);        bool const uicIsValid = (uic && uicAvailable);        bool const rccIsValid = (rcc && rccAvailable);        // Disabled AUTOMOC/UIC/RCC warning        bool const mocDisabled = (moc && !mocAvailable);        bool const uicDisabled = (uic && !uicAvailable);        bool const rccDisabled = (rcc && !rccAvailable);        if (mocDisabled || uicDisabled || rccDisabled) {          cmAlphaNum version = (qtVersion.second == 0)            ? cmAlphaNum("<QTVERSION>")            : cmAlphaNum(qtVersion.second);          cmAlphaNum component = uicDisabled ? "Widgets" : "Core";          std::string const msg = cmStrCat(            "AUTOGEN: No valid Qt version found for target ",            target->GetName(), ".  ",            cmQtAutoGen::Tools(mocDisabled, uicDisabled, rccDisabled),            " disabled.  Consider adding:\n", "  find_package(Qt", version,            " COMPONENTS ", component, ")\n", "to your CMakeLists.txt file.");          target->Makefile->IssueMessage(MessageType::AUTHOR_WARNING, msg);        }        if (mocIsValid || uicIsValid || rccIsValid) {          // Create autogen target initializer          Initializers_.emplace_back(cm::make_unique<cmQtAutoGenInitializer>(            this, target, qtVersion.first, mocIsValid, uicIsValid, rccIsValid,            globalAutoGenTarget, globalAutoRccTarget));        }      }    }  }}cmQtAutoGenGlobalInitializer::~cmQtAutoGenGlobalInitializer() = default;void cmQtAutoGenGlobalInitializer::GetOrCreateGlobalTarget(  cmLocalGenerator* localGen, std::string const& name,  std::string const& comment){  // Test if the target already exists  if (localGen->FindGeneratorTargetToUse(name) == nullptr) {    cmMakefile* makefile = localGen->GetMakefile();    // Create utility target    cmTarget* target = makefile->AddUtilityCommand(      name, cmMakefile::TargetOrigin::Generator, true,      makefile->GetHomeOutputDirectory().c_str() /*work dir*/,      std::vector<std::string>() /*output*/,      std::vector<std::string>() /*depends*/, cmCustomCommandLines(), false,      comment.c_str());    localGen->AddGeneratorTarget(new cmGeneratorTarget(target, localGen));    // Set FOLDER property in the target    {      char const* folder =        makefile->GetState()->GetGlobalProperty("AUTOGEN_TARGETS_FOLDER");      if (folder != nullptr) {        target->SetProperty("FOLDER", folder);      }    }  }}void cmQtAutoGenGlobalInitializer::AddToGlobalAutoGen(  cmLocalGenerator* localGen, std::string const& targetName){  auto it = GlobalAutoGenTargets_.find(localGen);  if (it != GlobalAutoGenTargets_.end()) {    cmGeneratorTarget* target = localGen->FindGeneratorTargetToUse(it->second);    if (target != nullptr) {      target->Target->AddUtility(targetName, localGen->GetMakefile());    }  }}void cmQtAutoGenGlobalInitializer::AddToGlobalAutoRcc(  cmLocalGenerator* localGen, std::string const& targetName){  auto it = GlobalAutoRccTargets_.find(localGen);  if (it != GlobalAutoRccTargets_.end()) {    cmGeneratorTarget* target = localGen->FindGeneratorTargetToUse(it->second);    if (target != nullptr) {      target->Target->AddUtility(targetName, localGen->GetMakefile());    }  }}cmQtAutoGen::CompilerFeaturesHandlecmQtAutoGenGlobalInitializer::GetCompilerFeatures(  std::string const& generator, std::string const& executable,  std::string& error){  // Check if we have cached features  {    auto it = this->CompilerFeatures_.find(executable);    if (it != this->CompilerFeatures_.end()) {      return it->second;    }  }  // Check if the executable exists  if (!cmSystemTools::FileExists(executable, true)) {    error = "The \"";    error += generator;    error += "\" executable ";    error += cmQtAutoGen::Quoted(executable);    error += " does not exist.";    return cmQtAutoGen::CompilerFeaturesHandle();  }  // Test the executable  std::string stdOut;  {    std::string stdErr;    std::vector<std::string> command;    command.emplace_back(executable);    command.emplace_back("-h");    int retVal = 0;    const bool runResult = cmSystemTools::RunSingleCommand(      command, &stdOut, &stdErr, &retVal, nullptr, cmSystemTools::OUTPUT_NONE,      cmDuration::zero(), cmProcessOutput::Auto);    if (!runResult) {      error = "Test run of \"";      error += generator;      error += "\" executable ";      error += cmQtAutoGen::Quoted(executable) + " failed.\n";      error += cmQtAutoGen::QuotedCommand(command);      error += "\n";      error += stdOut;      error += "\n";      error += stdErr;      return cmQtAutoGen::CompilerFeaturesHandle();    }  }  // Create valid handle  cmQtAutoGen::CompilerFeaturesHandle res =    std::make_shared<cmQtAutoGen::CompilerFeatures>();  res->HelpOutput = std::move(stdOut);  // Register compiler features  this->CompilerFeatures_.emplace(executable, res);  return res;}bool cmQtAutoGenGlobalInitializer::generate(){  return (InitializeCustomTargets() && SetupCustomTargets());}bool cmQtAutoGenGlobalInitializer::InitializeCustomTargets(){  // Initialize global autogen targets  {    std::string const comment = "Global AUTOGEN target";    for (auto const& pair : GlobalAutoGenTargets_) {      GetOrCreateGlobalTarget(pair.first, pair.second, comment);    }  }  // Initialize global autorcc targets  {    std::string const comment = "Global AUTORCC target";    for (auto const& pair : GlobalAutoRccTargets_) {      GetOrCreateGlobalTarget(pair.first, pair.second, comment);    }  }  // Initialize per target autogen targets  for (auto& initializer : Initializers_) {    if (!initializer->InitCustomTargets()) {      return false;    }  }  return true;}bool cmQtAutoGenGlobalInitializer::SetupCustomTargets(){  for (auto& initializer : Initializers_) {    if (!initializer->SetupCustomTargets()) {      return false;    }  }  return true;}
 |