diff options
author | Alex Turbov <i.zaufi@gmail.com> | 2021-07-27 19:47:06 (GMT) |
---|---|---|
committer | Brad King <brad.king@kitware.com> | 2021-08-03 14:55:47 (GMT) |
commit | 866b0595f6f3e26542148984e22f508fb1b3eb1e (patch) | |
tree | b20322d97975991d4be792a0edfc1bf29b343b4d | |
parent | 51d9194a96df4cb8cd144afce4edc6e2122fab2d (diff) | |
download | CMake-866b0595f6f3e26542148984e22f508fb1b3eb1e.zip CMake-866b0595f6f3e26542148984e22f508fb1b3eb1e.tar.gz CMake-866b0595f6f3e26542148984e22f508fb1b3eb1e.tar.bz2 |
Refactor: Introduce `cmArgumentList` container class
The `cmArgumentList` has been turned into a class (forward declared in
the header). It inherits from the `std::list` (yeah, but we don't intend
to store polymorphic classes in it). In addition to the standard methods,
now it's possible to move `HandlePredicate` (renamed to `ReduceOneArg`)
and `HandleBinaryOp` (renamed to `ReduceTwoArgs`) as its members.
Additionally, iterators managements (`IncrementArguments`) have been
refactored into two separate classes mimicking iterators. This also
allows having a uniform `for` loop and concentrates the logic of
iterators advancing in it instead of the loop's body. The arguments
processing algorithms operate with "windows" over a collection of
arguments. Hence there are two kinds of "iteration windows" -- allowing
to observe 2 or 3 elements per loop iteration. These iteration "windows"
also passed to reducers.
-rw-r--r-- | Source/cmConditionEvaluator.cxx | 338 | ||||
-rw-r--r-- | Source/cmConditionEvaluator.h | 25 | ||||
-rw-r--r-- | Tests/RunCMake/if/IncompleteMatches-stdout.txt | 6 | ||||
-rw-r--r-- | Tests/RunCMake/if/IncompleteMatches.cmake | 36 | ||||
-rw-r--r-- | Tests/RunCMake/if/IncompleteMatchesFail-result.txt | 1 | ||||
-rw-r--r-- | Tests/RunCMake/if/IncompleteMatchesFail-stderr.txt | 6 | ||||
-rw-r--r-- | Tests/RunCMake/if/IncompleteMatchesFail.cmake | 2 | ||||
-rw-r--r-- | Tests/RunCMake/if/RunCMakeTest.cmake | 2 |
8 files changed, 262 insertions, 154 deletions
diff --git a/Source/cmConditionEvaluator.cxx b/Source/cmConditionEvaluator.cxx index 8224121..444d029 100644 --- a/Source/cmConditionEvaluator.cxx +++ b/Source/cmConditionEvaluator.cxx @@ -7,6 +7,7 @@ #include <cstdlib> #include <functional> #include <iterator> +#include <list> #include <sstream> #include <utility> @@ -15,6 +16,7 @@ #include "cmsys/RegularExpression.hxx" +#include "cmExpandedCommandArgument.h" #include "cmMakefile.h" #include "cmMessageType.h" #include "cmProperty.h" @@ -89,46 +91,102 @@ std::string bool2string(bool const value) { return std::string(std::size_t(1), static_cast<char>('0' + int(value))); } +} // anonymous namespace -void IncrementArguments(cmConditionEvaluator::cmArgumentList& newArgs, - cmConditionEvaluator::cmArgumentList::iterator& argP1) -{ - using difference_type = - cmConditionEvaluator::cmArgumentList::difference_type; - std::advance(argP1, difference_type(argP1 != newArgs.end())); -} +#if defined(__SUNPRO_CC) +# define CM_INHERIT_CTOR(Class, Base, Tpl) \ + template <typename... Args> \ + Class(Args&&... args) \ + : Base Tpl(std::forward<Args>(args)...) \ + { \ + } +#else +# define CM_INHERIT_CTOR(Class, Base, Tpl) using Base Tpl ::Base; +#endif -void IncrementArguments(cmConditionEvaluator::cmArgumentList& newArgs, - cmConditionEvaluator::cmArgumentList::iterator& argP1, - cmConditionEvaluator::cmArgumentList::iterator& argP2) +// BEGIN cmConditionEvaluator::cmArgumentList +class cmConditionEvaluator::cmArgumentList + : public std::list<cmExpandedCommandArgument> { - IncrementArguments(newArgs, argP1); - argP2 = argP1; - using difference_type = - cmConditionEvaluator::cmArgumentList::difference_type; - std::advance(argP2, difference_type(argP1 != newArgs.end())); -} + using base_t = std::list<cmExpandedCommandArgument>; -void HandlePredicate(const bool value, - cmConditionEvaluator::cmArgumentList& newArgs, - cmConditionEvaluator::cmArgumentList::iterator arg, - cmConditionEvaluator::cmArgumentList::iterator argP1) -{ - *arg = cmExpandedCommandArgument(bool2string(value), true); - newArgs.erase(argP1); -} +public: + CM_INHERIT_CTOR(cmArgumentList, list, <cmExpandedCommandArgument>); -void HandleBinaryOp(const bool value, - cmConditionEvaluator::cmArgumentList& newArgs, - cmConditionEvaluator::cmArgumentList::iterator arg, - cmConditionEvaluator::cmArgumentList::iterator argP1, - cmConditionEvaluator::cmArgumentList::iterator argP2) -{ - *arg = cmExpandedCommandArgument(bool2string(value), true); - newArgs.erase(argP2); - newArgs.erase(argP1); -} -} // anonymous namespace + class CurrentAndNextIter + { + friend class cmConditionEvaluator::cmArgumentList; + + public: + base_t::iterator current; + base_t::iterator next; + + CurrentAndNextIter advance(base_t& args) + { + this->current = std::next(this->current); + this->next = + std::next(this->current, difference_type(this->current != args.end())); + return *this; + } + + private: + CurrentAndNextIter(base_t& args) + : current(args.begin()) + , next(std::next(this->current, + difference_type(this->current != args.end()))) + { + } + }; + + class CurrentAndTwoMoreIter + { + friend class cmConditionEvaluator::cmArgumentList; + + public: + base_t::iterator current; + base_t::iterator next; + base_t::iterator nextnext; + + CurrentAndTwoMoreIter advance(base_t& args) + { + this->current = std::next(this->current); + this->next = + std::next(this->current, difference_type(this->current != args.end())); + this->nextnext = + std::next(this->next, difference_type(this->next != args.end())); + return *this; + } + + private: + CurrentAndTwoMoreIter(base_t& args) + : current(args.begin()) + , next(std::next(this->current, + difference_type(this->current != args.end()))) + , nextnext( + std::next(this->next, difference_type(this->next != args.end()))) + { + } + }; + + CurrentAndNextIter make2ArgsIterator() { return *this; } + CurrentAndTwoMoreIter make3ArgsIterator() { return *this; } + + template <typename Iter> + void ReduceOneArg(const bool value, Iter args) + { + *args.current = cmExpandedCommandArgument(bool2string(value), true); + this->erase(args.next); + } + + void ReduceTwoArgs(const bool value, CurrentAndTwoMoreIter args) + { + *args.current = cmExpandedCommandArgument(bool2string(value), true); + this->erase(args.nextnext); + this->erase(args.next); + } +}; + +// END cmConditionEvaluator::cmArgumentList cmConditionEvaluator::cmConditionEvaluator(cmMakefile& makefile, cmListFileBacktrace bt) @@ -455,15 +513,13 @@ bool cmConditionEvaluator::HandleLevel1(cmArgumentList& newArgs, std::string&, const auto policy64IsOld = this->Policy64Status == cmPolicies::OLD || this->Policy64Status == cmPolicies::WARN; - for (auto arg = newArgs.begin(), argP1 = arg; arg != newArgs.end(); - argP1 = ++arg) { - - IncrementArguments(newArgs, argP1); + for (auto args = newArgs.make2ArgsIterator(); args.current != newArgs.end(); + args.advance(newArgs)) { auto policyCheck = [&, this](const cmPolicies::PolicyID id, const cmPolicies::PolicyStatus status, const cm::static_string_view kw) { - if (status == cmPolicies::WARN && this->IsKeyword(kw, *arg)) { + if (status == cmPolicies::WARN && this->IsKeyword(kw, *args.current)) { std::ostringstream e; e << cmPolicies::GetPolicyWarning(id) << "\n" << kw @@ -481,73 +537,72 @@ bool cmConditionEvaluator::HandleLevel1(cmArgumentList& newArgs, std::string&, // NOTE Fail fast: All the predicates below require the next arg to be // valid - if (argP1 == newArgs.end()) { + if (args.next == newArgs.end()) { continue; } // does a file exist - if (this->IsKeyword(keyEXISTS, *arg)) { - HandlePredicate(cmSystemTools::FileExists(argP1->GetValue()), newArgs, - arg, argP1); + if (this->IsKeyword(keyEXISTS, *args.current)) { + newArgs.ReduceOneArg(cmSystemTools::FileExists(args.next->GetValue()), + args); } // does a directory with this name exist - else if (this->IsKeyword(keyIS_DIRECTORY, *arg)) { - HandlePredicate(cmSystemTools::FileIsDirectory(argP1->GetValue()), - newArgs, arg, argP1); + else if (this->IsKeyword(keyIS_DIRECTORY, *args.current)) { + newArgs.ReduceOneArg( + cmSystemTools::FileIsDirectory(args.next->GetValue()), args); } // does a symlink with this name exist - else if (this->IsKeyword(keyIS_SYMLINK, *arg)) { - HandlePredicate(cmSystemTools::FileIsSymlink(argP1->GetValue()), newArgs, - arg, argP1); + else if (this->IsKeyword(keyIS_SYMLINK, *args.current)) { + newArgs.ReduceOneArg(cmSystemTools::FileIsSymlink(args.next->GetValue()), + args); } // is the given path an absolute path ? - else if (this->IsKeyword(keyIS_ABSOLUTE, *arg)) { - HandlePredicate(cmSystemTools::FileIsFullPath(argP1->GetValue()), - newArgs, arg, argP1); + else if (this->IsKeyword(keyIS_ABSOLUTE, *args.current)) { + newArgs.ReduceOneArg( + cmSystemTools::FileIsFullPath(args.next->GetValue()), args); } // does a command exist - else if (this->IsKeyword(keyCOMMAND, *arg)) { - HandlePredicate( - this->Makefile.GetState()->GetCommand(argP1->GetValue()) != nullptr, - newArgs, arg, argP1); + else if (this->IsKeyword(keyCOMMAND, *args.current)) { + newArgs.ReduceOneArg( + bool(this->Makefile.GetState()->GetCommand(args.next->GetValue())), + args); } // does a policy exist - else if (this->IsKeyword(keyPOLICY, *arg)) { + else if (this->IsKeyword(keyPOLICY, *args.current)) { cmPolicies::PolicyID pid; - HandlePredicate(cmPolicies::GetPolicyID(argP1->GetValue().c_str(), pid), - newArgs, arg, argP1); + newArgs.ReduceOneArg( + cmPolicies::GetPolicyID(args.next->GetValue().c_str(), pid), args); } // does a target exist - else if (this->IsKeyword(keyTARGET, *arg)) { - HandlePredicate(this->Makefile.FindTargetToUse(argP1->GetValue()) != - nullptr, - newArgs, arg, argP1); + else if (this->IsKeyword(keyTARGET, *args.current)) { + newArgs.ReduceOneArg( + bool(this->Makefile.FindTargetToUse(args.next->GetValue())), args); } // is a variable defined - else if (this->IsKeyword(keyDEFINED, *arg)) { - const auto argP1len = argP1->GetValue().size(); + else if (this->IsKeyword(keyDEFINED, *args.current)) { + const auto argP1len = args.next->GetValue().size(); auto bdef = false; - if (argP1len > 4 && cmHasLiteralPrefix(argP1->GetValue(), "ENV{") && - argP1->GetValue().operator[](argP1len - 1) == '}') { - const auto env = argP1->GetValue().substr(4, argP1len - 5); + if (argP1len > 4 && cmHasLiteralPrefix(args.next->GetValue(), "ENV{") && + args.next->GetValue().operator[](argP1len - 1) == '}') { + const auto env = args.next->GetValue().substr(4, argP1len - 5); bdef = cmSystemTools::HasEnv(env); } else if (argP1len > 6 && - cmHasLiteralPrefix(argP1->GetValue(), "CACHE{") && - argP1->GetValue().operator[](argP1len - 1) == '}') { - const auto cache = argP1->GetValue().substr(6, argP1len - 7); - bdef = this->Makefile.GetState()->GetCacheEntryValue(cache) != nullptr; + cmHasLiteralPrefix(args.next->GetValue(), "CACHE{") && + args.next->GetValue().operator[](argP1len - 1) == '}') { + const auto cache = args.next->GetValue().substr(6, argP1len - 7); + bdef = bool(this->Makefile.GetState()->GetCacheEntryValue(cache)); } else { - bdef = this->Makefile.IsDefinitionSet(argP1->GetValue()); + bdef = this->Makefile.IsDefinitionSet(args.next->GetValue()); } - HandlePredicate(bdef, newArgs, arg, argP1); + newArgs.ReduceOneArg(bdef, args); } // does a test exist - else if (this->IsKeyword(keyTEST, *arg)) { + else if (this->IsKeyword(keyTEST, *args.current)) { if (policy64IsOld) { continue; } - HandlePredicate(this->Makefile.GetTest(argP1->GetValue()) != nullptr, - newArgs, arg, argP1); + newArgs.ReduceOneArg(bool(this->Makefile.GetTest(args.next->GetValue())), + args); } } return true; @@ -559,32 +614,32 @@ bool cmConditionEvaluator::HandleLevel2(cmArgumentList& newArgs, std::string& errorString, MessageType& status) { - for (auto arg = newArgs.begin(), argP1 = arg, argP2 = arg; - arg != newArgs.end(); argP1 = ++arg) { - - IncrementArguments(newArgs, argP1, argP2); + for (auto args = newArgs.make3ArgsIterator(); args.current != newArgs.end(); + args.advance(newArgs)) { int matchNo; - // NOTE Handle special case `if(... BLAH_BAH MATCHES)` + // NOTE Handle special case `if(... BLAH_BLAH MATCHES)` // (i.e., w/o regex to match which is possibly result of // variable expansion to an empty string) - if (argP1 != newArgs.end() && this->IsKeyword(keyMATCHES, *arg)) { - HandlePredicate(false, newArgs, arg, argP1); + if (args.next != newArgs.end() && + this->IsKeyword(keyMATCHES, *args.current)) { + newArgs.ReduceOneArg(false, args); } // NOTE Fail fast: All the binary ops below require 2 arguments. - else if (argP1 == newArgs.end() || argP2 == newArgs.end()) { + else if (args.next == newArgs.end() || args.nextnext == newArgs.end()) { continue; } - else if (this->IsKeyword(keyMATCHES, *argP1)) { - cmProp def = this->GetDefinitionIfUnquoted(*arg); + else if (this->IsKeyword(keyMATCHES, *args.next)) { + cmProp def = this->GetDefinitionIfUnquoted(*args.current); std::string def_buf; if (!def) { - def = &arg->GetValue(); - } else if (cmHasLiteralPrefix(arg->GetValue(), "CMAKE_MATCH_")) { + def = &args.current->GetValue(); + } else if (cmHasLiteralPrefix(args.current->GetValue(), + "CMAKE_MATCH_")) { // The string to match is owned by our match result variables. // Move it to our own buffer before clearing them. def_buf = *def; @@ -593,7 +648,7 @@ bool cmConditionEvaluator::HandleLevel2(cmArgumentList& newArgs, this->Makefile.ClearMatches(); - const auto& rex = argP2->GetValue(); + const auto& rex = args.nextnext->GetValue(); cmsys::RegularExpression regEntry; if (!regEntry.compile(rex)) { std::ostringstream error; @@ -607,15 +662,15 @@ bool cmConditionEvaluator::HandleLevel2(cmArgumentList& newArgs, if (match) { this->Makefile.StoreMatches(regEntry); } - HandleBinaryOp(match, newArgs, arg, argP1, argP2); + newArgs.ReduceTwoArgs(match, args); } else if ((matchNo = - this->matchKeys(*argP1, keyLESS, keyLESS_EQUAL, keyGREATER, + this->matchKeys(*args.next, keyLESS, keyLESS_EQUAL, keyGREATER, keyGREATER_EQUAL, keyEQUAL))) { - cmProp ldef = this->GetVariableOrString(*arg); - cmProp rdef = this->GetVariableOrString(*argP2); + cmProp ldef = this->GetVariableOrString(*args.current); + cmProp rdef = this->GetVariableOrString(*args.nextnext); double lhs; double rhs; @@ -623,23 +678,23 @@ bool cmConditionEvaluator::HandleLevel2(cmArgumentList& newArgs, return std::sscanf(ldef->c_str(), "%lg", &lhs) == 1 && std::sscanf(rdef->c_str(), "%lg", &rhs) == 1; }; + // clang-format off const auto result = parseDoubles() && - // clang-format off cmRt2CtSelector< std::less, std::less_equal, std::greater, std::greater_equal, std::equal_to >::eval(matchNo, lhs, rhs); // clang-format on - HandleBinaryOp(result, newArgs, arg, argP1, argP2); + newArgs.ReduceTwoArgs(result, args); } - else if ((matchNo = this->matchKeys(*argP1, keySTRLESS, keySTRLESS_EQUAL, - keySTRGREATER, keySTRGREATER_EQUAL, - keySTREQUAL))) { + else if ((matchNo = this->matchKeys(*args.next, keySTRLESS, + keySTRLESS_EQUAL, keySTRGREATER, + keySTRGREATER_EQUAL, keySTREQUAL))) { - const cmProp lhs = this->GetVariableOrString(*arg); - const cmProp rhs = this->GetVariableOrString(*argP2); + const cmProp lhs = this->GetVariableOrString(*args.current); + const cmProp rhs = this->GetVariableOrString(*args.nextnext); const auto val = (*lhs).compare(*rhs); // clang-format off const auto result = cmRt2CtSelector< @@ -648,45 +703,46 @@ bool cmConditionEvaluator::HandleLevel2(cmArgumentList& newArgs, std::equal_to >::eval(matchNo, val, 0); // clang-format on - HandleBinaryOp(result, newArgs, arg, argP1, argP2); + newArgs.ReduceTwoArgs(result, args); } else if ((matchNo = - this->matchKeys(*argP1, keyVERSION_LESS, keyVERSION_LESS_EQUAL, - keyVERSION_GREATER, keyVERSION_GREATER_EQUAL, - keyVERSION_EQUAL))) { + this->matchKeys(*args.next, keyVERSION_LESS, + keyVERSION_LESS_EQUAL, keyVERSION_GREATER, + keyVERSION_GREATER_EQUAL, keyVERSION_EQUAL))) { + const cmSystemTools::CompareOp xlat[5] = { cmSystemTools::OP_LESS, cmSystemTools::OP_LESS_EQUAL, cmSystemTools::OP_GREATER, cmSystemTools::OP_GREATER_EQUAL, cmSystemTools::OP_EQUAL }; const auto op = xlat[matchNo - 1]; - const cmProp lhs = this->GetVariableOrString(*arg); - const cmProp rhs = this->GetVariableOrString(*argP2); + const cmProp lhs = this->GetVariableOrString(*args.current); + const cmProp rhs = this->GetVariableOrString(*args.nextnext); const auto result = cmSystemTools::VersionCompare(op, lhs->c_str(), rhs->c_str()); - HandleBinaryOp(result, newArgs, arg, argP1, argP2); + newArgs.ReduceTwoArgs(result, args); } // is file A newer than file B - else if (this->IsKeyword(keyIS_NEWER_THAN, *argP1)) { + else if (this->IsKeyword(keyIS_NEWER_THAN, *args.next)) { auto fileIsNewer = 0; cmsys::Status ftcStatus = cmSystemTools::FileTimeCompare( - arg->GetValue(), argP2->GetValue(), &fileIsNewer); - HandleBinaryOp((!ftcStatus || fileIsNewer == 1 || fileIsNewer == 0), - newArgs, arg, argP1, argP2); + args.current->GetValue(), args.nextnext->GetValue(), &fileIsNewer); + newArgs.ReduceTwoArgs( + (!ftcStatus || fileIsNewer == 1 || fileIsNewer == 0), args); } - else if (this->IsKeyword(keyIN_LIST, *argP1)) { + else if (this->IsKeyword(keyIN_LIST, *args.next)) { if (this->Policy57Status != cmPolicies::OLD && this->Policy57Status != cmPolicies::WARN) { - cmProp def = this->GetVariableOrString(*arg); - cmProp def2 = this->Makefile.GetDefinition(argP2->GetValue()); + cmProp def = this->GetVariableOrString(*args.current); + cmProp def2 = this->Makefile.GetDefinition(args.nextnext->GetValue()); - HandleBinaryOp(def2 && cm::contains(cmExpandedList(*def2, true), *def), - newArgs, arg, argP1, argP2); + newArgs.ReduceTwoArgs( + def2 && cm::contains(cmExpandedList(*def2, true), *def), args); } else if (this->Policy57Status == cmPolicies::WARN) { std::ostringstream e; e << cmPolicies::GetPolicyWarning(cmPolicies::CMP0057) @@ -708,15 +764,12 @@ bool cmConditionEvaluator::HandleLevel3(cmArgumentList& newArgs, std::string& errorString, MessageType& status) { - for (auto arg = newArgs.begin(), argP1 = arg; arg != newArgs.end(); - argP1 = ++arg) { - - IncrementArguments(newArgs, argP1); - - if (argP1 != newArgs.end() && this->IsKeyword(keyNOT, *arg)) { - const auto rhs = - this->GetBooleanValueWithAutoDereference(*argP1, errorString, status); - HandlePredicate(!rhs, newArgs, arg, argP1); + for (auto args = newArgs.make2ArgsIterator(); args.next != newArgs.end(); + args.advance(newArgs)) { + if (this->IsKeyword(keyNOT, *args.current)) { + const auto rhs = this->GetBooleanValueWithAutoDereference( + *args.next, errorString, status); + newArgs.ReduceOneArg(!rhs, args); } } return true; @@ -728,20 +781,23 @@ bool cmConditionEvaluator::HandleLevel4(cmArgumentList& newArgs, std::string& errorString, MessageType& status) { - for (auto arg = newArgs.begin(), argP1 = arg, argP2 = arg; - arg != newArgs.end(); argP1 = ++arg) { - - IncrementArguments(newArgs, argP1, argP2); - - if (argP1 != newArgs.end() && argP2 != newArgs.end() && - (this->IsKeyword(keyAND, *argP1) || this->IsKeyword(keyOR, *argP1))) { - const auto lhs = - this->GetBooleanValueWithAutoDereference(*arg, errorString, status); - const auto rhs = - this->GetBooleanValueWithAutoDereference(*argP2, errorString, status); - HandleBinaryOp(this->IsKeyword(keyAND, *argP1) ? (lhs && rhs) - : (lhs || rhs), - newArgs, arg, argP1, argP2); + for (auto args = newArgs.make3ArgsIterator(); args.nextnext != newArgs.end(); + args.advance(newArgs)) { + + int matchNo; + + if ((matchNo = this->matchKeys(*args.next, keyAND, keyOR))) { + const auto lhs = this->GetBooleanValueWithAutoDereference( + *args.current, errorString, status); + const auto rhs = this->GetBooleanValueWithAutoDereference( + *args.nextnext, errorString, status); + // clang-format off + const auto result = + cmRt2CtSelector< + std::logical_and, std::logical_or + >::eval(matchNo, lhs, rhs); + // clang-format on + newArgs.ReduceTwoArgs(result, args); } } return true; diff --git a/Source/cmConditionEvaluator.h b/Source/cmConditionEvaluator.h index 8367007..00d896b 100644 --- a/Source/cmConditionEvaluator.h +++ b/Source/cmConditionEvaluator.h @@ -4,25 +4,22 @@ #include "cmConfigure.h" // IWYU pragma: keep -#include <list> #include <string> #include <vector> #include <cmext/string_view> -#include "cmExpandedCommandArgument.h" #include "cmListFileCache.h" #include "cmMessageType.h" #include "cmPolicies.h" #include "cmProperty.h" +class cmExpandedCommandArgument; class cmMakefile; class cmConditionEvaluator { public: - using cmArgumentList = std::list<cmExpandedCommandArgument>; - cmConditionEvaluator(cmMakefile& makefile, cmListFileBacktrace bt); // this is a shared function for both If and Else to determine if the @@ -32,6 +29,8 @@ public: std::string& errorString, MessageType& status); private: + class cmArgumentList; + // Filter the given variable definition based on policy CMP0054. cmProp GetDefinitionIfUnquoted( const cmExpandedCommandArgument& argument) const; @@ -51,6 +50,15 @@ private: MessageType& status, bool oneArg = false) const; + template <int N> + int matchKeysImpl(const cmExpandedCommandArgument&); + + template <int N, typename T, typename... Keys> + int matchKeysImpl(const cmExpandedCommandArgument&, T, Keys...); + + template <typename... Keys> + int matchKeys(const cmExpandedCommandArgument&, Keys...); + bool HandleLevel0(cmArgumentList& newArgs, std::string& errorString, MessageType& status); @@ -65,15 +73,6 @@ private: bool HandleLevel4(cmArgumentList& newArgs, std::string& errorString, MessageType& status); - template <int N> - int matchKeysImpl(const cmExpandedCommandArgument&); - - template <int N, typename T, typename... Keys> - int matchKeysImpl(const cmExpandedCommandArgument&, T, Keys...); - - template <typename... Keys> - int matchKeys(const cmExpandedCommandArgument&, Keys...); - cmMakefile& Makefile; cmListFileBacktrace Backtrace; cmPolicies::PolicyStatus Policy12Status; diff --git a/Tests/RunCMake/if/IncompleteMatches-stdout.txt b/Tests/RunCMake/if/IncompleteMatches-stdout.txt new file mode 100644 index 0000000..634d988 --- /dev/null +++ b/Tests/RunCMake/if/IncompleteMatches-stdout.txt @@ -0,0 +1,6 @@ +-- Test #1 passed +-- Test #2 passed +-- Test #3 passed +-- Test #4 passed +-- Test #5 passed +-- Test #6 passed diff --git a/Tests/RunCMake/if/IncompleteMatches.cmake b/Tests/RunCMake/if/IncompleteMatches.cmake new file mode 100644 index 0000000..7142cfc --- /dev/null +++ b/Tests/RunCMake/if/IncompleteMatches.cmake @@ -0,0 +1,36 @@ +if(MATCHES) + message(SEND_ERROR "Test #1 failed") +else() + message(STATUS "Test #1 passed") +endif() + +if("" MATCHES "") + message(STATUS "Test #2 passed") +else() + message(SEND_ERROR "Test #2 failed") +endif() + +if(MATCHES RHS) + message(SEND_ERROR "Test #3 failed") +else() + message(STATUS "Test #3 passed") +endif() + +set(RHS "") +if(MATCHES RHS) + message(SEND_ERROR "Test #4 failed") +else() + message(STATUS "Test #4 passed") +endif() + +if(MATCHES "^$") + message(SEND_ERROR "Test #5 failed") +else() + message(STATUS "Test #5 passed") +endif() + +if("" MATCHES "^$") + message(STATUS "Test #6 passed") +else() + message(SEND_ERROR "Test #6 failed") +endif() diff --git a/Tests/RunCMake/if/IncompleteMatchesFail-result.txt b/Tests/RunCMake/if/IncompleteMatchesFail-result.txt new file mode 100644 index 0000000..d00491f --- /dev/null +++ b/Tests/RunCMake/if/IncompleteMatchesFail-result.txt @@ -0,0 +1 @@ +1 diff --git a/Tests/RunCMake/if/IncompleteMatchesFail-stderr.txt b/Tests/RunCMake/if/IncompleteMatchesFail-stderr.txt new file mode 100644 index 0000000..6e8ac80 --- /dev/null +++ b/Tests/RunCMake/if/IncompleteMatchesFail-stderr.txt @@ -0,0 +1,6 @@ +CMake Error at IncompleteMatchesFail\.cmake:1 \(if\): + if given arguments: + + "LHS" "MATCHES" + + Unknown arguments specified diff --git a/Tests/RunCMake/if/IncompleteMatchesFail.cmake b/Tests/RunCMake/if/IncompleteMatchesFail.cmake new file mode 100644 index 0000000..a3ef2b8 --- /dev/null +++ b/Tests/RunCMake/if/IncompleteMatchesFail.cmake @@ -0,0 +1,2 @@ +if(LHS MATCHES) +endif() diff --git a/Tests/RunCMake/if/RunCMakeTest.cmake b/Tests/RunCMake/if/RunCMakeTest.cmake index f54edf7..239c167 100644 --- a/Tests/RunCMake/if/RunCMakeTest.cmake +++ b/Tests/RunCMake/if/RunCMakeTest.cmake @@ -10,6 +10,8 @@ run_cmake(elseif-message) run_cmake(misplaced-elseif) run_cmake(MatchesSelf) +run_cmake(IncompleteMatches) +run_cmake(IncompleteMatchesFail) run_cmake(TestNameThatExists) run_cmake(TestNameThatDoesNotExist) |