|
@@ -0,0 +1,568 @@
|
|
|
|
+/*============================================================================
|
|
|
|
+ CMake - Cross Platform Makefile Generator
|
|
|
|
+ Copyright 2012 Stephen Kelly <[email protected]>
|
|
|
|
+
|
|
|
|
+ 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 "cmMakefile.h"
|
|
|
|
+
|
|
|
|
+#include "cmGeneratorExpressionEvaluator.h"
|
|
|
|
+#include "cmGeneratorExpressionParser.h"
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static void reportError(cmGeneratorExpressionContext *context,
|
|
|
|
+ const std::string &expr, const std::string &result)
|
|
|
|
+{
|
|
|
|
+ context->HadError = true;
|
|
|
|
+ if (context->Quiet)
|
|
|
|
+ {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ cmOStringStream e;
|
|
|
|
+ e << "Error evaluating generator expression:\n"
|
|
|
|
+ << " " << expr << "\n"
|
|
|
|
+ << result;
|
|
|
|
+ context->Makefile->GetCMakeInstance()
|
|
|
|
+ ->IssueMessage(cmake::FATAL_ERROR, e.str().c_str(),
|
|
|
|
+ context->Backtrace);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+struct cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ virtual ~cmGeneratorExpressionNode() {}
|
|
|
|
+
|
|
|
|
+ virtual bool GeneratesContent() const { return true; }
|
|
|
|
+
|
|
|
|
+ virtual bool AcceptsSingleArbitraryContentParameter() const
|
|
|
|
+ { return false; }
|
|
|
|
+
|
|
|
|
+ virtual int NumExpectedParameters() const { return 1; }
|
|
|
|
+
|
|
|
|
+ virtual std::string Evaluate(const std::vector<std::string> ¶meters,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content
|
|
|
|
+ ) const = 0;
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const struct ZeroNode : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ ZeroNode() {}
|
|
|
|
+
|
|
|
|
+ virtual bool GeneratesContent() const { return false; }
|
|
|
|
+
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> &,
|
|
|
|
+ cmGeneratorExpressionContext *,
|
|
|
|
+ const GeneratorExpressionContent *) const
|
|
|
|
+ {
|
|
|
|
+ // Unreachable
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+} zeroNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const struct OneNode : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ OneNode() {}
|
|
|
|
+
|
|
|
|
+ virtual bool AcceptsSingleArbitraryContentParameter() const { return true; }
|
|
|
|
+
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> &,
|
|
|
|
+ cmGeneratorExpressionContext *,
|
|
|
|
+ const GeneratorExpressionContent *) const
|
|
|
|
+ {
|
|
|
|
+ // Unreachable
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+} oneNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+#define BOOLEAN_OP_NODE(OPNAME, OP, SUCCESS_VALUE, FAILURE_VALUE) \
|
|
|
|
+static const struct OP ## Node : public cmGeneratorExpressionNode \
|
|
|
|
+{ \
|
|
|
|
+ OP ## Node () {} \
|
|
|
|
+/* We let -1 carry the meaning 'at least one' */ \
|
|
|
|
+ virtual int NumExpectedParameters() const { return -1; } \
|
|
|
|
+ \
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> ¶meters, \
|
|
|
|
+ cmGeneratorExpressionContext *context, \
|
|
|
|
+ const GeneratorExpressionContent *content) const \
|
|
|
|
+ { \
|
|
|
|
+ std::vector<std::string>::const_iterator it = parameters.begin(); \
|
|
|
|
+ const std::vector<std::string>::const_iterator end = parameters.end(); \
|
|
|
|
+ for ( ; it != end; ++it) \
|
|
|
|
+ { \
|
|
|
|
+ if (*it == #FAILURE_VALUE) \
|
|
|
|
+ { \
|
|
|
|
+ return #FAILURE_VALUE; \
|
|
|
|
+ } \
|
|
|
|
+ else if (*it != #SUCCESS_VALUE) \
|
|
|
|
+ { \
|
|
|
|
+ reportError(context, content->GetOriginalExpression(), \
|
|
|
|
+ "Parameters to $<" #OP "> must resolve to either '0' or '1'."); \
|
|
|
|
+ return std::string(); \
|
|
|
|
+ } \
|
|
|
|
+ } \
|
|
|
|
+ return #SUCCESS_VALUE; \
|
|
|
|
+ } \
|
|
|
|
+} OPNAME;
|
|
|
|
+
|
|
|
|
+BOOLEAN_OP_NODE(andNode, AND, 1, 0)
|
|
|
|
+BOOLEAN_OP_NODE(orNode, OR, 0, 1)
|
|
|
|
+
|
|
|
|
+#undef BOOLEAN_OP_NODE
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const struct NotNode : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ NotNode() {}
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> ¶meters,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content) const
|
|
|
|
+ {
|
|
|
|
+ if (*parameters.begin() != "0" && *parameters.begin() != "1")
|
|
|
|
+ {
|
|
|
|
+ reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "$<NOT> parameter must resolve to exactly one '0' or '1' value.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ return *parameters.begin() == "0" ? "1" : "0";
|
|
|
|
+ }
|
|
|
|
+} notNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const struct ConfigurationNode : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ ConfigurationNode() {}
|
|
|
|
+ virtual int NumExpectedParameters() const { return 0; }
|
|
|
|
+
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> &,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *) const
|
|
|
|
+ {
|
|
|
|
+ return context->Config ? context->Config : "";
|
|
|
|
+ }
|
|
|
|
+} configurationNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const struct ConfigurationTestNode : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ ConfigurationTestNode() {}
|
|
|
|
+
|
|
|
|
+ virtual int NumExpectedParameters() const { return 1; }
|
|
|
|
+
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> ¶meters,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content) const
|
|
|
|
+ {
|
|
|
|
+ if (!context->Config)
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ cmsys::RegularExpression configValidator;
|
|
|
|
+ configValidator.compile("^[A-Za-z0-9_]*$");
|
|
|
|
+ if (!configValidator.find(parameters.begin()->c_str()))
|
|
|
|
+ {
|
|
|
|
+ reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "Expression syntax not recognized.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ return *parameters.begin() == context->Config ? "1" : "0";
|
|
|
|
+ }
|
|
|
|
+} configurationTestNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<bool linker, bool soname>
|
|
|
|
+struct TargetFilesystemArtifactResultCreator
|
|
|
|
+{
|
|
|
|
+ static std::string Create(cmTarget* target,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content);
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultCreator<false, true>
|
|
|
|
+{
|
|
|
|
+ static std::string Create(cmTarget* target,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content)
|
|
|
|
+ {
|
|
|
|
+ // The target soname file (.so.1).
|
|
|
|
+ if(target->IsDLLPlatform())
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "TARGET_SONAME_FILE is not allowed "
|
|
|
|
+ "for DLL target platforms.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ if(target->GetType() != cmTarget::SHARED_LIBRARY)
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "TARGET_SONAME_FILE is allowed only for "
|
|
|
|
+ "SHARED libraries.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ std::string result = target->GetDirectory(context->Config);
|
|
|
|
+ result += "/";
|
|
|
|
+ result += target->GetSOName(context->Config);
|
|
|
|
+ return result;
|
|
|
|
+ }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultCreator<true, false>
|
|
|
|
+{
|
|
|
|
+ static std::string Create(cmTarget* target,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content)
|
|
|
|
+ {
|
|
|
|
+ // The file used to link to the target (.so, .lib, .a).
|
|
|
|
+ if(!target->IsLinkable())
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "TARGET_LINKER_FILE is allowed only for libraries and "
|
|
|
|
+ "executables with ENABLE_EXPORTS.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ return target->GetFullPath(context->Config,
|
|
|
|
+ target->HasImportLibrary());
|
|
|
|
+ }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultCreator<false, false>
|
|
|
|
+{
|
|
|
|
+ static std::string Create(cmTarget* target,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *)
|
|
|
|
+ {
|
|
|
|
+ return target->GetFullPath(context->Config, false, true);
|
|
|
|
+ }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<bool dirQual, bool nameQual>
|
|
|
|
+struct TargetFilesystemArtifactResultGetter
|
|
|
|
+{
|
|
|
|
+ static std::string Get(const std::string &result);
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultGetter<false, true>
|
|
|
|
+{
|
|
|
|
+ static std::string Get(const std::string &result)
|
|
|
|
+ { return cmSystemTools::GetFilenameName(result); }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultGetter<true, false>
|
|
|
|
+{
|
|
|
|
+ static std::string Get(const std::string &result)
|
|
|
|
+ { return cmSystemTools::GetFilenamePath(result); }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<>
|
|
|
|
+struct TargetFilesystemArtifactResultGetter<false, false>
|
|
|
|
+{
|
|
|
|
+ static std::string Get(const std::string &result)
|
|
|
|
+ { return result; }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+template<bool linker, bool soname, bool dirQual, bool nameQual>
|
|
|
|
+struct TargetFilesystemArtifact : public cmGeneratorExpressionNode
|
|
|
|
+{
|
|
|
|
+ TargetFilesystemArtifact() {}
|
|
|
|
+
|
|
|
|
+ virtual int NumExpectedParameters() const { return 1; }
|
|
|
|
+
|
|
|
|
+ std::string Evaluate(const std::vector<std::string> ¶meters,
|
|
|
|
+ cmGeneratorExpressionContext *context,
|
|
|
|
+ const GeneratorExpressionContent *content) const
|
|
|
|
+ {
|
|
|
|
+ // Lookup the referenced target.
|
|
|
|
+ std::string name = *parameters.begin();
|
|
|
|
+
|
|
|
|
+ cmsys::RegularExpression targetValidator;
|
|
|
|
+ targetValidator.compile("^[A-Za-z0-9_]+$");
|
|
|
|
+ if (!targetValidator.find(name.c_str()))
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "Expression syntax not recognized.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ cmTarget* target = context->Makefile->FindTargetToUse(name.c_str());
|
|
|
|
+ if(!target)
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "No target \"" + name + "\"");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ if(target->GetType() >= cmTarget::UTILITY &&
|
|
|
|
+ target->GetType() != cmTarget::UNKNOWN_LIBRARY)
|
|
|
|
+ {
|
|
|
|
+ ::reportError(context, content->GetOriginalExpression(),
|
|
|
|
+ "Target \"" + name + "\" is not an executable or library.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ context->Targets.insert(target);
|
|
|
|
+
|
|
|
|
+ std::string result =
|
|
|
|
+ TargetFilesystemArtifactResultCreator<linker, soname>::Create(
|
|
|
|
+ target,
|
|
|
|
+ context,
|
|
|
|
+ content);
|
|
|
|
+ if (context->HadError)
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ return
|
|
|
|
+ TargetFilesystemArtifactResultGetter<dirQual, nameQual>::Get(result);
|
|
|
|
+ }
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, false, false, false> targetFileNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<true, false, false, false> targetLinkerFileNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, true, false, false> targetSoNameFileNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, false, false, true> targetFileNameNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<true, false, false, true> targetLinkerFileNameNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, true, false, true> targetSoNameFileNameNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, false, true, false> targetFileDirNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<true, false, true, false> targetLinkerFileDirNode;
|
|
|
|
+static const
|
|
|
|
+TargetFilesystemArtifact<false, true, true, false> targetSoNameFileDirNode;
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static const
|
|
|
|
+cmGeneratorExpressionNode* GetNode(const std::string &identifier)
|
|
|
|
+{
|
|
|
|
+ if (identifier == "0")
|
|
|
|
+ return &zeroNode;
|
|
|
|
+ if (identifier == "1")
|
|
|
|
+ return &oneNode;
|
|
|
|
+ if (identifier == "AND")
|
|
|
|
+ return &andNode;
|
|
|
|
+ if (identifier == "OR")
|
|
|
|
+ return &orNode;
|
|
|
|
+ if (identifier == "NOT")
|
|
|
|
+ return ¬Node;
|
|
|
|
+ else if (identifier == "CONFIGURATION")
|
|
|
|
+ return &configurationNode;
|
|
|
|
+ else if (identifier == "CONFIG")
|
|
|
|
+ return &configurationTestNode;
|
|
|
|
+ else if (identifier == "TARGET_FILE")
|
|
|
|
+ return &targetFileNode;
|
|
|
|
+ else if (identifier == "TARGET_LINKER_FILE")
|
|
|
|
+ return &targetLinkerFileNode;
|
|
|
|
+ else if (identifier == "TARGET_SONAME_FILE")
|
|
|
|
+ return &targetSoNameFileNode;
|
|
|
|
+ else if (identifier == "TARGET_FILE_NAME")
|
|
|
|
+ return &targetFileNameNode;
|
|
|
|
+ else if (identifier == "TARGET_LINKER_FILE_NAME")
|
|
|
|
+ return &targetLinkerFileNameNode;
|
|
|
|
+ else if (identifier == "TARGET_SONAME_FILE_NAME")
|
|
|
|
+ return &targetSoNameFileNameNode;
|
|
|
|
+ else if (identifier == "TARGET_FILE_DIR")
|
|
|
|
+ return &targetFileDirNode;
|
|
|
|
+ else if (identifier == "TARGET_LINKER_FILE_DIR")
|
|
|
|
+ return &targetLinkerFileDirNode;
|
|
|
|
+ else if (identifier == "TARGET_SONAME_FILE_DIR")
|
|
|
|
+ return &targetSoNameFileDirNode;
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+GeneratorExpressionContent::GeneratorExpressionContent(
|
|
|
|
+ const char *startContent,
|
|
|
|
+ unsigned int length)
|
|
|
|
+ : StartContent(startContent), ContentLength(length)
|
|
|
|
+{
|
|
|
|
+
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+std::string GeneratorExpressionContent::GetOriginalExpression() const
|
|
|
|
+{
|
|
|
|
+ return std::string(this->StartContent, this->ContentLength);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+std::string GeneratorExpressionContent::Evaluate(
|
|
|
|
+ cmGeneratorExpressionContext *context) const
|
|
|
|
+{
|
|
|
|
+ std::string identifier;
|
|
|
|
+ {
|
|
|
|
+ std::vector<cmGeneratorExpressionEvaluator*>::const_iterator it
|
|
|
|
+ = this->IdentifierChildren.begin();
|
|
|
|
+ const std::vector<cmGeneratorExpressionEvaluator*>::const_iterator end
|
|
|
|
+ = this->IdentifierChildren.end();
|
|
|
|
+ for ( ; it != end; ++it)
|
|
|
|
+ {
|
|
|
|
+ identifier += (*it)->Evaluate(context);
|
|
|
|
+ if (context->HadError)
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ const cmGeneratorExpressionNode *node = GetNode(identifier);
|
|
|
|
+
|
|
|
|
+ if (!node)
|
|
|
|
+ {
|
|
|
|
+ reportError(context, this->GetOriginalExpression(),
|
|
|
|
+ "Expression did not evaluate to a known generator expression");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (!node->GeneratesContent())
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (node->AcceptsSingleArbitraryContentParameter())
|
|
|
|
+ {
|
|
|
|
+ std::string result;
|
|
|
|
+ std::vector<std::vector<cmGeneratorExpressionEvaluator*> >::const_iterator
|
|
|
|
+ pit = this->ParamChildren.begin();
|
|
|
|
+ const
|
|
|
|
+ std::vector<std::vector<cmGeneratorExpressionEvaluator*> >::const_iterator
|
|
|
|
+ pend = this->ParamChildren.end();
|
|
|
|
+ for ( ; pit != pend; ++pit)
|
|
|
|
+ {
|
|
|
|
+ if (!result.empty())
|
|
|
|
+ {
|
|
|
|
+ result += ",";
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ std::vector<cmGeneratorExpressionEvaluator*>::const_iterator it
|
|
|
|
+ = pit->begin();
|
|
|
|
+ const std::vector<cmGeneratorExpressionEvaluator*>::const_iterator end
|
|
|
|
+ = pit->end();
|
|
|
|
+ for ( ; it != end; ++it)
|
|
|
|
+ {
|
|
|
|
+ result += (*it)->Evaluate(context);
|
|
|
|
+ if (context->HadError)
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ return result;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ std::vector<std::string> parameters;
|
|
|
|
+ {
|
|
|
|
+ std::vector<std::vector<cmGeneratorExpressionEvaluator*> >::const_iterator
|
|
|
|
+ pit = this->ParamChildren.begin();
|
|
|
|
+ const
|
|
|
|
+ std::vector<std::vector<cmGeneratorExpressionEvaluator*> >::const_iterator
|
|
|
|
+ pend = this->ParamChildren.end();
|
|
|
|
+ for ( ; pit != pend; ++pit)
|
|
|
|
+ {
|
|
|
|
+ std::string parameter;
|
|
|
|
+ std::vector<cmGeneratorExpressionEvaluator*>::const_iterator it =
|
|
|
|
+ pit->begin();
|
|
|
|
+ const std::vector<cmGeneratorExpressionEvaluator*>::const_iterator end =
|
|
|
|
+ pit->end();
|
|
|
|
+ for ( ; it != end; ++it)
|
|
|
|
+ {
|
|
|
|
+ parameter += (*it)->Evaluate(context);
|
|
|
|
+ if (context->HadError)
|
|
|
|
+ {
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ parameters.push_back(parameter);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ int numExpected = node->NumExpectedParameters();
|
|
|
|
+ if ((numExpected != -1 && (unsigned int)numExpected != parameters.size()))
|
|
|
|
+ {
|
|
|
|
+ if (numExpected == 0)
|
|
|
|
+ {
|
|
|
|
+ reportError(context, this->GetOriginalExpression(),
|
|
|
|
+ "$<" + identifier + "> expression requires no parameters.");
|
|
|
|
+ }
|
|
|
|
+ else if (numExpected == 1)
|
|
|
|
+ {
|
|
|
|
+ reportError(context, this->GetOriginalExpression(),
|
|
|
|
+ "$<" + identifier + "> expression requires "
|
|
|
|
+ "exactly one parameter.");
|
|
|
|
+ }
|
|
|
|
+ else
|
|
|
|
+ {
|
|
|
|
+ cmOStringStream e;
|
|
|
|
+ e << "$<" + identifier + "> expression requires "
|
|
|
|
+ << numExpected
|
|
|
|
+ << " comma separated parameters, but got "
|
|
|
|
+ << parameters.size() << " instead.";
|
|
|
|
+ reportError(context, this->GetOriginalExpression(), e.str());
|
|
|
|
+ }
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (numExpected == -1 && parameters.empty())
|
|
|
|
+ {
|
|
|
|
+ reportError(context, this->GetOriginalExpression(), "$<" + identifier
|
|
|
|
+ + "> expression requires at least one parameter.");
|
|
|
|
+ return std::string();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return node->Evaluate(parameters, context, this);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+static void deleteAll(const std::vector<cmGeneratorExpressionEvaluator*> &c)
|
|
|
|
+{
|
|
|
|
+ std::vector<cmGeneratorExpressionEvaluator*>::const_iterator it
|
|
|
|
+ = c.begin();
|
|
|
|
+ const std::vector<cmGeneratorExpressionEvaluator*>::const_iterator end
|
|
|
|
+ = c.end();
|
|
|
|
+ for ( ; it != end; ++it)
|
|
|
|
+ {
|
|
|
|
+ delete *it;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+//----------------------------------------------------------------------------
|
|
|
|
+GeneratorExpressionContent::~GeneratorExpressionContent()
|
|
|
|
+{
|
|
|
|
+ deleteAll(this->IdentifierChildren);
|
|
|
|
+
|
|
|
|
+ typedef std::vector<cmGeneratorExpressionEvaluator*> EvaluatorVector;
|
|
|
|
+ typedef std::vector<cmGeneratorExpressionToken> TokenVector;
|
|
|
|
+ std::vector<EvaluatorVector>::const_iterator pit =
|
|
|
|
+ this->ParamChildren.begin();
|
|
|
|
+ const std::vector<EvaluatorVector>::const_iterator pend =
|
|
|
|
+ this->ParamChildren.end();
|
|
|
|
+ for ( ; pit != pend; ++pit)
|
|
|
|
+ {
|
|
|
|
+ deleteAll(*pit);
|
|
|
|
+ }
|
|
|
|
+}
|