/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmListCommand.h" #include "cmsys/RegularExpression.hxx" #include #include #include #include #include #include // required for atoi #include "cmAlgorithms.h" #include "cmMakefile.h" #include "cmPolicies.h" #include "cmSystemTools.h" #include "cmake.h" class cmExecutionStatus; bool cmListCommand::InitialPass(std::vector const& args, cmExecutionStatus&) { if (args.size() < 2) { this->SetError("must be called with at least two arguments."); return false; } const std::string& subCommand = args[0]; if (subCommand == "LENGTH") { return this->HandleLengthCommand(args); } if (subCommand == "GET") { return this->HandleGetCommand(args); } if (subCommand == "APPEND") { return this->HandleAppendCommand(args); } if (subCommand == "FIND") { return this->HandleFindCommand(args); } if (subCommand == "INSERT") { return this->HandleInsertCommand(args); } if (subCommand == "JOIN") { return this->HandleJoinCommand(args); } if (subCommand == "REMOVE_AT") { return this->HandleRemoveAtCommand(args); } if (subCommand == "REMOVE_ITEM") { return this->HandleRemoveItemCommand(args); } if (subCommand == "REMOVE_DUPLICATES") { return this->HandleRemoveDuplicatesCommand(args); } if (subCommand == "SORT") { return this->HandleSortCommand(args); } if (subCommand == "SUBLIST") { return this->HandleSublistCommand(args); } if (subCommand == "REVERSE") { return this->HandleReverseCommand(args); } if (subCommand == "FILTER") { return this->HandleFilterCommand(args); } std::string e = "does not recognize sub-command " + subCommand; this->SetError(e); return false; } bool cmListCommand::GetListString(std::string& listString, const std::string& var) { // get the old value const char* cacheValue = this->Makefile->GetDefinition(var); if (!cacheValue) { return false; } listString = cacheValue; return true; } bool cmListCommand::GetList(std::vector& list, const std::string& var) { std::string listString; if (!this->GetListString(listString, var)) { return false; } // if the size of the list if (listString.empty()) { return true; } // expand the variable into a list cmSystemTools::ExpandListArgument(listString, list, true); // if no empty elements then just return if (std::find(list.begin(), list.end(), std::string()) == list.end()) { return true; } // if we have empty elements we need to check policy CMP0007 switch (this->Makefile->GetPolicyStatus(cmPolicies::CMP0007)) { case cmPolicies::WARN: { // Default is to warn and use old behavior // OLD behavior is to allow compatibility, so recall // ExpandListArgument without the true which will remove // empty values list.clear(); cmSystemTools::ExpandListArgument(listString, list); std::string warn = cmPolicies::GetPolicyWarning(cmPolicies::CMP0007); warn += " List has value = ["; warn += listString; warn += "]."; this->Makefile->IssueMessage(cmake::AUTHOR_WARNING, warn); return true; } case cmPolicies::OLD: // OLD behavior is to allow compatibility, so recall // ExpandListArgument without the true which will remove // empty values list.clear(); cmSystemTools::ExpandListArgument(listString, list); return true; case cmPolicies::NEW: return true; case cmPolicies::REQUIRED_IF_USED: case cmPolicies::REQUIRED_ALWAYS: this->Makefile->IssueMessage( cmake::FATAL_ERROR, cmPolicies::GetRequiredPolicyError(cmPolicies::CMP0007)); return false; } return true; } bool cmListCommand::HandleLengthCommand(std::vector const& args) { if (args.size() != 3) { this->SetError("sub-command LENGTH requires two arguments."); return false; } const std::string& listName = args[1]; const std::string& variableName = args[args.size() - 1]; std::vector varArgsExpanded; // do not check the return value here // if the list var is not found varArgsExpanded will have size 0 // and we will return 0 this->GetList(varArgsExpanded, listName); size_t length = varArgsExpanded.size(); char buffer[1024]; sprintf(buffer, "%d", static_cast(length)); this->Makefile->AddDefinition(variableName, buffer); return true; } bool cmListCommand::HandleGetCommand(std::vector const& args) { if (args.size() < 4) { this->SetError("sub-command GET requires at least three arguments."); return false; } const std::string& listName = args[1]; const std::string& variableName = args[args.size() - 1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->Makefile->AddDefinition(variableName, "NOTFOUND"); return true; } // FIXME: Add policy to make non-existing lists an error like empty lists. if (varArgsExpanded.empty()) { this->SetError("GET given empty list"); return false; } std::string value; size_t cc; const char* sep = ""; size_t nitem = varArgsExpanded.size(); for (cc = 2; cc < args.size() - 1; cc++) { int item = atoi(args[cc].c_str()); value += sep; sep = ";"; if (item < 0) { item = static_cast(nitem) + item; } if (item < 0 || nitem <= static_cast(item)) { std::ostringstream str; str << "index: " << item << " out of range (-" << nitem << ", " << nitem - 1 << ")"; this->SetError(str.str()); return false; } value += varArgsExpanded[item]; } this->Makefile->AddDefinition(variableName, value.c_str()); return true; } bool cmListCommand::HandleAppendCommand(std::vector const& args) { assert(args.size() >= 2); // Skip if nothing to append. if (args.size() < 3) { return true; } const std::string& listName = args[1]; // expand the variable std::string listString; this->GetListString(listString, listName); if (!listString.empty() && !args.empty()) { listString += ";"; } listString += cmJoin(cmMakeRange(args).advance(2), ";"); this->Makefile->AddDefinition(listName, listString.c_str()); return true; } bool cmListCommand::HandleFindCommand(std::vector const& args) { if (args.size() != 4) { this->SetError("sub-command FIND requires three arguments."); return false; } const std::string& listName = args[1]; const std::string& variableName = args[args.size() - 1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->Makefile->AddDefinition(variableName, "-1"); return true; } std::vector::iterator it = std::find(varArgsExpanded.begin(), varArgsExpanded.end(), args[2]); if (it != varArgsExpanded.end()) { std::ostringstream indexStream; indexStream << std::distance(varArgsExpanded.begin(), it); this->Makefile->AddDefinition(variableName, indexStream.str().c_str()); return true; } this->Makefile->AddDefinition(variableName, "-1"); return true; } bool cmListCommand::HandleInsertCommand(std::vector const& args) { if (args.size() < 4) { this->SetError("sub-command INSERT requires at least three arguments."); return false; } const std::string& listName = args[1]; // expand the variable int item = atoi(args[2].c_str()); std::vector varArgsExpanded; if ((!this->GetList(varArgsExpanded, listName) || varArgsExpanded.empty()) && item != 0) { std::ostringstream str; str << "index: " << item << " out of range (0, 0)"; this->SetError(str.str()); return false; } if (!varArgsExpanded.empty()) { size_t nitem = varArgsExpanded.size(); if (item < 0) { item = static_cast(nitem) + item; } if (item < 0 || nitem <= static_cast(item)) { std::ostringstream str; str << "index: " << item << " out of range (-" << varArgsExpanded.size() << ", " << (varArgsExpanded.empty() ? 0 : (varArgsExpanded.size() - 1)) << ")"; this->SetError(str.str()); return false; } } varArgsExpanded.insert(varArgsExpanded.begin() + item, args.begin() + 3, args.end()); std::string value = cmJoin(varArgsExpanded, ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleJoinCommand(std::vector const& args) { if (args.size() != 4) { std::ostringstream error; error << "sub-command JOIN requires three arguments (" << args.size() - 1 << " found)."; this->SetError(error.str()); return false; } const std::string& listName = args[1]; const std::string& glue = args[2]; const std::string& variableName = args[3]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->Makefile->AddDefinition(variableName, ""); return true; } std::string value = cmJoin(cmMakeRange(varArgsExpanded.begin(), varArgsExpanded.end()), glue); this->Makefile->AddDefinition(variableName, value.c_str()); return true; } bool cmListCommand::HandleRemoveItemCommand( std::vector const& args) { if (args.size() < 3) { this->SetError("sub-command REMOVE_ITEM requires two or more arguments."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError("sub-command REMOVE_ITEM requires list to be present."); return false; } std::vector remove(args.begin() + 2, args.end()); std::sort(remove.begin(), remove.end()); std::vector::const_iterator remEnd = std::unique(remove.begin(), remove.end()); std::vector::const_iterator remBegin = remove.begin(); std::vector::const_iterator argsEnd = cmRemoveMatching(varArgsExpanded, cmMakeRange(remBegin, remEnd)); std::vector::const_iterator argsBegin = varArgsExpanded.begin(); std::string value = cmJoin(cmMakeRange(argsBegin, argsEnd), ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleReverseCommand(std::vector const& args) { assert(args.size() >= 2); if (args.size() > 2) { this->SetError("sub-command REVERSE only takes one argument."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError("sub-command REVERSE requires list to be present."); return false; } std::string value = cmJoin(cmReverseRange(varArgsExpanded), ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleRemoveDuplicatesCommand( std::vector const& args) { assert(args.size() >= 2); if (args.size() > 2) { this->SetError("sub-command REMOVE_DUPLICATES only takes one argument."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError( "sub-command REMOVE_DUPLICATES requires list to be present."); return false; } std::vector::const_iterator argsEnd = cmRemoveDuplicates(varArgsExpanded); std::vector::const_iterator argsBegin = varArgsExpanded.begin(); std::string value = cmJoin(cmMakeRange(argsBegin, argsEnd), ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleSortCommand(std::vector const& args) { assert(args.size() >= 2); if (args.size() > 2) { this->SetError("sub-command SORT only takes one argument."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError("sub-command SORT requires list to be present."); return false; } std::sort(varArgsExpanded.begin(), varArgsExpanded.end()); std::string value = cmJoin(varArgsExpanded, ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleSublistCommand(std::vector const& args) { if (args.size() != 5) { std::ostringstream error; error << "sub-command SUBLIST requires four arguments (" << args.size() - 1 << " found)."; this->SetError(error.str()); return false; } const std::string& listName = args[1]; const std::string& variableName = args[args.size() - 1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName) || varArgsExpanded.empty()) { this->Makefile->AddDefinition(variableName, ""); return true; } const int start = atoi(args[2].c_str()); const int length = atoi(args[3].c_str()); using size_type = decltype(varArgsExpanded)::size_type; if (start < 0 || size_type(start) >= varArgsExpanded.size()) { std::ostringstream error; error << "begin index: " << start << " is out of range 0 - " << varArgsExpanded.size() - 1; this->SetError(error.str()); return false; } if (length < -1) { std::ostringstream error; error << "length: " << length << " should be -1 or greater"; this->SetError(error.str()); return false; } const size_type end = (length == -1 || size_type(start + length) > varArgsExpanded.size()) ? varArgsExpanded.size() : size_type(start + length); std::vector sublist(varArgsExpanded.begin() + start, varArgsExpanded.begin() + end); this->Makefile->AddDefinition(variableName, cmJoin(sublist, ";").c_str()); return true; } bool cmListCommand::HandleRemoveAtCommand(std::vector const& args) { if (args.size() < 3) { this->SetError("sub-command REMOVE_AT requires at least " "two arguments."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError("sub-command REMOVE_AT requires list to be present."); return false; } // FIXME: Add policy to make non-existing lists an error like empty lists. if (varArgsExpanded.empty()) { this->SetError("REMOVE_AT given empty list"); return false; } size_t cc; std::vector removed; size_t nitem = varArgsExpanded.size(); for (cc = 2; cc < args.size(); ++cc) { int item = atoi(args[cc].c_str()); if (item < 0) { item = static_cast(nitem) + item; } if (item < 0 || nitem <= static_cast(item)) { std::ostringstream str; str << "index: " << item << " out of range (-" << nitem << ", " << nitem - 1 << ")"; this->SetError(str.str()); return false; } removed.push_back(static_cast(item)); } std::sort(removed.begin(), removed.end()); std::vector::const_iterator remEnd = std::unique(removed.begin(), removed.end()); std::vector::const_iterator remBegin = removed.begin(); std::vector::const_iterator argsEnd = cmRemoveIndices(varArgsExpanded, cmMakeRange(remBegin, remEnd)); std::vector::const_iterator argsBegin = varArgsExpanded.begin(); std::string value = cmJoin(cmMakeRange(argsBegin, argsEnd), ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; } bool cmListCommand::HandleFilterCommand(std::vector const& args) { if (args.size() < 2) { this->SetError("sub-command FILTER requires a list to be specified."); return false; } if (args.size() < 3) { this->SetError("sub-command FILTER requires an operator to be specified."); return false; } if (args.size() < 4) { this->SetError("sub-command FILTER requires a mode to be specified."); return false; } const std::string& listName = args[1]; // expand the variable std::vector varArgsExpanded; if (!this->GetList(varArgsExpanded, listName)) { this->SetError("sub-command FILTER requires list to be present."); return false; } const std::string& op = args[2]; bool includeMatches; if (op == "INCLUDE") { includeMatches = true; } else if (op == "EXCLUDE") { includeMatches = false; } else { this->SetError("sub-command FILTER does not recognize operator " + op); return false; } const std::string& mode = args[3]; if (mode == "REGEX") { if (args.size() != 5) { this->SetError("sub-command FILTER, mode REGEX " "requires five arguments."); return false; } return this->FilterRegex(args, includeMatches, listName, varArgsExpanded); } this->SetError("sub-command FILTER does not recognize mode " + mode); return false; } class MatchesRegex { public: MatchesRegex(cmsys::RegularExpression& in_regex, bool in_includeMatches) : regex(in_regex) , includeMatches(in_includeMatches) { } bool operator()(const std::string& target) { return regex.find(target) ^ includeMatches; } private: cmsys::RegularExpression& regex; const bool includeMatches; }; bool cmListCommand::FilterRegex(std::vector const& args, bool includeMatches, std::string const& listName, std::vector& varArgsExpanded) { const std::string& pattern = args[4]; cmsys::RegularExpression regex(pattern); if (!regex.is_valid()) { std::string error = "sub-command FILTER, mode REGEX "; error += "failed to compile regex \""; error += pattern; error += "\"."; this->SetError(error); return false; } std::vector::iterator argsBegin = varArgsExpanded.begin(); std::vector::iterator argsEnd = varArgsExpanded.end(); std::vector::iterator newArgsEnd = std::remove_if(argsBegin, argsEnd, MatchesRegex(regex, includeMatches)); std::string value = cmJoin(cmMakeRange(argsBegin, newArgsEnd), ";"); this->Makefile->AddDefinition(listName, value.c_str()); return true; }