123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 |
- /*============================================================================
- CMake - Cross Platform Makefile Generator
- Copyright 2000-2009 Kitware, Inc., Insight Software Consortium
- 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 "cmForEachCommand.h"
- #include <cmsys/auto_ptr.hxx>
- 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.
- cmsys::auto_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;
- std::string tmps;
- cmListFileArgument arg;
- 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(unsigned int c = 0; c < this->Functions.size(); ++c)
- {
- status.Clear();
- mf.ExecuteCommand(this->Functions[c],status);
- if (status.GetReturnInvoked())
- {
- inStatus.SetReturnInvoked(true);
- // 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(cmSystemTools::GetFatalErrorOccured() )
- {
- return true;
- }
- }
- }
- // restore the variable to its prior value
- mf.AddDefinition(this->Args[0],oldDef.c_str());
- return true;
- }
- else
- {
- // 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.size() < 1)
- {
- 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
- cmForEachFunctionBlocker *f = new cmForEachFunctionBlocker();
- 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
- )
- {
- cmOStringStream 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);
- return true;
- }
- //----------------------------------------------------------------------------
- bool cmForEachCommand::HandleInMode(std::vector<std::string> const& args)
- {
- cmsys::auto_ptr<cmForEachFunctionBlocker> f(new cmForEachFunctionBlocker());
- 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
- {
- cmOStringStream 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 auto_ptr
- return true;
- }
|