123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 |
- /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
- file Copyright.txt or https://cmake.org/licensing for details. */
- #include "cmForEachCommand.h"
- #include <memory> // IWYU pragma: keep
- #include <sstream>
- #include <stdio.h>
- #include <stdlib.h>
- #include "cmAlgorithms.h"
- #include "cmExecutionStatus.h"
- #include "cmMakefile.h"
- #include "cmSystemTools.h"
- #include "cmake.h"
- cmForEachFunctionBlocker::cmForEachFunctionBlocker(cmMakefile* mf)
- : Makefile(mf)
- , Depth(0)
- {
- this->Makefile->PushLoopBlock();
- }
- cmForEachFunctionBlocker::~cmForEachFunctionBlocker()
- {
- this->Makefile->PopLoopBlock();
- }
- bool cmForEachFunctionBlocker::IsFunctionBlocked(const cmListFileFunction& lff,
- cmMakefile& mf,
- cmExecutionStatus& inStatus)
- {
- if (!cmSystemTools::Strucmp(lff.Name.c_str(), "foreach")) {
- // record the number of nested foreach commands
- this->Depth++;
- } else if (!cmSystemTools::Strucmp(lff.Name.c_str(), "endforeach")) {
- // if this is the endofreach for this statement
- if (!this->Depth) {
- // Remove the function blocker for this scope or bail.
- std::unique_ptr<cmFunctionBlocker> fb(
- mf.RemoveFunctionBlocker(this, lff));
- if (!fb.get()) {
- return false;
- }
- // at end of for each execute recorded commands
- // store the old value
- std::string oldDef;
- if (mf.GetDefinition(this->Args[0])) {
- oldDef = mf.GetDefinition(this->Args[0]);
- }
- std::vector<std::string>::const_iterator j = this->Args.begin();
- ++j;
- for (; j != this->Args.end(); ++j) {
- // set the variable to the loop value
- mf.AddDefinition(this->Args[0], j->c_str());
- // Invoke all the functions that were collected in the block.
- cmExecutionStatus status;
- for (cmListFileFunction const& func : this->Functions) {
- status.Clear();
- mf.ExecuteCommand(func, status);
- if (status.GetReturnInvoked()) {
- inStatus.SetReturnInvoked();
- // restore the variable to its prior value
- mf.AddDefinition(this->Args[0], oldDef.c_str());
- return true;
- }
- if (status.GetBreakInvoked()) {
- // restore the variable to its prior value
- mf.AddDefinition(this->Args[0], oldDef.c_str());
- return true;
- }
- if (status.GetContinueInvoked()) {
- break;
- }
- if (cmSystemTools::GetFatalErrorOccured()) {
- return true;
- }
- }
- }
- // restore the variable to its prior value
- mf.AddDefinition(this->Args[0], oldDef.c_str());
- return true;
- }
- // close out a nested foreach
- this->Depth--;
- }
- // record the command
- this->Functions.push_back(lff);
- // always return true
- return true;
- }
- bool cmForEachFunctionBlocker::ShouldRemove(const cmListFileFunction& lff,
- cmMakefile& mf)
- {
- if (!cmSystemTools::Strucmp(lff.Name.c_str(), "endforeach")) {
- std::vector<std::string> expandedArguments;
- mf.ExpandArguments(lff.Arguments, expandedArguments);
- // if the endforeach has arguments then make sure
- // they match the begin foreach arguments
- if ((expandedArguments.empty() ||
- (expandedArguments[0] == this->Args[0]))) {
- return true;
- }
- }
- return false;
- }
- bool cmForEachCommand::InitialPass(std::vector<std::string> const& args,
- cmExecutionStatus&)
- {
- if (args.empty()) {
- this->SetError("called with incorrect number of arguments");
- return false;
- }
- if (args.size() > 1 && args[1] == "IN") {
- return this->HandleInMode(args);
- }
- // create a function blocker
- auto f = cm::make_unique<cmForEachFunctionBlocker>(this->Makefile);
- if (args.size() > 1) {
- if (args[1] == "RANGE") {
- int start = 0;
- int stop = 0;
- int step = 0;
- if (args.size() == 3) {
- stop = atoi(args[2].c_str());
- }
- if (args.size() == 4) {
- start = atoi(args[2].c_str());
- stop = atoi(args[3].c_str());
- }
- if (args.size() == 5) {
- start = atoi(args[2].c_str());
- stop = atoi(args[3].c_str());
- step = atoi(args[4].c_str());
- }
- if (step == 0) {
- if (start > stop) {
- step = -1;
- } else {
- step = 1;
- }
- }
- if ((start > stop && step > 0) || (start < stop && step < 0) ||
- step == 0) {
- std::ostringstream str;
- str << "called with incorrect range specification: start ";
- str << start << ", stop " << stop << ", step " << step;
- this->SetError(str.str());
- return false;
- }
- std::vector<std::string> range;
- char buffer[100];
- range.push_back(args[0]);
- int cc;
- for (cc = start;; cc += step) {
- if ((step > 0 && cc > stop) || (step < 0 && cc < stop)) {
- break;
- }
- sprintf(buffer, "%d", cc);
- range.push_back(buffer);
- if (cc == stop) {
- break;
- }
- }
- f->Args = range;
- } else {
- f->Args = args;
- }
- } else {
- f->Args = args;
- }
- this->Makefile->AddFunctionBlocker(f.release());
- return true;
- }
- bool cmForEachCommand::HandleInMode(std::vector<std::string> const& args)
- {
- std::unique_ptr<cmForEachFunctionBlocker> f(
- new cmForEachFunctionBlocker(this->Makefile));
- f->Args.push_back(args[0]);
- enum Doing
- {
- DoingNone,
- DoingLists,
- DoingItems
- };
- Doing doing = DoingNone;
- for (unsigned int i = 2; i < args.size(); ++i) {
- if (doing == DoingItems) {
- f->Args.push_back(args[i]);
- } else if (args[i] == "LISTS") {
- doing = DoingLists;
- } else if (args[i] == "ITEMS") {
- doing = DoingItems;
- } else if (doing == DoingLists) {
- const char* value = this->Makefile->GetDefinition(args[i]);
- if (value && *value) {
- cmSystemTools::ExpandListArgument(value, f->Args, true);
- }
- } else {
- std::ostringstream e;
- e << "Unknown argument:\n"
- << " " << args[i] << "\n";
- this->Makefile->IssueMessage(cmake::FATAL_ERROR, e.str());
- return true;
- }
- }
- this->Makefile->AddFunctionBlocker(f.release()); // TODO: pass unique_ptr
- return true;
- }
|