| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmQtAutoGen.h"#include "cmQtAutoGenInitializer.h"#include "cmAlgorithms.h"#include "cmCustomCommand.h"#include "cmCustomCommandLines.h"#include "cmDuration.h"#include "cmFilePathChecksum.h"#include "cmGeneratorTarget.h"#include "cmGlobalGenerator.h"#include "cmLinkItem.h"#include "cmLocalGenerator.h"#include "cmMakefile.h"#include "cmOutputConverter.h"#include "cmPolicies.h"#include "cmProcessOutput.h"#include "cmSourceFile.h"#include "cmSourceGroup.h"#include "cmState.h"#include "cmStateTypes.h"#include "cmSystemTools.h"#include "cmTarget.h"#include "cm_sys_stat.h"#include "cmake.h"#include "cmsys/FStream.hxx"#include "cmsys/SystemInformation.hxx"#include <algorithm>#include <array>#include <deque>#include <map>#include <set>#include <sstream>#include <string>#include <utility>#include <vector>inline static const char* SafeString(const char* value){  return (value != nullptr) ? value : "";}inline static std::string GetSafeProperty(cmGeneratorTarget const* target,                                          const char* key){  return std::string(SafeString(target->GetProperty(key)));}inline static std::string GetSafeProperty(cmSourceFile const* sf,                                          const char* key){  return std::string(SafeString(sf->GetProperty(key)));}static std::size_t GetParallelCPUCount(){  static std::size_t count = 0;  // Detect only on the first call  if (count == 0) {    cmsys::SystemInformation info;    info.RunCPUCheck();    count = info.GetNumberOfPhysicalCPU();    count = std::max<std::size_t>(count, 1);    count = std::min<std::size_t>(count, cmQtAutoGen::ParallelMax);  }  return count;}static void AddDefinitionEscaped(cmMakefile* makefile, const char* key,                                 std::string const& value){  makefile->AddDefinition(key,                          cmOutputConverter::EscapeForCMake(value).c_str());}static void AddDefinitionEscaped(cmMakefile* makefile, const char* key,                                 const std::vector<std::string>& values){  makefile->AddDefinition(    key, cmOutputConverter::EscapeForCMake(cmJoin(values, ";")).c_str());}static void AddDefinitionEscaped(cmMakefile* makefile, const char* key,                                 const std::set<std::string>& values){  makefile->AddDefinition(    key, cmOutputConverter::EscapeForCMake(cmJoin(values, ";")).c_str());}static void AddDefinitionEscaped(  cmMakefile* makefile, const char* key,  const std::vector<std::vector<std::string>>& lists){  std::vector<std::string> seplist;  for (const std::vector<std::string>& list : lists) {    std::string blist = "{";    blist += cmJoin(list, ";");    blist += "}";    seplist.push_back(std::move(blist));  }  makefile->AddDefinition(key, cmOutputConverter::EscapeForCMake(                                 cmJoin(seplist, cmQtAutoGen::ListSep))                                 .c_str());}static bool AddToSourceGroup(cmMakefile* makefile, std::string const& fileName,                             cmQtAutoGen::GeneratorT genType){  cmSourceGroup* sourceGroup = nullptr;  // Acquire source group  {    std::string property;    std::string groupName;    {      std::array<std::string, 2> props;      // Use generator specific group name      switch (genType) {        case cmQtAutoGen::GeneratorT::MOC:          props[0] = "AUTOMOC_SOURCE_GROUP";          break;        case cmQtAutoGen::GeneratorT::RCC:          props[0] = "AUTORCC_SOURCE_GROUP";          break;        default:          props[0] = "AUTOGEN_SOURCE_GROUP";          break;      }      props[1] = "AUTOGEN_SOURCE_GROUP";      for (std::string& prop : props) {        const char* propName = makefile->GetState()->GetGlobalProperty(prop);        if ((propName != nullptr) && (*propName != '\0')) {          groupName = propName;          property = std::move(prop);          break;        }      }    }    // Generate a source group on demand    if (!groupName.empty()) {      sourceGroup = makefile->GetOrCreateSourceGroup(groupName);      if (sourceGroup == nullptr) {        std::ostringstream ost;        ost << cmQtAutoGen::GeneratorNameUpper(genType);        ost << ": " << property;        ost << ": Could not find or create the source group ";        ost << cmQtAutoGen::Quoted(groupName);        cmSystemTools::Error(ost.str().c_str());        return false;      }    }  }  if (sourceGroup != nullptr) {    sourceGroup->AddGroupFile(fileName);  }  return true;}static void AddCleanFile(cmMakefile* makefile, std::string const& fileName){  makefile->AppendProperty("ADDITIONAL_MAKE_CLEAN_FILES", fileName.c_str(),                           false);}static std::string FileProjectRelativePath(cmMakefile* makefile,                                           std::string const& fileName){  std::string res;  {    std::string pSource = cmSystemTools::RelativePath(      makefile->GetCurrentSourceDirectory(), fileName.c_str());    std::string pBinary = cmSystemTools::RelativePath(      makefile->GetCurrentBinaryDirectory(), fileName.c_str());    if (pSource.size() < pBinary.size()) {      res = std::move(pSource);    } else if (pBinary.size() < fileName.size()) {      res = std::move(pBinary);    } else {      res = fileName;    }  }  return res;}/* @brief Tests if targetDepend is a STATIC_LIBRARY and if any of its * recursive STATIC_LIBRARY dependencies depends on targetOrigin * (STATIC_LIBRARY cycle). */static bool StaticLibraryCycle(cmGeneratorTarget const* targetOrigin,                               cmGeneratorTarget const* targetDepend,                               std::string const& config){  bool cycle = false;  if ((targetOrigin->GetType() == cmStateEnums::STATIC_LIBRARY) &&      (targetDepend->GetType() == cmStateEnums::STATIC_LIBRARY)) {    std::set<cmGeneratorTarget const*> knownLibs;    std::deque<cmGeneratorTarget const*> testLibs;    // Insert initial static_library dependency    knownLibs.insert(targetDepend);    testLibs.push_back(targetDepend);    while (!testLibs.empty()) {      cmGeneratorTarget const* testTarget = testLibs.front();      testLibs.pop_front();      // Check if the test target is the origin target (cycle)      if (testTarget == targetOrigin) {        cycle = true;        break;      }      // Collect all static_library dependencies from the test target      cmLinkImplementationLibraries const* libs =        testTarget->GetLinkImplementationLibraries(config);      if (libs != nullptr) {        for (cmLinkItem const& item : libs->Libraries) {          cmGeneratorTarget const* depTarget = item.Target;          if ((depTarget != nullptr) &&              (depTarget->GetType() == cmStateEnums::STATIC_LIBRARY) &&              knownLibs.insert(depTarget).second) {            testLibs.push_back(depTarget);          }        }      }    }  }  return cycle;}cmQtAutoGenInitializer::cmQtAutoGenInitializer(  cmGeneratorTarget* target, bool mocEnabled, bool uicEnabled, bool rccEnabled,  std::string const& qtVersionMajor)  : Target(target)  , MocEnabled(mocEnabled)  , UicEnabled(uicEnabled)  , RccEnabled(rccEnabled)  , QtVersionMajor(qtVersionMajor)  , MultiConfig(MultiConfigT::WRAPPER){  this->QtVersionMinor =    cmQtAutoGenInitializer::GetQtMinorVersion(target, this->QtVersionMajor);}void cmQtAutoGenInitializer::InitCustomTargets(){  cmMakefile* makefile = this->Target->Target->GetMakefile();  cmLocalGenerator* localGen = this->Target->GetLocalGenerator();  cmGlobalGenerator* globalGen = localGen->GetGlobalGenerator();  // Configurations  this->ConfigDefault = makefile->GetConfigurations(this->ConfigsList);  if (this->ConfigsList.empty()) {    this->ConfigsList.push_back(this->ConfigDefault);  }  // Multi configuration  {    if (!globalGen->IsMultiConfig()) {      this->MultiConfig = MultiConfigT::SINGLE;    }    // FIXME: Xcode does not support per-config sources, yet.    //        (EXCLUDED_SOURCE_FILE_NAMES)    // if (globalGen->GetName().find("Xcode") != std::string::npos) {    //  return MultiConfigT::MULTI;    //}    // FIXME: Visual Studio does not support per-config sources, yet.    //        (EXCLUDED_SOURCE_FILE_NAMES)    // if (globalGen->GetName().find("Visual Studio") != std::string::npos) {    //  return MultiConfigT::MULTI;    //}  }  // Autogen target name  this->AutogenTargetName = this->Target->GetName();  this->AutogenTargetName += "_autogen";  // Autogen directories  {    // Collapsed current binary directory    std::string const cbd = cmSystemTools::CollapseFullPath(      "", makefile->GetCurrentBinaryDirectory());    // Autogen info dir    this->DirInfo = cbd;    this->DirInfo += makefile->GetCMakeInstance()->GetCMakeFilesDirectory();    this->DirInfo += "/";    this->DirInfo += this->AutogenTargetName;    this->DirInfo += ".dir";    cmSystemTools::ConvertToUnixSlashes(this->DirInfo);    // Autogen build dir    this->DirBuild = GetSafeProperty(this->Target, "AUTOGEN_BUILD_DIR");    if (this->DirBuild.empty()) {      this->DirBuild = cbd;      this->DirBuild += "/";      this->DirBuild += this->AutogenTargetName;    }    cmSystemTools::ConvertToUnixSlashes(this->DirBuild);    // Working directory    this->DirWork = cbd;    cmSystemTools::ConvertToUnixSlashes(this->DirWork);  }  // Autogen files  {    this->AutogenInfoFile = this->DirInfo;    this->AutogenInfoFile += "/AutogenInfo.cmake";    this->AutogenSettingsFile = this->DirInfo;    this->AutogenSettingsFile += "/AutogenOldSettings.txt";  }  // Autogen target FOLDER property  {    const char* folder =      makefile->GetState()->GetGlobalProperty("AUTOMOC_TARGETS_FOLDER");    if (folder == nullptr) {      folder =        makefile->GetState()->GetGlobalProperty("AUTOGEN_TARGETS_FOLDER");    }    // Inherit FOLDER property from target (#13688)    if (folder == nullptr) {      folder = SafeString(this->Target->Target->GetProperty("FOLDER"));    }    if (folder != nullptr) {      this->AutogenFolder = folder;    }  }  std::set<std::string> autogenDependFiles;  std::set<cmTarget*> autogenDependTargets;  std::vector<std::string> autogenProvides;  // Remove build directories on cleanup  AddCleanFile(makefile, this->DirBuild);  // Remove old settings on cleanup  {    std::string base = this->DirInfo;    base += "/AutogenOldSettings";    if (this->MultiConfig == MultiConfigT::SINGLE) {      AddCleanFile(makefile, base.append(".cmake"));    } else {      for (std::string const& cfg : this->ConfigsList) {        std::string filename = base;        filename += "_";        filename += cfg;        filename += ".cmake";        AddCleanFile(makefile, filename);      }    }  }  // Add moc compilation to generated files list  if (this->MocEnabled) {    std::string const mocsComp = this->DirBuild + "/mocs_compilation.cpp";    auto files = this->AddGeneratedSource(mocsComp, GeneratorT::MOC);    for (std::string& file : files) {      autogenProvides.push_back(std::move(file));    }  }  // Add autogen includes directory to the origin target INCLUDE_DIRECTORIES  if (this->MocEnabled || this->UicEnabled) {    std::string includeDir = this->DirBuild + "/include";    if (this->MultiConfig != MultiConfigT::SINGLE) {      includeDir += "_$<CONFIG>";    }    this->Target->AddIncludeDirectory(includeDir, true);  }  // Acquire rcc executable and features  if (this->RccEnabled) {    {      std::string err;      if (this->QtVersionMajor == "5") {        cmGeneratorTarget* tgt =          localGen->FindGeneratorTargetToUse("Qt5::rcc");        if (tgt != nullptr) {          this->RccExecutable = SafeString(tgt->ImportedGetLocation(""));        } else {          err = "AUTORCC: Qt5::rcc target not found";        }      } else if (QtVersionMajor == "4") {        cmGeneratorTarget* tgt =          localGen->FindGeneratorTargetToUse("Qt4::rcc");        if (tgt != nullptr) {          this->RccExecutable = SafeString(tgt->ImportedGetLocation(""));        } else {          err = "AUTORCC: Qt4::rcc target not found";        }      } else {        err = "The AUTORCC feature supports only Qt 4 and Qt 5";      }      if (!err.empty()) {        err += " (";        err += this->Target->GetName();        err += ")";        cmSystemTools::Error(err.c_str());      }    }    // Detect if rcc supports (-)-list    if (!this->RccExecutable.empty() && (this->QtVersionMajor == "5")) {      std::vector<std::string> command;      command.push_back(this->RccExecutable);      command.push_back("--help");      std::string rccStdOut;      std::string rccStdErr;      int retVal = 0;      bool result = cmSystemTools::RunSingleCommand(        command, &rccStdOut, &rccStdErr, &retVal, nullptr,        cmSystemTools::OUTPUT_NONE, cmDuration::zero(), cmProcessOutput::Auto);      if (result && retVal == 0 &&          rccStdOut.find("--list") != std::string::npos) {        this->RccListOptions.push_back("--list");      } else {        this->RccListOptions.push_back("-list");      }    }  }  // Extract relevant source files  std::vector<std::string> generatedSources;  std::vector<std::string> generatedHeaders;  {    std::string const qrcExt = "qrc";    std::vector<cmSourceFile*> srcFiles;    this->Target->GetConfigCommonSourceFiles(srcFiles);    for (cmSourceFile* sf : srcFiles) {      if (sf->GetPropertyAsBool("SKIP_AUTOGEN")) {        continue;      }      // sf->GetExtension() is only valid after sf->GetFullPath() ...      std::string const& fPath = sf->GetFullPath();      std::string const& ext = sf->GetExtension();      // Register generated files that will be scanned by moc or uic      if (this->MocEnabled || this->UicEnabled) {        cmSystemTools::FileFormat const fileType =          cmSystemTools::GetFileFormat(ext.c_str());        if ((fileType == cmSystemTools::CXX_FILE_FORMAT) ||            (fileType == cmSystemTools::HEADER_FILE_FORMAT)) {          std::string const absPath = cmSystemTools::GetRealPath(fPath);          if ((this->MocEnabled && !sf->GetPropertyAsBool("SKIP_AUTOMOC")) ||              (this->UicEnabled && !sf->GetPropertyAsBool("SKIP_AUTOUIC"))) {            // Register source            const bool generated = sf->GetPropertyAsBool("GENERATED");            if (fileType == cmSystemTools::HEADER_FILE_FORMAT) {              if (generated) {                generatedHeaders.push_back(absPath);              } else {                this->Headers.push_back(absPath);              }            } else {              if (generated) {                generatedSources.push_back(absPath);              } else {                this->Sources.push_back(absPath);              }            }          }        }      }      // Register rcc enabled files      if (this->RccEnabled && (ext == qrcExt) &&          !sf->GetPropertyAsBool("SKIP_AUTORCC")) {        // Register qrc file        {          Qrc qrc;          qrc.QrcFile = cmSystemTools::GetRealPath(fPath);          qrc.QrcName =            cmSystemTools::GetFilenameWithoutLastExtension(qrc.QrcFile);          qrc.Generated = sf->GetPropertyAsBool("GENERATED");          // RCC options          {            std::string const opts = GetSafeProperty(sf, "AUTORCC_OPTIONS");            if (!opts.empty()) {              cmSystemTools::ExpandListArgument(opts, qrc.Options);            }          }          this->Qrcs.push_back(std::move(qrc));        }      }    }    // cmGeneratorTarget::GetConfigCommonSourceFiles computes the target's    // sources meta data cache. Clear it so that OBJECT library targets that    // are AUTOGEN initialized after this target get their added    // mocs_compilation.cpp source acknowledged by this target.    this->Target->ClearSourcesCache();  }  // Read skip files from makefile sources  if (this->MocEnabled || this->UicEnabled) {    std::string pathError;    for (cmSourceFile* sf : makefile->GetSourceFiles()) {      // sf->GetExtension() is only valid after sf->GetFullPath() ...      // Since we're iterating over source files that might be not in the      // target we need to check for path errors (not existing files).      std::string const& fPath = sf->GetFullPath(&pathError);      if (!pathError.empty()) {        pathError.clear();        continue;      }      cmSystemTools::FileFormat const fileType =        cmSystemTools::GetFileFormat(sf->GetExtension().c_str());      if (!(fileType == cmSystemTools::CXX_FILE_FORMAT) &&          !(fileType == cmSystemTools::HEADER_FILE_FORMAT)) {        continue;      }      const bool skipAll = sf->GetPropertyAsBool("SKIP_AUTOGEN");      const bool mocSkip =        this->MocEnabled && (skipAll || sf->GetPropertyAsBool("SKIP_AUTOMOC"));      const bool uicSkip =        this->UicEnabled && (skipAll || sf->GetPropertyAsBool("SKIP_AUTOUIC"));      if (mocSkip || uicSkip) {        std::string const absFile = cmSystemTools::GetRealPath(fPath);        if (mocSkip) {          this->MocSkip.insert(absFile);        }        if (uicSkip) {          this->UicSkip.insert(absFile);        }      }    }  }  // Process GENERATED sources and headers  if (!generatedSources.empty() || !generatedHeaders.empty()) {    // Check status of policy CMP0071    bool policyAccept = false;    bool policyWarn = false;    cmPolicies::PolicyStatus const CMP0071_status =      makefile->GetPolicyStatus(cmPolicies::CMP0071);    switch (CMP0071_status) {      case cmPolicies::WARN:        policyWarn = true;        CM_FALLTHROUGH;      case cmPolicies::OLD:        // Ignore GENERATED file        break;      case cmPolicies::REQUIRED_IF_USED:      case cmPolicies::REQUIRED_ALWAYS:      case cmPolicies::NEW:        // Process GENERATED file        policyAccept = true;        break;    }    if (policyAccept) {      // Accept GENERATED sources      for (std::string const& absFile : generatedHeaders) {        this->Headers.push_back(absFile);        autogenDependFiles.insert(absFile);      }      for (std::string const& absFile : generatedSources) {        this->Sources.push_back(absFile);        autogenDependFiles.insert(absFile);      }    } else {      if (policyWarn) {        std::string msg;        msg += cmPolicies::GetPolicyWarning(cmPolicies::CMP0071);        msg += "\n";        std::string tools;        std::string property;        if (this->MocEnabled && this->UicEnabled) {          tools = "AUTOMOC and AUTOUIC";          property = "SKIP_AUTOGEN";        } else if (this->MocEnabled) {          tools = "AUTOMOC";          property = "SKIP_AUTOMOC";        } else if (this->UicEnabled) {          tools = "AUTOUIC";          property = "SKIP_AUTOUIC";        }        msg += "For compatibility, CMake is excluding the GENERATED source "               "file(s):\n";        for (const std::string& absFile : generatedHeaders) {          msg.append("  ").append(Quoted(absFile)).append("\n");        }        for (const std::string& absFile : generatedSources) {          msg.append("  ").append(Quoted(absFile)).append("\n");        }        msg += "from processing by ";        msg += tools;        msg +=          ". If any of the files should be processed, set CMP0071 to NEW. "          "If any of the files should not be processed, "          "explicitly exclude them by setting the source file property ";        msg += property;        msg += ":\n  set_property(SOURCE file.h PROPERTY ";        msg += property;        msg += " ON)\n";        makefile->IssueMessage(cmake::AUTHOR_WARNING, msg);      }    }    // Clear lists    generatedSources.clear();    generatedHeaders.clear();  }  // Sort headers and sources  if (this->MocEnabled || this->UicEnabled) {    std::sort(this->Headers.begin(), this->Headers.end());    std::sort(this->Sources.begin(), this->Sources.end());  }  // Process qrc files  if (!this->Qrcs.empty()) {    const bool QtV5 = (this->QtVersionMajor == "5");    // Target rcc options    std::vector<std::string> optionsTarget;    cmSystemTools::ExpandListArgument(      GetSafeProperty(this->Target, "AUTORCC_OPTIONS"), optionsTarget);    // Check if file name is unique    for (Qrc& qrc : this->Qrcs) {      qrc.Unique = true;      for (Qrc const& qrc2 : this->Qrcs) {        if ((&qrc != &qrc2) && (qrc.QrcName == qrc2.QrcName)) {          qrc.Unique = false;          break;        }      }    }    // Path checksum and file names    {      cmFilePathChecksum const fpathCheckSum(makefile);      for (Qrc& qrc : this->Qrcs) {        qrc.PathChecksum = fpathCheckSum.getPart(qrc.QrcFile);        // RCC output file name        {          std::string rccFile = this->DirBuild + "/";          rccFile += qrc.PathChecksum;          rccFile += "/qrc_";          rccFile += qrc.QrcName;          rccFile += ".cpp";          qrc.RccFile = std::move(rccFile);        }        {          std::string base = this->DirInfo;          base += "/RCC";          base += qrc.QrcName;          if (!qrc.Unique) {            base += qrc.PathChecksum;          }          qrc.InfoFile = base;          qrc.InfoFile += "Info.cmake";          qrc.SettingsFile = base;          qrc.SettingsFile += "Settings.txt";        }      }    }    // RCC options    for (Qrc& qrc : this->Qrcs) {      // Target options      std::vector<std::string> opts = optionsTarget;      // Merge computed "-name XYZ" option      {        std::string name = qrc.QrcName;        // Replace '-' with '_'. The former is not valid for symbol names.        std::replace(name.begin(), name.end(), '-', '_');        if (!qrc.Unique) {          name += "_";          name += qrc.PathChecksum;        }        std::vector<std::string> nameOpts;        nameOpts.emplace_back("-name");        nameOpts.emplace_back(std::move(name));        RccMergeOptions(opts, nameOpts, QtV5);      }      // Merge file option      RccMergeOptions(opts, qrc.Options, QtV5);      qrc.Options = std::move(opts);    }    for (Qrc& qrc : this->Qrcs) {      // Register file at target      std::vector<std::string> const ccOutput =        this->AddGeneratedSource(qrc.RccFile, GeneratorT::RCC);      cmCustomCommandLines commandLines;      {        cmCustomCommandLine currentLine;        currentLine.push_back(cmSystemTools::GetCMakeCommand());        currentLine.push_back("-E");        currentLine.push_back("cmake_autorcc");        currentLine.push_back(qrc.InfoFile);        currentLine.push_back("$<CONFIGURATION>");        commandLines.push_back(std::move(currentLine));      }      std::string ccComment = "Automatic RCC for ";      ccComment += FileProjectRelativePath(makefile, qrc.QrcFile);      if (qrc.Generated) {        // Create custom rcc target        std::string ccName;        {          ccName = this->Target->GetName();          ccName += "_arcc_";          ccName += qrc.QrcName;          if (!qrc.Unique) {            ccName += "_";            ccName += qrc.PathChecksum;          }          std::vector<std::string> ccDepends;          // Add the .qrc and info file to the custom target dependencies          ccDepends.push_back(qrc.QrcFile);          ccDepends.push_back(qrc.InfoFile);          cmTarget* autoRccTarget = makefile->AddUtilityCommand(            ccName, cmMakefile::TargetOrigin::Generator, true,            this->DirWork.c_str(), ccOutput, ccDepends, commandLines, false,            ccComment.c_str());          // Create autogen generator target          localGen->AddGeneratorTarget(            new cmGeneratorTarget(autoRccTarget, localGen));          // Set FOLDER property in autogen target          if (!this->AutogenFolder.empty()) {            autoRccTarget->SetProperty("FOLDER", this->AutogenFolder.c_str());          }        }        // Add autogen target to the origin target dependencies        this->Target->Target->AddUtility(ccName, makefile);      } else {        // Create custom rcc command        {          std::vector<std::string> ccByproducts;          std::vector<std::string> ccDepends;          // Add the .qrc and info file to the custom command dependencies          ccDepends.push_back(qrc.QrcFile);          ccDepends.push_back(qrc.InfoFile);          // Add the resource files to the dependencies          {            std::string error;            if (RccListInputs(qrc.QrcFile, qrc.Resources, error)) {              for (std::string const& fileName : qrc.Resources) {                // Add resource file to the custom command dependencies                ccDepends.push_back(fileName);              }            } else {              cmSystemTools::Error(error.c_str());            }          }          makefile->AddCustomCommandToOutput(ccOutput, ccByproducts, ccDepends,                                             /*main_dependency*/ std::string(),                                             commandLines, ccComment.c_str(),                                             this->DirWork.c_str());        }        // Reconfigure when .qrc file changes        makefile->AddCMakeDependFile(qrc.QrcFile);      }    }  }  // Create _autogen target  if (this->MocEnabled || this->UicEnabled) {    // Add user defined autogen target dependencies    {      std::string const deps =        GetSafeProperty(this->Target, "AUTOGEN_TARGET_DEPENDS");      if (!deps.empty()) {        std::vector<std::string> extraDeps;        cmSystemTools::ExpandListArgument(deps, extraDeps);        for (std::string const& depName : extraDeps) {          // Allow target and file dependencies          auto* depTarget = makefile->FindTargetToUse(depName);          if (depTarget != nullptr) {            autogenDependTargets.insert(depTarget);          } else {            autogenDependFiles.insert(depName);          }        }      }    }    // Compose target comment    std::string autogenComment;    {      std::string tools;      if (this->MocEnabled) {        tools += "MOC";      }      if (this->UicEnabled) {        if (!tools.empty()) {          tools += " and ";        }        tools += "UIC";      }      autogenComment = "Automatic ";      autogenComment += tools;      autogenComment += " for target ";      autogenComment += this->Target->GetName();    }    // Compose command lines    cmCustomCommandLines commandLines;    {      cmCustomCommandLine currentLine;      currentLine.push_back(cmSystemTools::GetCMakeCommand());      currentLine.push_back("-E");      currentLine.push_back("cmake_autogen");      currentLine.push_back(this->AutogenInfoFile);      currentLine.push_back("$<CONFIGURATION>");      commandLines.push_back(std::move(currentLine));    }    // Use PRE_BUILD on demand    bool usePRE_BUILD = false;    if (globalGen->GetName().find("Visual Studio") != std::string::npos) {      // Under VS use a PRE_BUILD event instead of a separate target to      // reduce the number of targets loaded into the IDE.      // This also works around a VS 11 bug that may skip updating the target:      //  https://connect.microsoft.com/VisualStudio/feedback/details/769495      usePRE_BUILD = true;    }    // Disable PRE_BUILD in some cases    if (usePRE_BUILD) {      // Cannot use PRE_BUILD with file depends      if (!autogenDependFiles.empty()) {        usePRE_BUILD = false;      }    }    // Create the autogen target/command    if (usePRE_BUILD) {      // Add additional autogen target dependencies to origin target      for (cmTarget* depTarget : autogenDependTargets) {        this->Target->Target->AddUtility(depTarget->GetName(), makefile);      }      // Add the pre-build command directly to bypass the OBJECT_LIBRARY      // rejection in cmMakefile::AddCustomCommandToTarget because we know      // PRE_BUILD will work for an OBJECT_LIBRARY in this specific case.      //      // PRE_BUILD does not support file dependencies!      const std::vector<std::string> no_output;      const std::vector<std::string> no_deps;      cmCustomCommand cc(makefile, no_output, autogenProvides, no_deps,                         commandLines, autogenComment.c_str(),                         this->DirWork.c_str());      cc.SetEscapeOldStyle(false);      cc.SetEscapeAllowMakeVars(true);      this->Target->Target->AddPreBuildCommand(cc);    } else {      // Add link library target dependencies to the autogen target      // dependencies      {        // add_dependencies/addUtility do not support generator expressions.        // We depend only on the libraries found in all configs therefore.        std::map<cmGeneratorTarget const*, std::size_t> commonTargets;        for (std::string const& config : this->ConfigsList) {          cmLinkImplementationLibraries const* libs =            this->Target->GetLinkImplementationLibraries(config);          if (libs != nullptr) {            for (cmLinkItem const& item : libs->Libraries) {              cmGeneratorTarget const* libTarget = item.Target;              if ((libTarget != nullptr) &&                  !StaticLibraryCycle(this->Target, libTarget, config)) {                // Increment target config count                commonTargets[libTarget]++;              }            }          }        }        for (auto const& item : commonTargets) {          if (item.second == this->ConfigsList.size()) {            autogenDependTargets.insert(item.first->Target);          }        }      }      // Create autogen target      cmTarget* autogenTarget = makefile->AddUtilityCommand(        this->AutogenTargetName, cmMakefile::TargetOrigin::Generator, true,        this->DirWork.c_str(), /*byproducts=*/autogenProvides,        std::vector<std::string>(autogenDependFiles.begin(),                                 autogenDependFiles.end()),        commandLines, false, autogenComment.c_str());      // Create autogen generator target      localGen->AddGeneratorTarget(        new cmGeneratorTarget(autogenTarget, localGen));      // Forward origin utilities to autogen target      for (std::string const& depName : this->Target->Target->GetUtilities()) {        autogenTarget->AddUtility(depName, makefile);      }      // Add additional autogen target dependencies to autogen target      for (cmTarget* depTarget : autogenDependTargets) {        autogenTarget->AddUtility(depTarget->GetName(), makefile);      }      // Set FOLDER property in autogen target      if (!this->AutogenFolder.empty()) {        autogenTarget->SetProperty("FOLDER", this->AutogenFolder.c_str());      }      // Add autogen target to the origin target dependencies      this->Target->Target->AddUtility(this->AutogenTargetName, makefile);    }  }}void cmQtAutoGenInitializer::SetupCustomTargets(){  cmMakefile* makefile = this->Target->Target->GetMakefile();  // forget the variables added here afterwards again:  cmMakefile::ScopePushPop varScope(makefile);  static_cast<void>(varScope);  // Configuration suffixes  std::map<std::string, std::string> configSuffixes;  for (std::string const& cfg : this->ConfigsList) {    std::string& suffix = configSuffixes[cfg];    suffix = "_";    suffix += cfg;  }  // Basic setup  AddDefinitionEscaped(makefile, "_multi_config",                       MultiConfigName(this->MultiConfig));  AddDefinitionEscaped(makefile, "_build_dir", this->DirBuild);  {    std::string parallel = GetSafeProperty(this->Target, "AUTOGEN_PARALLEL");    // Autodetect number of CPUs    if (parallel.empty() || (parallel == "AUTO")) {      parallel = std::to_string(GetParallelCPUCount());    }    AddDefinitionEscaped(makefile, "_parallel", parallel);  }  if (this->MocEnabled || this->UicEnabled) {    AddDefinitionEscaped(makefile, "_qt_version_major", this->QtVersionMajor);    AddDefinitionEscaped(makefile, "_settings_file",                         this->AutogenSettingsFile);    AddDefinitionEscaped(makefile, "_sources", this->Sources);    AddDefinitionEscaped(makefile, "_headers", this->Headers);    if (this->MocEnabled) {      this->SetupCustomTargetsMoc();    }    if (this->UicEnabled) {      this->SetupCustomTargetsUic();    }  }  if (this->RccEnabled) {    AddDefinitionEscaped(makefile, "_qt_rcc_executable", this->RccExecutable);    AddDefinitionEscaped(makefile, "_qt_rcc_list_options",                         this->RccListOptions);  }  // Create info directory on demand  if (!cmSystemTools::MakeDirectory(this->DirInfo)) {    std::string emsg = ("Could not create directory: ");    emsg += Quoted(this->DirInfo);    cmSystemTools::Error(emsg.c_str());  }  auto ReOpenInfoFile = [](cmsys::ofstream& ofs,                           std::string const& fileName) -> bool {    // Ensure we have write permission    mode_t perm = 0;#if defined(_WIN32) && !defined(__CYGWIN__)    mode_t mode_write = S_IWRITE;#else    mode_t mode_write = S_IWUSR;#endif    cmSystemTools::GetPermissions(fileName, perm);    if (!(perm & mode_write)) {      cmSystemTools::SetPermissions(fileName, perm | mode_write);    }    ofs.open(fileName.c_str(), std::ios::app);    if (!ofs) {      // File open error      std::string error = "Internal CMake error when trying to open file: ";      error += Quoted(fileName);      error += " for writing.";      cmSystemTools::Error(error.c_str());    }    return static_cast<bool>(ofs);  };  // Generate autogen target info file  if (this->MocEnabled || this->UicEnabled) {    {      std::string infoFileIn = cmSystemTools::GetCMakeRoot();      infoFileIn += "/Modules/AutogenInfo.cmake.in";      makefile->ConfigureFile(        infoFileIn.c_str(), this->AutogenInfoFile.c_str(), false, true, false);    }    // Append custom definitions to info file    // --------------------------------------    cmsys::ofstream ofs;    if (ReOpenInfoFile(ofs, this->AutogenInfoFile)) {      auto OfsWriteMap = [&ofs](        const char* key, std::map<std::string, std::string> const& map) {        for (auto const& item : map) {          ofs << "set(" << key << "_" << item.first << " "              << cmOutputConverter::EscapeForCMake(item.second) << ")\n";        }      };      ofs << "# Configurations options\n";      OfsWriteMap("AM_CONFIG_SUFFIX", configSuffixes);      OfsWriteMap("AM_MOC_DEFINITIONS", this->ConfigMocDefines);      OfsWriteMap("AM_MOC_INCLUDES", this->ConfigMocIncludes);      OfsWriteMap("AM_UIC_TARGET_OPTIONS", this->ConfigUicOptions);      // Settings files (only require for multi configuration generators)      if (this->MultiConfig != MultiConfigT::SINGLE) {        std::map<std::string, std::string> settingsFiles;        for (std::string const& cfg : this->ConfigsList) {          settingsFiles[cfg] =            AppendFilenameSuffix(this->AutogenSettingsFile, "_" + cfg);        }        OfsWriteMap("AM_SETTINGS_FILE", settingsFiles);      }    }  }  // Generate auto RCC info files  if (this->RccEnabled) {    std::string infoFileIn = cmSystemTools::GetCMakeRoot();    infoFileIn += "/Modules/AutoRccInfo.cmake.in";    for (Qrc const& qrc : this->Qrcs) {      // Configure info file      makefile->ConfigureFile(infoFileIn.c_str(), qrc.InfoFile.c_str(), false,                              true, false);      // Append custom definitions to info file      // --------------------------------------      cmsys::ofstream ofs;      if (ReOpenInfoFile(ofs, qrc.InfoFile)) {        {          ofs << "# Job\n";          auto OfsWrite = [&ofs](const char* key, std::string const& value) {            ofs << "set(" << key << " "                << cmOutputConverter::EscapeForCMake(value) << ")\n";          };          OfsWrite("ARCC_SETTINGS_FILE", qrc.SettingsFile);          OfsWrite("ARCC_SOURCE", qrc.QrcFile);          OfsWrite("ARCC_OUTPUT", qrc.RccFile);          OfsWrite("ARCC_OPTIONS", cmJoin(qrc.Options, ";"));          OfsWrite("ARCC_INPUTS", cmJoin(qrc.Resources, ";"));        }        {          ofs << "# Configurations options\n";          auto OfsWriteMap = [&ofs](            const char* key, std::map<std::string, std::string> const& map) {            for (auto const& item : map) {              ofs << "set(" << key << "_" << item.first << " "                  << cmOutputConverter::EscapeForCMake(item.second) << ")\n";            }          };          OfsWriteMap("ARCC_CONFIG_SUFFIX", configSuffixes);          // Settings files (only require for multi configuration generators)          if (this->MultiConfig != MultiConfigT::SINGLE) {            std::map<std::string, std::string> settingsFiles;            for (std::string const& cfg : this->ConfigsList) {              settingsFiles[cfg] =                AppendFilenameSuffix(qrc.SettingsFile, "_" + cfg);            }            OfsWriteMap("ARCC_SETTINGS_FILE", settingsFiles);          }        }      } else {        break;      }    }  }}void cmQtAutoGenInitializer::SetupCustomTargetsMoc(){  cmLocalGenerator* localGen = this->Target->GetLocalGenerator();  cmMakefile* makefile = this->Target->Target->GetMakefile();  AddDefinitionEscaped(makefile, "_moc_skip", this->MocSkip);  AddDefinitionEscaped(makefile, "_moc_options",                       GetSafeProperty(this->Target, "AUTOMOC_MOC_OPTIONS"));  AddDefinitionEscaped(makefile, "_moc_relaxed_mode",                       makefile->IsOn("CMAKE_AUTOMOC_RELAXED_MODE") ? "TRUE"                                                                    : "FALSE");  AddDefinitionEscaped(makefile, "_moc_macro_names",                       GetSafeProperty(this->Target, "AUTOMOC_MACRO_NAMES"));  AddDefinitionEscaped(    makefile, "_moc_depend_filters",    GetSafeProperty(this->Target, "AUTOMOC_DEPEND_FILTERS"));  // Compiler predefines  if (this->Target->GetPropertyAsBool("AUTOMOC_COMPILER_PREDEFINES") &&      this->QtVersionGreaterOrEqual(5, 8)) {    AddDefinitionEscaped(      makefile, "_moc_predefs_cmd",      makefile->GetSafeDefinition("CMAKE_CXX_COMPILER_PREDEFINES_COMMAND"));  }  // Moc includes and compile definitions  {    auto GetIncludeDirs = [this,                           localGen](std::string const& cfg) -> std::string {      // Get the include dirs for this target, without stripping the implicit      // include dirs off, see      // https://gitlab.kitware.com/cmake/cmake/issues/13667      std::vector<std::string> includeDirs;      localGen->GetIncludeDirectories(includeDirs, this->Target, "CXX", cfg,                                      false);      return cmJoin(includeDirs, ";");    };    auto GetCompileDefinitions =      [this, localGen](std::string const& cfg) -> std::string {      std::set<std::string> defines;      localGen->AddCompileDefinitions(defines, this->Target, cfg, "CXX");      return cmJoin(defines, ";");    };    // Default configuration settings    std::string const includeDirs = GetIncludeDirs(this->ConfigDefault);    std::string const compileDefs = GetCompileDefinitions(this->ConfigDefault);    // Other configuration settings    for (std::string const& cfg : this->ConfigsList) {      {        std::string const configIncludeDirs = GetIncludeDirs(cfg);        if (configIncludeDirs != includeDirs) {          this->ConfigMocIncludes[cfg] = configIncludeDirs;        }      }      {        std::string const configCompileDefs = GetCompileDefinitions(cfg);        if (configCompileDefs != compileDefs) {          this->ConfigMocDefines[cfg] = configCompileDefs;        }      }    }    AddDefinitionEscaped(makefile, "_moc_include_dirs", includeDirs);    AddDefinitionEscaped(makefile, "_moc_compile_defs", compileDefs);  }  // Moc executable  {    std::string mocExec;    std::string err;    if (this->QtVersionMajor == "5") {      cmGeneratorTarget* tgt = localGen->FindGeneratorTargetToUse("Qt5::moc");      if (tgt != nullptr) {        mocExec = SafeString(tgt->ImportedGetLocation(""));      } else {        err = "AUTOMOC: Qt5::moc target not found";      }    } else if (this->QtVersionMajor == "4") {      cmGeneratorTarget* tgt = localGen->FindGeneratorTargetToUse("Qt4::moc");      if (tgt != nullptr) {        mocExec = SafeString(tgt->ImportedGetLocation(""));      } else {        err = "AUTOMOC: Qt4::moc target not found";      }    } else {      err = "The AUTOMOC feature supports only Qt 4 and Qt 5";    }    if (err.empty()) {      AddDefinitionEscaped(makefile, "_qt_moc_executable", mocExec);    } else {      err += " (";      err += this->Target->GetName();      err += ")";      cmSystemTools::Error(err.c_str());    }  }}void cmQtAutoGenInitializer::SetupCustomTargetsUic(){  cmMakefile* makefile = this->Target->Target->GetMakefile();  // Uic search paths  {    std::vector<std::string> uicSearchPaths;    {      std::string const usp =        GetSafeProperty(this->Target, "AUTOUIC_SEARCH_PATHS");      if (!usp.empty()) {        cmSystemTools::ExpandListArgument(usp, uicSearchPaths);        std::string const srcDir = makefile->GetCurrentSourceDirectory();        for (std::string& path : uicSearchPaths) {          path = cmSystemTools::CollapseFullPath(path, srcDir);        }      }    }    AddDefinitionEscaped(makefile, "_uic_search_paths", uicSearchPaths);  }  // Uic target options  {    auto UicGetOpts = [this](std::string const& cfg) -> std::string {      std::vector<std::string> opts;      this->Target->GetAutoUicOptions(opts, cfg);      return cmJoin(opts, ";");    };    // Default settings    std::string const uicOpts = UicGetOpts(this->ConfigDefault);    AddDefinitionEscaped(makefile, "_uic_target_options", uicOpts);    // Configuration specific settings    for (std::string const& cfg : this->ConfigsList) {      std::string const configUicOpts = UicGetOpts(cfg);      if (configUicOpts != uicOpts) {        this->ConfigUicOptions[cfg] = configUicOpts;      }    }  }  // .ui files skip and options  {    std::vector<std::string> uiFileFiles;    std::vector<std::vector<std::string>> uiFileOptions;    {      std::string const uiExt = "ui";      std::string pathError;      for (cmSourceFile* sf : makefile->GetSourceFiles()) {        // sf->GetExtension() is only valid after sf->GetFullPath() ...        // Since we're iterating over source files that might be not in the        // target we need to check for path errors (not existing files).        std::string const& fPath = sf->GetFullPath(&pathError);        if (!pathError.empty()) {          pathError.clear();          continue;        }        if (sf->GetExtension() == uiExt) {          std::string const absFile = cmSystemTools::GetRealPath(fPath);          // Check if the .ui file should be skipped          if (sf->GetPropertyAsBool("SKIP_AUTOUIC") ||              sf->GetPropertyAsBool("SKIP_AUTOGEN")) {            this->UicSkip.insert(absFile);          }          // Check if the .ui file has uic options          std::string const uicOpts = GetSafeProperty(sf, "AUTOUIC_OPTIONS");          if (!uicOpts.empty()) {            // Check if file isn't skipped            if (this->UicSkip.count(absFile) == 0) {              uiFileFiles.push_back(absFile);              std::vector<std::string> optsVec;              cmSystemTools::ExpandListArgument(uicOpts, optsVec);              uiFileOptions.push_back(std::move(optsVec));            }          }        }      }    }    AddDefinitionEscaped(makefile, "_qt_uic_options_files", uiFileFiles);    AddDefinitionEscaped(makefile, "_qt_uic_options_options", uiFileOptions);  }  AddDefinitionEscaped(makefile, "_uic_skip", this->UicSkip);  // Uic executable  {    std::string err;    std::string uicExec;    cmLocalGenerator* localGen = this->Target->GetLocalGenerator();    if (this->QtVersionMajor == "5") {      cmGeneratorTarget* tgt = localGen->FindGeneratorTargetToUse("Qt5::uic");      if (tgt != nullptr) {        uicExec = SafeString(tgt->ImportedGetLocation(""));      } else {        // Project does not use Qt5Widgets, but has AUTOUIC ON anyway      }    } else if (this->QtVersionMajor == "4") {      cmGeneratorTarget* tgt = localGen->FindGeneratorTargetToUse("Qt4::uic");      if (tgt != nullptr) {        uicExec = SafeString(tgt->ImportedGetLocation(""));      } else {        err = "AUTOUIC: Qt4::uic target not found";      }    } else {      err = "The AUTOUIC feature supports only Qt 4 and Qt 5";    }    if (err.empty()) {      AddDefinitionEscaped(makefile, "_qt_uic_executable", uicExec);    } else {      err += " (";      err += this->Target->GetName();      err += ")";      cmSystemTools::Error(err.c_str());    }  }}std::vector<std::string> cmQtAutoGenInitializer::AddGeneratedSource(  std::string const& filename, GeneratorT genType){  std::vector<std::string> genFiles;  // Register source file in makefile and source group  if (this->MultiConfig != MultiConfigT::MULTI) {    genFiles.push_back(filename);  } else {    for (std::string const& cfg : this->ConfigsList) {      genFiles.push_back(AppendFilenameSuffix(filename, "_" + cfg));    }  }  {    cmMakefile* makefile = this->Target->Target->GetMakefile();    for (std::string const& genFile : genFiles) {      {        cmSourceFile* gFile = makefile->GetOrCreateSource(genFile, true);        gFile->SetProperty("GENERATED", "1");        gFile->SetProperty("SKIP_AUTOGEN", "On");      }      AddToSourceGroup(makefile, genFile, genType);    }  }  // Add source file to target  if (this->MultiConfig != MultiConfigT::MULTI) {    this->Target->AddSource(filename);  } else {    for (std::string const& cfg : this->ConfigsList) {      std::string src = "$<$<CONFIG:";      src += cfg;      src += ">:";      src += AppendFilenameSuffix(filename, "_" + cfg);      src += ">";      this->Target->AddSource(src);    }  }  return genFiles;}std::string cmQtAutoGenInitializer::GetQtMajorVersion(  cmGeneratorTarget const* target){  cmMakefile* makefile = target->Target->GetMakefile();  std::string qtMajor = makefile->GetSafeDefinition("QT_VERSION_MAJOR");  if (qtMajor.empty()) {    qtMajor = makefile->GetSafeDefinition("Qt5Core_VERSION_MAJOR");  }  const char* targetQtVersion =    target->GetLinkInterfaceDependentStringProperty("QT_MAJOR_VERSION", "");  if (targetQtVersion != nullptr) {    qtMajor = targetQtVersion;  }  return qtMajor;}std::string cmQtAutoGenInitializer::GetQtMinorVersion(  cmGeneratorTarget const* target, std::string const& qtVersionMajor){  cmMakefile* makefile = target->Target->GetMakefile();  std::string qtMinor;  if (qtVersionMajor == "5") {    qtMinor = makefile->GetSafeDefinition("Qt5Core_VERSION_MINOR");  }  if (qtMinor.empty()) {    qtMinor = makefile->GetSafeDefinition("QT_VERSION_MINOR");  }  const char* targetQtVersion =    target->GetLinkInterfaceDependentStringProperty("QT_MINOR_VERSION", "");  if (targetQtVersion != nullptr) {    qtMinor = targetQtVersion;  }  return qtMinor;}bool cmQtAutoGenInitializer::QtVersionGreaterOrEqual(  unsigned long requestMajor, unsigned long requestMinor) const{  unsigned long majorUL(0);  unsigned long minorUL(0);  if (cmSystemTools::StringToULong(this->QtVersionMajor.c_str(), &majorUL) &&      cmSystemTools::StringToULong(this->QtVersionMinor.c_str(), &minorUL)) {    return (majorUL > requestMajor) ||      (majorUL == requestMajor && minorUL >= requestMinor);  }  return false;}/// @brief Reads the resource files list from from a .qrc file/// @arg fileName Must be the absolute path of the .qrc file/// @return True if the rcc file was successfully readbool cmQtAutoGenInitializer::RccListInputs(std::string const& fileName,                                           std::vector<std::string>& files,                                           std::string& error){  if (!cmSystemTools::FileExists(fileName.c_str())) {    error = "rcc resource file does not exist:\n  ";    error += Quoted(fileName);    error += "\n";    return false;  }  if (!RccListOptions.empty()) {    // Use rcc for file listing    if (RccExecutable.empty()) {      error = "rcc executable not available";      return false;    }    // Run rcc list command in the directory of the qrc file with the pathless    // qrc file name argument. This way rcc prints relative paths.    // This avoids issues on Windows when the qrc file is in a path that    // contains non-ASCII characters.    std::string const fileDir = cmSystemTools::GetFilenamePath(fileName);    std::string const fileNameName = cmSystemTools::GetFilenameName(fileName);    bool result = false;    int retVal = 0;    std::string rccStdOut;    std::string rccStdErr;    {      std::vector<std::string> cmd;      cmd.push_back(RccExecutable);      cmd.insert(cmd.end(), RccListOptions.begin(), RccListOptions.end());      cmd.push_back(fileNameName);      result = cmSystemTools::RunSingleCommand(        cmd, &rccStdOut, &rccStdErr, &retVal, fileDir.c_str(),        cmSystemTools::OUTPUT_NONE, cmDuration::zero(), cmProcessOutput::Auto);    }    if (!result || retVal) {      error = "rcc list process failed for:\n  ";      error += Quoted(fileName);      error += "\n";      error += rccStdOut;      error += "\n";      error += rccStdErr;      error += "\n";      return false;    }    if (!RccListParseOutput(rccStdOut, rccStdErr, files, error)) {      return false;    }  } else {    // We can't use rcc for the file listing.    // Read the qrc file content into string and parse it.    {      std::string qrcContents;      {        cmsys::ifstream ifs(fileName.c_str());        if (ifs) {          std::ostringstream osst;          osst << ifs.rdbuf();          qrcContents = osst.str();        } else {          error = "rcc file not readable:\n  ";          error += Quoted(fileName);          error += "\n";          return false;        }      }      // Parse string content      RccListParseContent(qrcContents, files);    }  }  // Convert relative paths to absolute paths  RccListConvertFullPath(cmSystemTools::GetFilenamePath(fileName), files);  return true;}
 |