Merge topic 'autogen-same-name'
61a607e8
Help: Document AUTORCC behavior for same .qrc name casee4f508e4
Tests/QtAutogen: Test same moc/qrc source names in different directories4e9b97d7
QtAutogen: Allow multiple qrc files with the same name41c9e14a
QtAutogen: Allow multiple moc files with the same name3c3b37b0
QtAutogen: Use std:: instead of ::std::0a5dd3c7
cmFilePathUuid: Add class to generate deterministic unique file names
This commit is contained in:
commit
33d4aff50d
|
@ -19,5 +19,10 @@ Additional command line options for rcc can be set via the
|
||||||
The global property :prop_gbl:`AUTOGEN_TARGETS_FOLDER` can be used to group
|
The global property :prop_gbl:`AUTOGEN_TARGETS_FOLDER` can be used to group
|
||||||
the autorcc targets together in an IDE, e.g. in MSVS.
|
the autorcc targets together in an IDE, e.g. in MSVS.
|
||||||
|
|
||||||
|
When there are multiple ``.qrc`` files with the same name, CMake will
|
||||||
|
generate unspecified unique names for ``rcc``. Therefore if
|
||||||
|
``Q_INIT_RESOURCE()`` or ``Q_CLEANUP_RESOURCE()`` need to be used the
|
||||||
|
``.qrc`` file name must be unique.
|
||||||
|
|
||||||
See the :manual:`cmake-qt(7)` manual for more information on using CMake
|
See the :manual:`cmake-qt(7)` manual for more information on using CMake
|
||||||
with Qt.
|
with Qt.
|
||||||
|
|
|
@ -238,6 +238,8 @@ set(SRCS
|
||||||
cmFileLockPool.h
|
cmFileLockPool.h
|
||||||
cmFileLockResult.cxx
|
cmFileLockResult.cxx
|
||||||
cmFileLockResult.h
|
cmFileLockResult.h
|
||||||
|
cmFilePathUuid.cxx
|
||||||
|
cmFilePathUuid.h
|
||||||
cmFileTimeComparison.cxx
|
cmFileTimeComparison.cxx
|
||||||
cmFileTimeComparison.h
|
cmFileTimeComparison.h
|
||||||
cmFortranLexer.cxx
|
cmFortranLexer.cxx
|
||||||
|
|
|
@ -0,0 +1,132 @@
|
||||||
|
/*============================================================================
|
||||||
|
CMake - Cross Platform Makefile Generator
|
||||||
|
Copyright 2016 Sebastian Holtermann (sebholt@xwmw.org)
|
||||||
|
|
||||||
|
Distributed under the OSI-approved BSD License (the "License");
|
||||||
|
see accompanying file Copyright.txt for details.
|
||||||
|
|
||||||
|
This software is distributed WITHOUT ANY WARRANTY; without even the
|
||||||
|
implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
See the License for more information.
|
||||||
|
============================================================================*/
|
||||||
|
|
||||||
|
#include "cmFilePathUuid.h"
|
||||||
|
|
||||||
|
#include "cmCryptoHash.h"
|
||||||
|
#include "cmMakefile.h"
|
||||||
|
#include "cmSystemTools.h"
|
||||||
|
#include "cmsys/Base64.h"
|
||||||
|
|
||||||
|
cmFilePathUuid::cmFilePathUuid(cmMakefile* makefile)
|
||||||
|
{
|
||||||
|
initParentDirs(makefile->GetCurrentSourceDirectory(),
|
||||||
|
makefile->GetCurrentBinaryDirectory(),
|
||||||
|
makefile->GetHomeDirectory(),
|
||||||
|
makefile->GetHomeOutputDirectory());
|
||||||
|
}
|
||||||
|
|
||||||
|
cmFilePathUuid::cmFilePathUuid(const std::string& currentSrcDir,
|
||||||
|
const std::string& currentBinDir,
|
||||||
|
const std::string& projectSrcDir,
|
||||||
|
const std::string& projectBinDir)
|
||||||
|
{
|
||||||
|
initParentDirs(currentSrcDir, currentBinDir, projectSrcDir, projectBinDir);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmFilePathUuid::initParentDirs(const std::string& currentSrcDir,
|
||||||
|
const std::string& currentBinDir,
|
||||||
|
const std::string& projectSrcDir,
|
||||||
|
const std::string& projectBinDir)
|
||||||
|
{
|
||||||
|
parentDirs[0].first = cmsys::SystemTools::GetRealPath(currentSrcDir);
|
||||||
|
parentDirs[1].first = cmsys::SystemTools::GetRealPath(currentBinDir);
|
||||||
|
parentDirs[2].first = cmsys::SystemTools::GetRealPath(projectSrcDir);
|
||||||
|
parentDirs[3].first = cmsys::SystemTools::GetRealPath(projectBinDir);
|
||||||
|
|
||||||
|
parentDirs[0].second = "CurrentSource";
|
||||||
|
parentDirs[1].second = "CurrentBinary";
|
||||||
|
parentDirs[2].second = "ProjectSource";
|
||||||
|
parentDirs[3].second = "ProjectBinary";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string cmFilePathUuid::get(const std::string& filePath,
|
||||||
|
const char* outputPrefix,
|
||||||
|
const char* outputSuffix)
|
||||||
|
{
|
||||||
|
std::string sourceFilename = cmsys::SystemTools::GetFilenameName(filePath);
|
||||||
|
std::string sourceBasename =
|
||||||
|
cmsys::SystemTools::GetFilenameWithoutLastExtension(sourceFilename);
|
||||||
|
|
||||||
|
// Acquire checksum string
|
||||||
|
std::string checksum;
|
||||||
|
{
|
||||||
|
std::string sourceRelPath;
|
||||||
|
std::string sourceRelSeed;
|
||||||
|
GetRelPathSeed(filePath, sourceRelPath, sourceRelSeed);
|
||||||
|
checksum = GetChecksumString(sourceFilename, sourceRelPath, sourceRelSeed);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compose the file name
|
||||||
|
std::string uuid;
|
||||||
|
if (outputPrefix) {
|
||||||
|
uuid += outputPrefix;
|
||||||
|
}
|
||||||
|
uuid += sourceBasename.substr(0, partLengthName);
|
||||||
|
uuid += "_";
|
||||||
|
uuid += checksum.substr(0, partLengthCheckSum);
|
||||||
|
if (outputSuffix) {
|
||||||
|
uuid += outputSuffix;
|
||||||
|
}
|
||||||
|
return uuid;
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmFilePathUuid::GetRelPathSeed(const std::string& filePath,
|
||||||
|
std::string& sourceRelPath,
|
||||||
|
std::string& sourceRelSeed)
|
||||||
|
{
|
||||||
|
const std::string sourceNameReal = cmsys::SystemTools::GetRealPath(filePath);
|
||||||
|
std::string parentDirectory;
|
||||||
|
// Find closest project parent directory
|
||||||
|
for (size_t ii = 0; ii != numParentDirs; ++ii) {
|
||||||
|
const std::string& pDir = parentDirs[ii].first;
|
||||||
|
if (!pDir.empty() &&
|
||||||
|
cmsys::SystemTools::IsSubDirectory(sourceNameReal, pDir)) {
|
||||||
|
sourceRelSeed = parentDirs[ii].second;
|
||||||
|
parentDirectory = pDir;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Check if the file path is below a known project directory
|
||||||
|
if (parentDirectory.empty()) {
|
||||||
|
// Use file syste root as fallback parent directory
|
||||||
|
sourceRelSeed = "FileSystemRoot";
|
||||||
|
cmsys::SystemTools::SplitPathRootComponent(sourceNameReal,
|
||||||
|
&parentDirectory);
|
||||||
|
}
|
||||||
|
sourceRelPath = cmsys::SystemTools::RelativePath(
|
||||||
|
parentDirectory, cmsys::SystemTools::GetParentDirectory(sourceNameReal));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string cmFilePathUuid::GetChecksumString(
|
||||||
|
const std::string& sourceFilename, const std::string& sourceRelPath,
|
||||||
|
const std::string& sourceRelSeed)
|
||||||
|
{
|
||||||
|
std::string checksumBase64;
|
||||||
|
{
|
||||||
|
// Calculate the file ( seed + relative path + name ) checksum
|
||||||
|
std::vector<unsigned char> hashBytes =
|
||||||
|
cmCryptoHash::New("SHA256")->ByteHashString(
|
||||||
|
(sourceRelSeed + sourceRelPath + sourceFilename).c_str());
|
||||||
|
// Convert hash bytes to Base64 text string
|
||||||
|
std::vector<unsigned char> base64Bytes(hashBytes.size() * 2, 0);
|
||||||
|
cmsysBase64_Encode(&hashBytes[0], hashBytes.size(), &base64Bytes[0], 0);
|
||||||
|
checksumBase64 = reinterpret_cast<const char*>(&base64Bytes[0]);
|
||||||
|
}
|
||||||
|
// Base64 allows '/', '+' and '=' characters which are problematic
|
||||||
|
// when used in file names. Replace them with safer alternatives.
|
||||||
|
std::replace(checksumBase64.begin(), checksumBase64.end(), '/', '-');
|
||||||
|
std::replace(checksumBase64.begin(), checksumBase64.end(), '+', '_');
|
||||||
|
std::replace(checksumBase64.begin(), checksumBase64.end(), '=', '_');
|
||||||
|
|
||||||
|
return checksumBase64;
|
||||||
|
}
|
|
@ -0,0 +1,77 @@
|
||||||
|
/*============================================================================
|
||||||
|
CMake - Cross Platform Makefile Generator
|
||||||
|
Copyright 2016 Sebastian Holtermann (sebholt@xwmw.org)
|
||||||
|
|
||||||
|
Distributed under the OSI-approved BSD License (the "License");
|
||||||
|
see accompanying file Copyright.txt for details.
|
||||||
|
|
||||||
|
This software is distributed WITHOUT ANY WARRANTY; without even the
|
||||||
|
implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
See the License for more information.
|
||||||
|
============================================================================*/
|
||||||
|
|
||||||
|
#ifndef cmFilePathUuid_h
|
||||||
|
#define cmFilePathUuid_h
|
||||||
|
|
||||||
|
#include "cmStandardIncludes.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
class cmMakefile;
|
||||||
|
|
||||||
|
/** \class cmFilePathUuid
|
||||||
|
* @brief Generates a unique pathless file name with a checksum component
|
||||||
|
* calculated from the file path.
|
||||||
|
*
|
||||||
|
* The checksum is calculated from the relative file path to the
|
||||||
|
* closest known project directory. This guarantees reproducibility
|
||||||
|
* when source and build directory differ e.g. for different project
|
||||||
|
* build directories.
|
||||||
|
*/
|
||||||
|
class cmFilePathUuid
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// Maximum number of characters to use from the file name
|
||||||
|
static const size_t partLengthName = 14;
|
||||||
|
/// Maximum number of characters to use from the path checksum
|
||||||
|
static const size_t partLengthCheckSum = 14;
|
||||||
|
|
||||||
|
/// @brief Initilizes the parent directories from a makefile
|
||||||
|
cmFilePathUuid(cmMakefile* makefile);
|
||||||
|
|
||||||
|
/// @brief Initilizes the parent directories manually
|
||||||
|
cmFilePathUuid(const std::string& currentSrcDir,
|
||||||
|
const std::string& currentBinDir,
|
||||||
|
const std::string& projectSrcDir,
|
||||||
|
const std::string& projectBinDir);
|
||||||
|
|
||||||
|
/* @brief Calculates and returns the uuid for a file path
|
||||||
|
*
|
||||||
|
* @arg outputPrefix optional string to prepend to the result
|
||||||
|
* @arg outputSuffix optional string to append to the result
|
||||||
|
*/
|
||||||
|
std::string get(const std::string& filePath, const char* outputPrefix = NULL,
|
||||||
|
const char* outputSuffix = NULL);
|
||||||
|
|
||||||
|
private:
|
||||||
|
void initParentDirs(const std::string& currentSrcDir,
|
||||||
|
const std::string& currentBinDir,
|
||||||
|
const std::string& projectSrcDir,
|
||||||
|
const std::string& projectBinDir);
|
||||||
|
|
||||||
|
/// Returns the relative path and the parent directory key string (seed)
|
||||||
|
void GetRelPathSeed(const std::string& filePath, std::string& sourceRelPath,
|
||||||
|
std::string& sourceRelSeed);
|
||||||
|
|
||||||
|
std::string GetChecksumString(const std::string& sourceFilename,
|
||||||
|
const std::string& sourceRelPath,
|
||||||
|
const std::string& sourceRelSeed);
|
||||||
|
|
||||||
|
/// Size of the parent directory list
|
||||||
|
static const size_t numParentDirs = 4;
|
||||||
|
/// List of (directory name, seed name) pairs
|
||||||
|
std::pair<std::string, std::string> parentDirs[numParentDirs];
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
|
@ -13,6 +13,7 @@
|
||||||
|
|
||||||
#include "cmQtAutoGeneratorInitializer.h"
|
#include "cmQtAutoGeneratorInitializer.h"
|
||||||
|
|
||||||
|
#include "cmFilePathUuid.h"
|
||||||
#include "cmLocalGenerator.h"
|
#include "cmLocalGenerator.h"
|
||||||
#include "cmMakefile.h"
|
#include "cmMakefile.h"
|
||||||
#include "cmSourceFile.h"
|
#include "cmSourceFile.h"
|
||||||
|
@ -25,6 +26,34 @@
|
||||||
#include "cmGlobalVisualStudioGenerator.h"
|
#include "cmGlobalVisualStudioGenerator.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static std::string GetAutogenTargetName(cmGeneratorTarget const* target)
|
||||||
|
{
|
||||||
|
std::string autogenTargetName = target->GetName();
|
||||||
|
autogenTargetName += "_automoc";
|
||||||
|
return autogenTargetName;
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string GetAutogenTargetDir(cmGeneratorTarget const* target)
|
||||||
|
{
|
||||||
|
cmMakefile* makefile = target->Target->GetMakefile();
|
||||||
|
std::string targetDir = makefile->GetCurrentBinaryDirectory();
|
||||||
|
targetDir += makefile->GetCMakeInstance()->GetCMakeFilesDirectory();
|
||||||
|
targetDir += "/";
|
||||||
|
targetDir += GetAutogenTargetName(target);
|
||||||
|
targetDir += ".dir/";
|
||||||
|
return targetDir;
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::string GetAutogenTargetBuildDir(cmGeneratorTarget const* target)
|
||||||
|
{
|
||||||
|
cmMakefile* makefile = target->Target->GetMakefile();
|
||||||
|
std::string targetDir = makefile->GetCurrentBinaryDirectory();
|
||||||
|
targetDir += "/";
|
||||||
|
targetDir += GetAutogenTargetName(target);
|
||||||
|
targetDir += ".dir/";
|
||||||
|
return targetDir;
|
||||||
|
}
|
||||||
|
|
||||||
static void SetupSourceFiles(cmGeneratorTarget const* target,
|
static void SetupSourceFiles(cmGeneratorTarget const* target,
|
||||||
std::vector<std::string>& skipMoc,
|
std::vector<std::string>& skipMoc,
|
||||||
std::vector<std::string>& mocSources,
|
std::vector<std::string>& mocSources,
|
||||||
|
@ -38,6 +67,7 @@ static void SetupSourceFiles(cmGeneratorTarget const* target,
|
||||||
|
|
||||||
std::vector<std::string> newRccFiles;
|
std::vector<std::string> newRccFiles;
|
||||||
|
|
||||||
|
cmFilePathUuid fpathUuid(makefile);
|
||||||
for (std::vector<cmSourceFile*>::const_iterator fileIt = srcFiles.begin();
|
for (std::vector<cmSourceFile*>::const_iterator fileIt = srcFiles.begin();
|
||||||
fileIt != srcFiles.end(); ++fileIt) {
|
fileIt != srcFiles.end(); ++fileIt) {
|
||||||
cmSourceFile* sf = *fileIt;
|
cmSourceFile* sf = *fileIt;
|
||||||
|
@ -55,13 +85,12 @@ static void SetupSourceFiles(cmGeneratorTarget const* target,
|
||||||
if (target->GetPropertyAsBool("AUTORCC")) {
|
if (target->GetPropertyAsBool("AUTORCC")) {
|
||||||
if (ext == "qrc" &&
|
if (ext == "qrc" &&
|
||||||
!cmSystemTools::IsOn(sf->GetPropertyForUser("SKIP_AUTORCC"))) {
|
!cmSystemTools::IsOn(sf->GetPropertyForUser("SKIP_AUTORCC"))) {
|
||||||
std::string basename =
|
|
||||||
cmsys::SystemTools::GetFilenameWithoutLastExtension(absFile);
|
|
||||||
|
|
||||||
std::string rcc_output_dir = target->GetSupportDirectory();
|
std::string rcc_output_file = GetAutogenTargetBuildDir(target);
|
||||||
cmSystemTools::MakeDirectory(rcc_output_dir.c_str());
|
// Create output directory
|
||||||
std::string rcc_output_file = rcc_output_dir;
|
cmSystemTools::MakeDirectory(rcc_output_file.c_str());
|
||||||
rcc_output_file += "/qrc_" + basename + ".cpp";
|
rcc_output_file += fpathUuid.get(absFile, "qrc_", ".cpp");
|
||||||
|
|
||||||
makefile->AppendProperty("ADDITIONAL_MAKE_CLEAN_FILES",
|
makefile->AppendProperty("ADDITIONAL_MAKE_CLEAN_FILES",
|
||||||
rcc_output_file.c_str(), false);
|
rcc_output_file.c_str(), false);
|
||||||
makefile->GetOrCreateSource(rcc_output_file, true);
|
makefile->GetOrCreateSource(rcc_output_file, true);
|
||||||
|
@ -365,24 +394,6 @@ static void MergeRccOptions(std::vector<std::string>& opts,
|
||||||
opts.insert(opts.end(), extraOpts.begin(), extraOpts.end());
|
opts.insert(opts.end(), extraOpts.begin(), extraOpts.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string GetAutogenTargetName(cmGeneratorTarget const* target)
|
|
||||||
{
|
|
||||||
std::string autogenTargetName = target->GetName();
|
|
||||||
autogenTargetName += "_automoc";
|
|
||||||
return autogenTargetName;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string GetAutogenTargetDir(cmGeneratorTarget const* target)
|
|
||||||
{
|
|
||||||
cmMakefile* makefile = target->Target->GetMakefile();
|
|
||||||
std::string targetDir = makefile->GetCurrentBinaryDirectory();
|
|
||||||
targetDir += makefile->GetCMakeInstance()->GetCMakeFilesDirectory();
|
|
||||||
targetDir += "/";
|
|
||||||
targetDir += GetAutogenTargetName(target);
|
|
||||||
targetDir += ".dir/";
|
|
||||||
return targetDir;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void copyTargetProperty(cmTarget* destinationTarget,
|
static void copyTargetProperty(cmTarget* destinationTarget,
|
||||||
cmTarget* sourceTarget,
|
cmTarget* sourceTarget,
|
||||||
const std::string& propertyName)
|
const std::string& propertyName)
|
||||||
|
@ -737,6 +748,7 @@ void cmQtAutoGeneratorInitializer::InitializeAutogenTarget(
|
||||||
) {
|
) {
|
||||||
std::vector<cmSourceFile*> srcFiles;
|
std::vector<cmSourceFile*> srcFiles;
|
||||||
target->GetConfigCommonSourceFiles(srcFiles);
|
target->GetConfigCommonSourceFiles(srcFiles);
|
||||||
|
cmFilePathUuid fpathUuid(makefile);
|
||||||
for (std::vector<cmSourceFile*>::const_iterator fileIt = srcFiles.begin();
|
for (std::vector<cmSourceFile*>::const_iterator fileIt = srcFiles.begin();
|
||||||
fileIt != srcFiles.end(); ++fileIt) {
|
fileIt != srcFiles.end(); ++fileIt) {
|
||||||
cmSourceFile* sf = *fileIt;
|
cmSourceFile* sf = *fileIt;
|
||||||
|
@ -747,15 +759,13 @@ void cmQtAutoGeneratorInitializer::InitializeAutogenTarget(
|
||||||
if (target->GetPropertyAsBool("AUTORCC")) {
|
if (target->GetPropertyAsBool("AUTORCC")) {
|
||||||
if (ext == "qrc" &&
|
if (ext == "qrc" &&
|
||||||
!cmSystemTools::IsOn(sf->GetPropertyForUser("SKIP_AUTORCC"))) {
|
!cmSystemTools::IsOn(sf->GetPropertyForUser("SKIP_AUTORCC"))) {
|
||||||
std::string basename =
|
{
|
||||||
cmsys::SystemTools::GetFilenameWithoutLastExtension(absFile);
|
std::string rcc_output_file = GetAutogenTargetBuildDir(target);
|
||||||
|
// Create output directory
|
||||||
std::string rcc_output_dir = target->GetSupportDirectory();
|
cmSystemTools::MakeDirectory(rcc_output_file.c_str());
|
||||||
cmSystemTools::MakeDirectory(rcc_output_dir.c_str());
|
rcc_output_file += fpathUuid.get(absFile, "qrc_", ".cpp");
|
||||||
std::string rcc_output_file = rcc_output_dir;
|
rcc_output.push_back(rcc_output_file);
|
||||||
rcc_output_file += "/qrc_" + basename + ".cpp";
|
}
|
||||||
rcc_output.push_back(rcc_output_file);
|
|
||||||
|
|
||||||
if (!cmSystemTools::IsOn(sf->GetPropertyForUser("GENERATED"))) {
|
if (!cmSystemTools::IsOn(sf->GetPropertyForUser("GENERATED"))) {
|
||||||
if (qtMajorVersion == "5") {
|
if (qtMajorVersion == "5") {
|
||||||
ListQt5RccInputs(sf, target, depends);
|
ListQt5RccInputs(sf, target, depends);
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
#include "cmQtAutoGenerators.h"
|
#include "cmQtAutoGenerators.h"
|
||||||
|
|
||||||
#include "cmAlgorithms.h"
|
#include "cmAlgorithms.h"
|
||||||
|
#include "cmFilePathUuid.h"
|
||||||
#include "cmGlobalGenerator.h"
|
#include "cmGlobalGenerator.h"
|
||||||
#include "cmMakefile.h"
|
#include "cmMakefile.h"
|
||||||
#include "cmOutputConverter.h"
|
#include "cmOutputConverter.h"
|
||||||
|
@ -87,6 +88,23 @@ static std::string extractSubDir(const std::string& absPath,
|
||||||
return subDir;
|
return subDir;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool FileNameIsUnique(const std::string& filePath,
|
||||||
|
const std::map<std::string, std::string>& fileMap)
|
||||||
|
{
|
||||||
|
size_t count(0);
|
||||||
|
const std::string fileName = cmsys::SystemTools::GetFilenameName(filePath);
|
||||||
|
for (std::map<std::string, std::string>::const_iterator si = fileMap.begin();
|
||||||
|
si != fileMap.end(); ++si) {
|
||||||
|
if (cmsys::SystemTools::GetFilenameName(si->first) == fileName) {
|
||||||
|
++count;
|
||||||
|
if (count > 1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
cmQtAutoGenerators::cmQtAutoGenerators()
|
cmQtAutoGenerators::cmQtAutoGenerators()
|
||||||
: Verbose(cmsys::SystemTools::HasEnv("VERBOSE"))
|
: Verbose(cmsys::SystemTools::HasEnv("VERBOSE"))
|
||||||
, ColorOutput(true)
|
, ColorOutput(true)
|
||||||
|
@ -358,11 +376,13 @@ void cmQtAutoGenerators::WriteOldMocDefinitionsFile(
|
||||||
|
|
||||||
void cmQtAutoGenerators::Init()
|
void cmQtAutoGenerators::Init()
|
||||||
{
|
{
|
||||||
|
this->TargetBuildSubDir = this->TargetName;
|
||||||
|
this->TargetBuildSubDir += ".dir/";
|
||||||
|
|
||||||
this->OutMocCppFilenameRel = this->TargetName;
|
this->OutMocCppFilenameRel = this->TargetName;
|
||||||
this->OutMocCppFilenameRel += ".cpp";
|
this->OutMocCppFilenameRel += ".cpp";
|
||||||
|
|
||||||
this->OutMocCppFilename = this->Builddir;
|
this->OutMocCppFilenameAbs = this->Builddir + this->OutMocCppFilenameRel;
|
||||||
this->OutMocCppFilename += this->OutMocCppFilenameRel;
|
|
||||||
|
|
||||||
std::vector<std::string> cdefList;
|
std::vector<std::string> cdefList;
|
||||||
cmSystemTools::ExpandListArgument(this->MocCompileDefinitionsStr, cdefList);
|
cmSystemTools::ExpandListArgument(this->MocCompileDefinitionsStr, cdefList);
|
||||||
|
@ -439,7 +459,7 @@ static std::string ReadAll(const std::string& filename)
|
||||||
|
|
||||||
bool cmQtAutoGenerators::RunAutogen(cmMakefile* makefile)
|
bool cmQtAutoGenerators::RunAutogen(cmMakefile* makefile)
|
||||||
{
|
{
|
||||||
if (!cmsys::SystemTools::FileExists(this->OutMocCppFilename.c_str()) ||
|
if (!cmsys::SystemTools::FileExists(this->OutMocCppFilenameAbs.c_str()) ||
|
||||||
(this->OldCompileSettingsStr != this->CurrentCompileSettingsStr)) {
|
(this->OldCompileSettingsStr != this->CurrentCompileSettingsStr)) {
|
||||||
this->GenerateAll = true;
|
this->GenerateAll = true;
|
||||||
}
|
}
|
||||||
|
@ -933,6 +953,8 @@ void cmQtAutoGenerators::ParseHeaders(
|
||||||
std::map<std::string, std::string>& notIncludedMocs,
|
std::map<std::string, std::string>& notIncludedMocs,
|
||||||
std::map<std::string, std::vector<std::string> >& includedUis)
|
std::map<std::string, std::vector<std::string> >& includedUis)
|
||||||
{
|
{
|
||||||
|
cmFilePathUuid fpathUuid(this->Srcdir, this->Builddir,
|
||||||
|
this->ProjectSourceDir, this->ProjectBinaryDir);
|
||||||
for (std::set<std::string>::const_iterator hIt = absHeaders.begin();
|
for (std::set<std::string>::const_iterator hIt = absHeaders.begin();
|
||||||
hIt != absHeaders.end(); ++hIt) {
|
hIt != absHeaders.end(); ++hIt) {
|
||||||
const std::string& headerName = *hIt;
|
const std::string& headerName = *hIt;
|
||||||
|
@ -946,13 +968,10 @@ void cmQtAutoGenerators::ParseHeaders(
|
||||||
this->LogInfo(err.str());
|
this->LogInfo(err.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string basename =
|
|
||||||
cmsys::SystemTools::GetFilenameWithoutLastExtension(headerName);
|
|
||||||
|
|
||||||
const std::string currentMoc = "moc_" + basename + ".cpp";
|
|
||||||
std::string macroName;
|
std::string macroName;
|
||||||
if (requiresMocing(contents, macroName)) {
|
if (requiresMocing(contents, macroName)) {
|
||||||
notIncludedMocs[headerName] = currentMoc;
|
notIncludedMocs[headerName] =
|
||||||
|
this->TargetBuildSubDir + fpathUuid.get(headerName, "moc_", ".cpp");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this->ParseForUic(headerName, contents, includedUis);
|
this->ParseForUic(headerName, contents, includedUis);
|
||||||
|
@ -1029,7 +1048,7 @@ bool cmQtAutoGenerators::GenerateMocFiles(
|
||||||
// check if we even need to update _automoc.cpp
|
// check if we even need to update _automoc.cpp
|
||||||
if (!automocCppChanged) {
|
if (!automocCppChanged) {
|
||||||
// compare contents of the _automoc.cpp file
|
// compare contents of the _automoc.cpp file
|
||||||
const std::string oldContents = ReadAll(this->OutMocCppFilename);
|
const std::string oldContents = ReadAll(this->OutMocCppFilenameAbs);
|
||||||
if (oldContents == automocSource) {
|
if (oldContents == automocSource) {
|
||||||
// nothing changed: don't touch the _automoc.cpp file
|
// nothing changed: don't touch the _automoc.cpp file
|
||||||
if (this->Verbose) {
|
if (this->Verbose) {
|
||||||
|
@ -1052,7 +1071,7 @@ bool cmQtAutoGenerators::GenerateMocFiles(
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
cmsys::ofstream outfile;
|
cmsys::ofstream outfile;
|
||||||
outfile.open(this->OutMocCppFilename.c_str(), std::ios::trunc);
|
outfile.open(this->OutMocCppFilenameAbs.c_str(), std::ios::trunc);
|
||||||
outfile << automocSource;
|
outfile << automocSource;
|
||||||
outfile.close();
|
outfile.close();
|
||||||
}
|
}
|
||||||
|
@ -1183,7 +1202,7 @@ bool cmQtAutoGenerators::GenerateUi(const std::string& realName,
|
||||||
cmsys::SystemTools::MakeDirectory(this->Builddir.c_str());
|
cmsys::SystemTools::MakeDirectory(this->Builddir.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
const ::std::string uiBuildFile = this->Builddir + uiOutputFile;
|
const std::string uiBuildFile = this->Builddir + uiOutputFile;
|
||||||
|
|
||||||
int sourceNewerThanUi = 0;
|
int sourceNewerThanUi = 0;
|
||||||
bool success = cmsys::SystemTools::FileTimeCompare(uiInputFile, uiBuildFile,
|
bool success = cmsys::SystemTools::FileTimeCompare(uiInputFile, uiBuildFile,
|
||||||
|
@ -1255,15 +1274,18 @@ bool cmQtAutoGenerators::GenerateQrcFiles()
|
||||||
{
|
{
|
||||||
// generate single map with input / output names
|
// generate single map with input / output names
|
||||||
std::map<std::string, std::string> qrcGenMap;
|
std::map<std::string, std::string> qrcGenMap;
|
||||||
for (std::vector<std::string>::const_iterator si = this->RccSources.begin();
|
{
|
||||||
si != this->RccSources.end(); ++si) {
|
cmFilePathUuid fpathUuid(this->Srcdir, this->Builddir,
|
||||||
const std::string ext = cmsys::SystemTools::GetFilenameLastExtension(*si);
|
this->ProjectSourceDir, this->ProjectBinaryDir);
|
||||||
if (ext == ".qrc") {
|
for (std::vector<std::string>::const_iterator si =
|
||||||
std::string basename =
|
this->RccSources.begin();
|
||||||
cmsys::SystemTools::GetFilenameWithoutLastExtension(*si);
|
si != this->RccSources.end(); ++si) {
|
||||||
std::string qrcOutputFile = "CMakeFiles/" + this->OriginTargetName +
|
const std::string ext =
|
||||||
".dir/qrc_" + basename + ".cpp";
|
cmsys::SystemTools::GetFilenameLastExtension(*si);
|
||||||
qrcGenMap[*si] = qrcOutputFile;
|
if (ext == ".qrc") {
|
||||||
|
qrcGenMap[*si] =
|
||||||
|
(this->TargetBuildSubDir + fpathUuid.get(*si, "qrc_", ".cpp"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1285,7 +1307,8 @@ bool cmQtAutoGenerators::GenerateQrcFiles()
|
||||||
for (std::map<std::string, std::string>::const_iterator si =
|
for (std::map<std::string, std::string>::const_iterator si =
|
||||||
qrcGenMap.begin();
|
qrcGenMap.begin();
|
||||||
si != qrcGenMap.end(); ++si) {
|
si != qrcGenMap.end(); ++si) {
|
||||||
if (!this->GenerateQrc(si->first, si->second)) {
|
bool unique = FileNameIsUnique(si->first, qrcGenMap);
|
||||||
|
if (!this->GenerateQrc(si->first, si->second, unique)) {
|
||||||
if (this->RunRccFailed) {
|
if (this->RunRccFailed) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -1295,11 +1318,24 @@ bool cmQtAutoGenerators::GenerateQrcFiles()
|
||||||
}
|
}
|
||||||
|
|
||||||
bool cmQtAutoGenerators::GenerateQrc(const std::string& qrcInputFile,
|
bool cmQtAutoGenerators::GenerateQrc(const std::string& qrcInputFile,
|
||||||
const std::string& qrcOutputFile)
|
const std::string& qrcOutputFile,
|
||||||
|
bool unique_n)
|
||||||
{
|
{
|
||||||
const std::string basename =
|
std::string symbolName;
|
||||||
cmsys::SystemTools::GetFilenameWithoutLastExtension(qrcInputFile);
|
if (unique_n) {
|
||||||
const ::std::string qrcBuildFile = this->Builddir + qrcOutputFile;
|
symbolName =
|
||||||
|
cmsys::SystemTools::GetFilenameWithoutLastExtension(qrcInputFile);
|
||||||
|
} else {
|
||||||
|
symbolName =
|
||||||
|
cmsys::SystemTools::GetFilenameWithoutLastExtension(qrcOutputFile);
|
||||||
|
// Remove "qrc_" at string begin
|
||||||
|
symbolName.erase(0, 4);
|
||||||
|
}
|
||||||
|
// Replace '-' with '_'. The former is valid for
|
||||||
|
// file names but not for symbol names.
|
||||||
|
std::replace(symbolName.begin(), symbolName.end(), '-', '_');
|
||||||
|
|
||||||
|
const std::string qrcBuildFile = this->Builddir + qrcOutputFile;
|
||||||
|
|
||||||
int sourceNewerThanQrc = 0;
|
int sourceNewerThanQrc = 0;
|
||||||
bool generateQrc = !cmsys::SystemTools::FileTimeCompare(
|
bool generateQrc = !cmsys::SystemTools::FileTimeCompare(
|
||||||
|
@ -1325,7 +1361,7 @@ bool cmQtAutoGenerators::GenerateQrc(const std::string& qrcInputFile,
|
||||||
}
|
}
|
||||||
|
|
||||||
command.push_back("-name");
|
command.push_back("-name");
|
||||||
command.push_back(basename);
|
command.push_back(symbolName);
|
||||||
command.push_back("-o");
|
command.push_back("-o");
|
||||||
command.push_back(qrcBuildFile);
|
command.push_back(qrcBuildFile);
|
||||||
command.push_back(qrcInputFile);
|
command.push_back(qrcInputFile);
|
||||||
|
|
|
@ -51,7 +51,8 @@ private:
|
||||||
const std::string& uiOutputFile);
|
const std::string& uiOutputFile);
|
||||||
bool GenerateQrcFiles();
|
bool GenerateQrcFiles();
|
||||||
bool GenerateQrc(const std::string& qrcInputFile,
|
bool GenerateQrc(const std::string& qrcInputFile,
|
||||||
const std::string& qrcOutputFile);
|
const std::string& qrcOutputFile, bool unique_n);
|
||||||
|
|
||||||
void ParseCppFile(
|
void ParseCppFile(
|
||||||
const std::string& absFilename,
|
const std::string& absFilename,
|
||||||
const std::vector<std::string>& headerExtensions,
|
const std::vector<std::string>& headerExtensions,
|
||||||
|
@ -123,8 +124,9 @@ private:
|
||||||
std::string CurrentCompileSettingsStr;
|
std::string CurrentCompileSettingsStr;
|
||||||
std::string OldCompileSettingsStr;
|
std::string OldCompileSettingsStr;
|
||||||
|
|
||||||
|
std::string TargetBuildSubDir;
|
||||||
std::string OutMocCppFilenameRel;
|
std::string OutMocCppFilenameRel;
|
||||||
std::string OutMocCppFilename;
|
std::string OutMocCppFilenameAbs;
|
||||||
std::list<std::string> MocIncludes;
|
std::list<std::string> MocIncludes;
|
||||||
std::list<std::string> MocDefinitions;
|
std::list<std::string> MocDefinitions;
|
||||||
std::vector<std::string> MocOptions;
|
std::vector<std::string> MocOptions;
|
||||||
|
|
|
@ -110,6 +110,10 @@ set_target_properties(
|
||||||
AUTOMOC TRUE
|
AUTOMOC TRUE
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Test AUTOMOC and AUTORCC on source files with the same name
|
||||||
|
# but in different subdirectories
|
||||||
|
add_subdirectory(sameName)
|
||||||
|
|
||||||
include(GenerateExportHeader)
|
include(GenerateExportHeader)
|
||||||
# The order is relevant here. B depends on A, and B headers depend on A
|
# The order is relevant here. B depends on A, and B headers depend on A
|
||||||
# headers both subdirectories use CMAKE_INCLUDE_CURRENT_DIR_IN_INTERFACE and we
|
# headers both subdirectories use CMAKE_INCLUDE_CURRENT_DIR_IN_INTERFACE and we
|
||||||
|
|
|
@ -0,0 +1,21 @@
|
||||||
|
# Test AUTOMOC and AUTORCC on source files with the same name
|
||||||
|
# but in different subdirectories
|
||||||
|
|
||||||
|
add_executable(sameName
|
||||||
|
aaa/bbb/item.cpp
|
||||||
|
aaa/bbb/data.qrc
|
||||||
|
aaa/item.cpp
|
||||||
|
aaa/data.qrc
|
||||||
|
bbb/aaa/item.cpp
|
||||||
|
bbb/aaa/data.qrc
|
||||||
|
bbb/item.cpp
|
||||||
|
bbb/data.qrc
|
||||||
|
ccc/item.cpp
|
||||||
|
ccc/data.qrc
|
||||||
|
item.cpp
|
||||||
|
data.qrc
|
||||||
|
main.cpp
|
||||||
|
)
|
||||||
|
target_include_directories(sameName PRIVATE ${CMAKE_CURRENT_BINARY_DIR})
|
||||||
|
target_link_libraries(sameName ${QT_LIBRARIES})
|
||||||
|
set_target_properties( sameName PROPERTIES AUTOMOC TRUE AUTORCC TRUE )
|
|
@ -0,0 +1,6 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource prefix="aaa/bbb">
|
||||||
|
<file>item.hpp</file>
|
||||||
|
<file>item.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,10 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
namespace aaa {
|
||||||
|
namespace bbb {
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
#ifndef AAA_BBB_ITEM_HPP
|
||||||
|
#define AAA_BBB_ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
namespace aaa {
|
||||||
|
namespace bbb {
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,6 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource prefix="aaa/">
|
||||||
|
<file>item.hpp</file>
|
||||||
|
<file>item.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,8 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
namespace aaa {
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
#ifndef AAA_ITEM_HPP
|
||||||
|
#define AAA_ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
namespace aaa {
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,6 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource prefix="bbb/aaa/">
|
||||||
|
<file>item.hpp</file>
|
||||||
|
<file>item.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,10 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
namespace bbb {
|
||||||
|
namespace aaa {
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
#ifndef BBB_AAA_ITEM_HPP
|
||||||
|
#define BBB_AAA_ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
namespace bbb {
|
||||||
|
namespace aaa {
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,6 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource prefix="bbb/">
|
||||||
|
<file>item.hpp</file>
|
||||||
|
<file>item.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,8 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
namespace bbb {
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
#ifndef BBB_ITEM_HPP
|
||||||
|
#define BBB_ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
namespace bbb {
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,6 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource prefix="ccc/">
|
||||||
|
<file>item.hpp</file>
|
||||||
|
<file>item.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,23 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
namespace ccc {
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
class MocTest : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT;
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
|
||||||
|
void MocTest::go()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Include own moc files
|
||||||
|
#include "item.moc"
|
||||||
|
#include "moc_item.cpp"
|
|
@ -0,0 +1,16 @@
|
||||||
|
#ifndef CCC_ITEM_HPP
|
||||||
|
#define CCC_ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
namespace ccc {
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,5 @@
|
||||||
|
<!DOCTYPE RCC><RCC version="1.0">
|
||||||
|
<qresource>
|
||||||
|
<file>main.cpp</file>
|
||||||
|
</qresource>
|
||||||
|
</RCC>
|
|
@ -0,0 +1,5 @@
|
||||||
|
#include "item.hpp"
|
||||||
|
|
||||||
|
void Item::go()
|
||||||
|
{
|
||||||
|
}
|
|
@ -0,0 +1,13 @@
|
||||||
|
#ifndef ITEM_HPP
|
||||||
|
#define ITEM_HPP
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
class Item : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_SLOT
|
||||||
|
void go();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,16 @@
|
||||||
|
#include "aaa/bbb/item.hpp"
|
||||||
|
#include "aaa/item.hpp"
|
||||||
|
#include "bbb/aaa/item.hpp"
|
||||||
|
#include "bbb/item.hpp"
|
||||||
|
#include "ccc/item.hpp"
|
||||||
|
|
||||||
|
int main(int argv, char** args)
|
||||||
|
{
|
||||||
|
// Object instances
|
||||||
|
::aaa::Item aaa_item;
|
||||||
|
::aaa::bbb::Item aaa_bbb_item;
|
||||||
|
::bbb::Item bbb_item;
|
||||||
|
::bbb::aaa::Item bbb_aaa_item;
|
||||||
|
::ccc::Item ccc_item;
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
Reference in New Issue