ENH: Refactor function blocker deletion
When a function blocker decides to remove itself we previously removed it at every return point from the C++ scope in which its removal is needed. This teaches function blockers to transfer ownership of themselves from cmMakefile to an automatic variable for deletion on return. Since this removes blockers before they replay their commands, we no longer need to avoid running blockers on their own commands.
This commit is contained in:
parent
a541cac325
commit
2c81e5fb5c
|
@ -20,13 +20,6 @@ bool cmForEachFunctionBlocker::
|
||||||
IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
cmExecutionStatus &inStatus)
|
cmExecutionStatus &inStatus)
|
||||||
{
|
{
|
||||||
// Prevent recusion and don't let this blocker block its own
|
|
||||||
// commands.
|
|
||||||
if (this->Executing)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"foreach"))
|
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"foreach"))
|
||||||
{
|
{
|
||||||
// record the number of nested foreach commands
|
// record the number of nested foreach commands
|
||||||
|
@ -37,6 +30,10 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
// if this is the endofreach for this statement
|
// if this is the endofreach for this statement
|
||||||
if (!this->Depth)
|
if (!this->Depth)
|
||||||
{
|
{
|
||||||
|
// Remove the function blocker for this scope or bail.
|
||||||
|
cmsys::auto_ptr<cmFunctionBlocker> fb(mf.RemoveFunctionBlocker(lff));
|
||||||
|
if(!fb.get()) { return false; }
|
||||||
|
|
||||||
// at end of for each execute recorded commands
|
// at end of for each execute recorded commands
|
||||||
// store the old value
|
// store the old value
|
||||||
std::string oldDef;
|
std::string oldDef;
|
||||||
|
@ -44,7 +41,6 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
{
|
{
|
||||||
oldDef = mf.GetDefinition(this->Args[0].c_str());
|
oldDef = mf.GetDefinition(this->Args[0].c_str());
|
||||||
}
|
}
|
||||||
this->Executing = true;
|
|
||||||
std::vector<std::string>::const_iterator j = this->Args.begin();
|
std::vector<std::string>::const_iterator j = this->Args.begin();
|
||||||
++j;
|
++j;
|
||||||
|
|
||||||
|
@ -65,21 +61,18 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
inStatus.SetReturnInvoked(true);
|
inStatus.SetReturnInvoked(true);
|
||||||
// restore the variable to its prior value
|
// restore the variable to its prior value
|
||||||
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (status.GetBreakInvoked())
|
if (status.GetBreakInvoked())
|
||||||
{
|
{
|
||||||
// restore the variable to its prior value
|
// restore the variable to its prior value
|
||||||
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// restore the variable to its prior value
|
// restore the variable to its prior value
|
||||||
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
mf.AddDefinition(this->Args[0].c_str(),oldDef.c_str());
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
class cmForEachFunctionBlocker : public cmFunctionBlocker
|
class cmForEachFunctionBlocker : public cmFunctionBlocker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
cmForEachFunctionBlocker() {this->Executing = false; Depth = 0;}
|
cmForEachFunctionBlocker() {this->Depth = 0;}
|
||||||
virtual ~cmForEachFunctionBlocker() {}
|
virtual ~cmForEachFunctionBlocker() {}
|
||||||
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
cmMakefile &mf,
|
cmMakefile &mf,
|
||||||
|
@ -39,7 +39,6 @@ public:
|
||||||
|
|
||||||
std::vector<std::string> Args;
|
std::vector<std::string> Args;
|
||||||
std::vector<cmListFileFunction> Functions;
|
std::vector<cmListFileFunction> Functions;
|
||||||
bool Executing;
|
|
||||||
private:
|
private:
|
||||||
int Depth;
|
int Depth;
|
||||||
};
|
};
|
||||||
|
|
|
@ -27,13 +27,6 @@ IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
cmMakefile &mf,
|
cmMakefile &mf,
|
||||||
cmExecutionStatus &inStatus)
|
cmExecutionStatus &inStatus)
|
||||||
{
|
{
|
||||||
// Prevent recusion and don't let this blocker block its own
|
|
||||||
// commands.
|
|
||||||
if (this->Executing)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// we start by recording all the functions
|
// we start by recording all the functions
|
||||||
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"if"))
|
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"if"))
|
||||||
{
|
{
|
||||||
|
@ -45,8 +38,11 @@ IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
// if this is the endif for this if statement, then start executing
|
// if this is the endif for this if statement, then start executing
|
||||||
if (!this->ScopeDepth)
|
if (!this->ScopeDepth)
|
||||||
{
|
{
|
||||||
|
// Remove the function blocker for this scope or bail.
|
||||||
|
cmsys::auto_ptr<cmFunctionBlocker> fb(mf.RemoveFunctionBlocker(lff));
|
||||||
|
if(!fb.get()) { return false; }
|
||||||
|
|
||||||
// execute the functions for the true parts of the if statement
|
// execute the functions for the true parts of the if statement
|
||||||
this->Executing = true;
|
|
||||||
cmExecutionStatus status;
|
cmExecutionStatus status;
|
||||||
int scopeDepth = 0;
|
int scopeDepth = 0;
|
||||||
for(unsigned int c = 0; c < this->Functions.size(); ++c)
|
for(unsigned int c = 0; c < this->Functions.size(); ++c)
|
||||||
|
@ -104,7 +100,6 @@ IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
err += ").";
|
err += ").";
|
||||||
mf.IssueMessage(cmake::FATAL_ERROR, err);
|
mf.IssueMessage(cmake::FATAL_ERROR, err);
|
||||||
cmSystemTools::SetFatalErrorOccured();
|
cmSystemTools::SetFatalErrorOccured();
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -124,18 +119,15 @@ IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
if (status.GetReturnInvoked())
|
if (status.GetReturnInvoked())
|
||||||
{
|
{
|
||||||
inStatus.SetReturnInvoked(true);
|
inStatus.SetReturnInvoked(true);
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (status.GetBreakInvoked())
|
if (status.GetBreakInvoked())
|
||||||
{
|
{
|
||||||
inStatus.SetBreakInvoked(true);
|
inStatus.SetBreakInvoked(true);
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,7 @@ class cmIfFunctionBlocker : public cmFunctionBlocker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
cmIfFunctionBlocker() {
|
cmIfFunctionBlocker() {
|
||||||
this->HasRun = false; this->ScopeDepth = 0; this->Executing = false;}
|
this->HasRun = false; this->ScopeDepth = 0; }
|
||||||
virtual ~cmIfFunctionBlocker() {}
|
virtual ~cmIfFunctionBlocker() {}
|
||||||
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
cmMakefile &mf,
|
cmMakefile &mf,
|
||||||
|
@ -43,7 +43,6 @@ public:
|
||||||
bool IsBlocking;
|
bool IsBlocking;
|
||||||
bool HasRun;
|
bool HasRun;
|
||||||
unsigned int ScopeDepth;
|
unsigned int ScopeDepth;
|
||||||
bool Executing;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** \class cmIfCommand
|
/** \class cmIfCommand
|
||||||
|
|
|
@ -2395,7 +2395,8 @@ bool cmMakefile::ExpandArguments(
|
||||||
return !cmSystemTools::GetFatalErrorOccured();
|
return !cmSystemTools::GetFatalErrorOccured();
|
||||||
}
|
}
|
||||||
|
|
||||||
void cmMakefile::RemoveFunctionBlocker(const cmListFileFunction& lff)
|
cmsys::auto_ptr<cmFunctionBlocker>
|
||||||
|
cmMakefile::RemoveFunctionBlocker(const cmListFileFunction& lff)
|
||||||
{
|
{
|
||||||
// loop over all function blockers to see if any block this command
|
// loop over all function blockers to see if any block this command
|
||||||
std::list<cmFunctionBlocker *>::reverse_iterator pos;
|
std::list<cmFunctionBlocker *>::reverse_iterator pos;
|
||||||
|
@ -2406,12 +2407,11 @@ void cmMakefile::RemoveFunctionBlocker(const cmListFileFunction& lff)
|
||||||
{
|
{
|
||||||
cmFunctionBlocker* b = *pos;
|
cmFunctionBlocker* b = *pos;
|
||||||
this->FunctionBlockers.remove(b);
|
this->FunctionBlockers.remove(b);
|
||||||
delete b;
|
return cmsys::auto_ptr<cmFunctionBlocker>(b);
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return cmsys::auto_ptr<cmFunctionBlocker>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void cmMakefile::SetHomeDirectory(const char* dir)
|
void cmMakefile::SetHomeDirectory(const char* dir)
|
||||||
|
|
|
@ -31,6 +31,7 @@
|
||||||
#include "cmSourceGroup.h"
|
#include "cmSourceGroup.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <cmsys/auto_ptr.hxx>
|
||||||
#include <cmsys/RegularExpression.hxx>
|
#include <cmsys/RegularExpression.hxx>
|
||||||
|
|
||||||
class cmFunctionBlocker;
|
class cmFunctionBlocker;
|
||||||
|
@ -89,9 +90,13 @@ public:
|
||||||
*/
|
*/
|
||||||
void AddFunctionBlocker(cmFunctionBlocker *fb)
|
void AddFunctionBlocker(cmFunctionBlocker *fb)
|
||||||
{ this->FunctionBlockers.push_back(fb);}
|
{ this->FunctionBlockers.push_back(fb);}
|
||||||
void RemoveFunctionBlocker(cmFunctionBlocker *fb)
|
|
||||||
{ this->FunctionBlockers.remove(fb);}
|
/**
|
||||||
void RemoveFunctionBlocker(const cmListFileFunction& lff);
|
* Remove the function blocker whose scope ends with the given command.
|
||||||
|
* This returns ownership of the function blocker object.
|
||||||
|
*/
|
||||||
|
cmsys::auto_ptr<cmFunctionBlocker>
|
||||||
|
RemoveFunctionBlocker(const cmListFileFunction& lff);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Try running cmake and building a file. This is used for dynalically
|
* Try running cmake and building a file. This is used for dynalically
|
||||||
|
|
|
@ -21,13 +21,6 @@ bool cmWhileFunctionBlocker::
|
||||||
IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
cmExecutionStatus &inStatus)
|
cmExecutionStatus &inStatus)
|
||||||
{
|
{
|
||||||
// Prevent recusion and don't let this blocker block its own
|
|
||||||
// commands.
|
|
||||||
if (this->Executing)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// at end of for each execute recorded commands
|
// at end of for each execute recorded commands
|
||||||
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"while"))
|
if (!cmSystemTools::Strucmp(lff.Name.c_str(),"while"))
|
||||||
{
|
{
|
||||||
|
@ -39,6 +32,10 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
// if this is the endwhile for this while loop then execute
|
// if this is the endwhile for this while loop then execute
|
||||||
if (!this->Depth)
|
if (!this->Depth)
|
||||||
{
|
{
|
||||||
|
// Remove the function blocker for this scope or bail.
|
||||||
|
cmsys::auto_ptr<cmFunctionBlocker> fb(mf.RemoveFunctionBlocker(lff));
|
||||||
|
if(!fb.get()) { return false; }
|
||||||
|
|
||||||
std::string errorString;
|
std::string errorString;
|
||||||
|
|
||||||
std::vector<std::string> expandedArguments;
|
std::vector<std::string> expandedArguments;
|
||||||
|
@ -46,7 +43,6 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
bool isTrue =
|
bool isTrue =
|
||||||
cmIfCommand::IsTrue(expandedArguments,errorString,&mf);
|
cmIfCommand::IsTrue(expandedArguments,errorString,&mf);
|
||||||
|
|
||||||
this->Executing = true;
|
|
||||||
while (isTrue)
|
while (isTrue)
|
||||||
{
|
{
|
||||||
// Invoke all the functions that were collected in the block.
|
// Invoke all the functions that were collected in the block.
|
||||||
|
@ -57,12 +53,10 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
if (status.GetReturnInvoked())
|
if (status.GetReturnInvoked())
|
||||||
{
|
{
|
||||||
inStatus.SetReturnInvoked(true);
|
inStatus.SetReturnInvoked(true);
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (status.GetBreakInvoked())
|
if (status.GetBreakInvoked())
|
||||||
{
|
{
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -71,7 +65,6 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
isTrue =
|
isTrue =
|
||||||
cmIfCommand::IsTrue(expandedArguments,errorString,&mf);
|
cmIfCommand::IsTrue(expandedArguments,errorString,&mf);
|
||||||
}
|
}
|
||||||
mf.RemoveFunctionBlocker(lff);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
class cmWhileFunctionBlocker : public cmFunctionBlocker
|
class cmWhileFunctionBlocker : public cmFunctionBlocker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
cmWhileFunctionBlocker() {Executing = false; Depth=0;}
|
cmWhileFunctionBlocker() {this->Depth=0;}
|
||||||
virtual ~cmWhileFunctionBlocker() {}
|
virtual ~cmWhileFunctionBlocker() {}
|
||||||
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
virtual bool IsFunctionBlocked(const cmListFileFunction& lff,
|
||||||
cmMakefile &mf,
|
cmMakefile &mf,
|
||||||
|
@ -39,7 +39,6 @@ public:
|
||||||
|
|
||||||
std::vector<cmListFileArgument> Args;
|
std::vector<cmListFileArgument> Args;
|
||||||
std::vector<cmListFileFunction> Functions;
|
std::vector<cmListFileFunction> Functions;
|
||||||
bool Executing;
|
|
||||||
private:
|
private:
|
||||||
int Depth;
|
int Depth;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue