
If two imported targets in different directories have the same name we should still be able to propagate transitive usage requirements from both. Fix the DAG checker to work with target pointers instead of target names since the pointers will not be duplicated even if the names are. Fixes: #18345
97 lines
3.5 KiB
C++
97 lines
3.5 KiB
C++
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
#ifndef cmGeneratorExpressionDAGChecker_h
|
|
#define cmGeneratorExpressionDAGChecker_h
|
|
|
|
#include "cmConfigure.h" // IWYU pragma: keep
|
|
|
|
#include "cmListFileCache.h"
|
|
|
|
#include <map>
|
|
#include <set>
|
|
#include <string>
|
|
|
|
struct GeneratorExpressionContent;
|
|
struct cmGeneratorExpressionContext;
|
|
class cmGeneratorTarget;
|
|
|
|
#define CM_SELECT_BOTH(F, A1, A2) F(A1, A2)
|
|
#define CM_SELECT_FIRST(F, A1, A2) F(A1)
|
|
#define CM_SELECT_SECOND(F, A1, A2) F(A2)
|
|
|
|
#define CM_FOR_EACH_TRANSITIVE_PROPERTY_IMPL(F, SELECT) \
|
|
SELECT(F, EvaluatingIncludeDirectories, INCLUDE_DIRECTORIES) \
|
|
SELECT(F, EvaluatingSystemIncludeDirectories, SYSTEM_INCLUDE_DIRECTORIES) \
|
|
SELECT(F, EvaluatingCompileDefinitions, COMPILE_DEFINITIONS) \
|
|
SELECT(F, EvaluatingCompileOptions, COMPILE_OPTIONS) \
|
|
SELECT(F, EvaluatingAutoUicOptions, AUTOUIC_OPTIONS) \
|
|
SELECT(F, EvaluatingSources, SOURCES) \
|
|
SELECT(F, EvaluatingCompileFeatures, COMPILE_FEATURES) \
|
|
SELECT(F, EvaluatingLinkOptions, LINK_OPTIONS) \
|
|
SELECT(F, EvaluatingLinkDepends, LINK_DEPENDS)
|
|
|
|
#define CM_FOR_EACH_TRANSITIVE_PROPERTY(F) \
|
|
CM_FOR_EACH_TRANSITIVE_PROPERTY_IMPL(F, CM_SELECT_BOTH)
|
|
|
|
#define CM_FOR_EACH_TRANSITIVE_PROPERTY_METHOD(F) \
|
|
CM_FOR_EACH_TRANSITIVE_PROPERTY_IMPL(F, CM_SELECT_FIRST)
|
|
|
|
#define CM_FOR_EACH_TRANSITIVE_PROPERTY_NAME(F) \
|
|
CM_FOR_EACH_TRANSITIVE_PROPERTY_IMPL(F, CM_SELECT_SECOND)
|
|
|
|
struct cmGeneratorExpressionDAGChecker
|
|
{
|
|
cmGeneratorExpressionDAGChecker(const cmListFileBacktrace& backtrace,
|
|
cmGeneratorTarget const* target,
|
|
const std::string& property,
|
|
const GeneratorExpressionContent* content,
|
|
cmGeneratorExpressionDAGChecker* parent);
|
|
cmGeneratorExpressionDAGChecker(cmGeneratorTarget const* target,
|
|
const std::string& property,
|
|
const GeneratorExpressionContent* content,
|
|
cmGeneratorExpressionDAGChecker* parent);
|
|
|
|
enum Result
|
|
{
|
|
DAG,
|
|
SELF_REFERENCE,
|
|
CYCLIC_REFERENCE,
|
|
ALREADY_SEEN
|
|
};
|
|
|
|
Result Check() const;
|
|
|
|
void ReportError(cmGeneratorExpressionContext* context,
|
|
const std::string& expr);
|
|
|
|
bool EvaluatingGenexExpression();
|
|
bool EvaluatingLinkLibraries(cmGeneratorTarget const* tgt = nullptr);
|
|
|
|
#define DECLARE_TRANSITIVE_PROPERTY_METHOD(METHOD) bool METHOD() const;
|
|
|
|
CM_FOR_EACH_TRANSITIVE_PROPERTY_METHOD(DECLARE_TRANSITIVE_PROPERTY_METHOD)
|
|
|
|
#undef DECLARE_TRANSITIVE_PROPERTY_METHOD
|
|
|
|
bool GetTransitivePropertiesOnly();
|
|
void SetTransitivePropertiesOnly() { this->TransitivePropertiesOnly = true; }
|
|
|
|
cmGeneratorTarget const* TopTarget() const;
|
|
|
|
private:
|
|
Result CheckGraph() const;
|
|
void Initialize();
|
|
|
|
private:
|
|
const cmGeneratorExpressionDAGChecker* const Parent;
|
|
cmGeneratorTarget const* Target;
|
|
const std::string Property;
|
|
std::map<cmGeneratorTarget const*, std::set<std::string>> Seen;
|
|
const GeneratorExpressionContent* const Content;
|
|
const cmListFileBacktrace Backtrace;
|
|
Result CheckResult;
|
|
bool TransitivePropertiesOnly;
|
|
};
|
|
|
|
#endif
|