b80557c7bd
The evaluation files must be known before cmTargetTraceDependencies attempts to find them, but we must actually generate the files after cmTargetTraceDependencies, as that can add to target SOURCES. The limitation is that the generated output name must not depend on the SOURCES of a target if the generated file is used by that target. Mark the output files as GENERATED so that trace dependencies does not expect them to already exist in the filesystem. Move the invokation of ForceLinkerLanguage in the Generate logic to after the generated file names are known. ForceLinkerLanguage tries to determine the sources of a target (in order to determine an already-known language) and otherwise fails to get information about the generated file. Test that the output of file(GENERATE) can be used as a target source file and that accessing the target SOURCES in the name of the output file is an error. Accessing the TARGET_OBJECTS would be a similar error if it was legal to use that generator expression in this context. That is not currently possible and is a different error condition, so test the current error output as a reminder to change the expected output if that becomes possible in the future. Test that generated rule files resulting from cmTargetTraceDependencies appear in the SOURCES generated in the output file.
158 lines
4.5 KiB
C++
158 lines
4.5 KiB
C++
/*============================================================================
|
|
CMake - Cross Platform Makefile Generator
|
|
Copyright 2012 Stephen Kelly <steveire@gmail.com>
|
|
|
|
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 cmGeneratorExpressionEvaluator_h
|
|
#define cmGeneratorExpressionEvaluator_h
|
|
|
|
#include <vector>
|
|
#include <string>
|
|
|
|
#include "cmListFileCache.h"
|
|
|
|
class cmTarget;
|
|
|
|
//----------------------------------------------------------------------------
|
|
struct cmGeneratorExpressionContext
|
|
{
|
|
cmGeneratorExpressionContext()
|
|
: Backtrace(NULL)
|
|
{
|
|
}
|
|
|
|
cmListFileBacktrace Backtrace;
|
|
std::set<cmTarget*> DependTargets;
|
|
std::set<cmTarget const*> AllTargets;
|
|
std::set<std::string> SeenTargetProperties;
|
|
std::set<cmTarget const*> SourceSensitiveTargets;
|
|
std::map<cmTarget const*, std::map<std::string, std::string> >
|
|
MaxLanguageStandard;
|
|
cmMakefile *Makefile;
|
|
std::string Config;
|
|
cmTarget const* HeadTarget; // The target whose property is being evaluated.
|
|
cmTarget const* CurrentTarget; // The dependent of HeadTarget which appears
|
|
// directly or indirectly in the property.
|
|
bool Quiet;
|
|
bool HadError;
|
|
bool HadContextSensitiveCondition;
|
|
bool HadHeadSensitiveCondition;
|
|
bool EvaluateForBuildsystem;
|
|
};
|
|
|
|
struct cmGeneratorExpressionDAGChecker;
|
|
struct cmGeneratorExpressionNode;
|
|
|
|
//----------------------------------------------------------------------------
|
|
struct cmGeneratorExpressionEvaluator
|
|
{
|
|
cmGeneratorExpressionEvaluator() {}
|
|
virtual ~cmGeneratorExpressionEvaluator() {}
|
|
|
|
enum Type
|
|
{
|
|
Text,
|
|
Generator
|
|
};
|
|
|
|
virtual Type GetType() const = 0;
|
|
|
|
virtual std::string Evaluate(cmGeneratorExpressionContext *context,
|
|
cmGeneratorExpressionDAGChecker *) const = 0;
|
|
|
|
private:
|
|
cmGeneratorExpressionEvaluator(const cmGeneratorExpressionEvaluator &);
|
|
void operator=(const cmGeneratorExpressionEvaluator &);
|
|
};
|
|
|
|
struct TextContent : public cmGeneratorExpressionEvaluator
|
|
{
|
|
TextContent(const char *start, size_t length)
|
|
: Content(start), Length(length)
|
|
{
|
|
|
|
}
|
|
|
|
std::string Evaluate(cmGeneratorExpressionContext *,
|
|
cmGeneratorExpressionDAGChecker *) const
|
|
{
|
|
return std::string(this->Content, this->Length);
|
|
}
|
|
|
|
Type GetType() const
|
|
{
|
|
return cmGeneratorExpressionEvaluator::Text;
|
|
}
|
|
|
|
void Extend(size_t length)
|
|
{
|
|
this->Length += length;
|
|
}
|
|
|
|
size_t GetLength()
|
|
{
|
|
return this->Length;
|
|
}
|
|
|
|
private:
|
|
const char *Content;
|
|
size_t Length;
|
|
};
|
|
|
|
//----------------------------------------------------------------------------
|
|
struct GeneratorExpressionContent : public cmGeneratorExpressionEvaluator
|
|
{
|
|
GeneratorExpressionContent(const char *startContent, size_t length);
|
|
void SetIdentifier(std::vector<cmGeneratorExpressionEvaluator*> identifier)
|
|
{
|
|
this->IdentifierChildren = identifier;
|
|
}
|
|
|
|
void SetParameters(
|
|
std::vector<std::vector<cmGeneratorExpressionEvaluator*> > parameters)
|
|
{
|
|
this->ParamChildren = parameters;
|
|
}
|
|
|
|
Type GetType() const
|
|
{
|
|
return cmGeneratorExpressionEvaluator::Generator;
|
|
}
|
|
|
|
std::string Evaluate(cmGeneratorExpressionContext *context,
|
|
cmGeneratorExpressionDAGChecker *) const;
|
|
|
|
std::string GetOriginalExpression() const;
|
|
|
|
~GeneratorExpressionContent();
|
|
|
|
private:
|
|
std::string EvaluateParameters(const cmGeneratorExpressionNode *node,
|
|
const std::string &identifier,
|
|
cmGeneratorExpressionContext *context,
|
|
cmGeneratorExpressionDAGChecker *dagChecker,
|
|
std::vector<std::string> ¶meters) const;
|
|
|
|
std::string ProcessArbitraryContent(
|
|
const cmGeneratorExpressionNode *node,
|
|
const std::string &identifier,
|
|
cmGeneratorExpressionContext *context,
|
|
cmGeneratorExpressionDAGChecker *dagChecker,
|
|
std::vector<std::vector<cmGeneratorExpressionEvaluator*> >::const_iterator
|
|
pit) const;
|
|
|
|
private:
|
|
std::vector<cmGeneratorExpressionEvaluator*> IdentifierChildren;
|
|
std::vector<std::vector<cmGeneratorExpressionEvaluator*> > ParamChildren;
|
|
const char *StartContent;
|
|
size_t ContentLength;
|
|
};
|
|
|
|
#endif
|