Merge topic 'break-command-strictness'
d54617d0
break: Add policy CMP0055 to check calls strictlybae604d9
Track nested loop levels in CMake language with a stack of counters
This commit is contained in:
commit
82582c96bd
|
@ -112,3 +112,4 @@ All Policies
|
||||||
/policy/CMP0052
|
/policy/CMP0052
|
||||||
/policy/CMP0053
|
/policy/CMP0053
|
||||||
/policy/CMP0054
|
/policy/CMP0054
|
||||||
|
/policy/CMP0055
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
CMP0055
|
||||||
|
-------
|
||||||
|
|
||||||
|
Strict checking for the :command:`break` command.
|
||||||
|
|
||||||
|
CMake 3.1 and lower allowed calls to the :command:`break` command
|
||||||
|
outside of a loop context and also ignored any given arguments.
|
||||||
|
This was undefined behavior.
|
||||||
|
|
||||||
|
The OLD behavior for this policy is to allow :command:`break` to be placed
|
||||||
|
outside of loop contexts and ignores any arguments. The NEW behavior for this
|
||||||
|
policy is to issue an error if a misplaced break or any arguments are found.
|
||||||
|
|
||||||
|
This policy was introduced in CMake version 3.2.
|
||||||
|
CMake version |release| warns when the policy is not set and uses
|
||||||
|
OLD behavior. Use the cmake_policy command to set it to OLD or
|
||||||
|
NEW explicitly.
|
|
@ -0,0 +1,6 @@
|
||||||
|
break-command-strictness
|
||||||
|
------------------------
|
||||||
|
|
||||||
|
* The :command:`break` command now rejects calls outside of a loop
|
||||||
|
context or that pass arguments to the command.
|
||||||
|
See policy :policy:`CMP0055`.
|
|
@ -12,10 +12,76 @@
|
||||||
#include "cmBreakCommand.h"
|
#include "cmBreakCommand.h"
|
||||||
|
|
||||||
// cmBreakCommand
|
// cmBreakCommand
|
||||||
bool cmBreakCommand::InitialPass(std::vector<std::string> const&,
|
bool cmBreakCommand::InitialPass(std::vector<std::string> const &args,
|
||||||
cmExecutionStatus &status)
|
cmExecutionStatus &status)
|
||||||
{
|
{
|
||||||
|
if(!this->Makefile->IsLoopBlock())
|
||||||
|
{
|
||||||
|
bool issueMessage = true;
|
||||||
|
cmOStringStream e;
|
||||||
|
cmake::MessageType messageType = cmake::AUTHOR_WARNING;
|
||||||
|
switch(this->Makefile->GetPolicyStatus(cmPolicies::CMP0055))
|
||||||
|
{
|
||||||
|
case cmPolicies::WARN:
|
||||||
|
e << (this->Makefile->GetPolicies()
|
||||||
|
->GetPolicyWarning(cmPolicies::CMP0055)) << "\n";
|
||||||
|
break;
|
||||||
|
case cmPolicies::OLD:
|
||||||
|
issueMessage = false;
|
||||||
|
break;
|
||||||
|
case cmPolicies::REQUIRED_ALWAYS:
|
||||||
|
case cmPolicies::REQUIRED_IF_USED:
|
||||||
|
case cmPolicies::NEW:
|
||||||
|
messageType = cmake::FATAL_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(issueMessage)
|
||||||
|
{
|
||||||
|
e << "A BREAK command was found outside of a proper "
|
||||||
|
"FOREACH or WHILE loop scope.";
|
||||||
|
this->Makefile->IssueMessage(messageType, e.str());
|
||||||
|
if(messageType == cmake::FATAL_ERROR)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
status.SetBreakInvoked(true);
|
status.SetBreakInvoked(true);
|
||||||
|
|
||||||
|
if(!args.empty())
|
||||||
|
{
|
||||||
|
bool issueMessage = true;
|
||||||
|
cmOStringStream e;
|
||||||
|
cmake::MessageType messageType = cmake::AUTHOR_WARNING;
|
||||||
|
switch(this->Makefile->GetPolicyStatus(cmPolicies::CMP0055))
|
||||||
|
{
|
||||||
|
case cmPolicies::WARN:
|
||||||
|
e << (this->Makefile->GetPolicies()
|
||||||
|
->GetPolicyWarning(cmPolicies::CMP0055)) << "\n";
|
||||||
|
break;
|
||||||
|
case cmPolicies::OLD:
|
||||||
|
issueMessage = false;
|
||||||
|
break;
|
||||||
|
case cmPolicies::REQUIRED_ALWAYS:
|
||||||
|
case cmPolicies::REQUIRED_IF_USED:
|
||||||
|
case cmPolicies::NEW:
|
||||||
|
messageType = cmake::FATAL_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(issueMessage)
|
||||||
|
{
|
||||||
|
e << "The BREAK command does not accept any arguments.";
|
||||||
|
this->Makefile->IssueMessage(messageType, e.str());
|
||||||
|
if(messageType == cmake::FATAL_ERROR)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,8 @@ 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)
|
||||||
{
|
{
|
||||||
|
cmMakefile::LoopBlockPop loopBlockPop(&mf);
|
||||||
|
|
||||||
// Remove the function blocker for this scope or bail.
|
// Remove the function blocker for this scope or bail.
|
||||||
cmsys::auto_ptr<cmFunctionBlocker>
|
cmsys::auto_ptr<cmFunctionBlocker>
|
||||||
fb(mf.RemoveFunctionBlocker(this, lff));
|
fb(mf.RemoveFunctionBlocker(this, lff));
|
||||||
|
@ -73,6 +75,7 @@ IsFunctionBlocked(const cmListFileFunction& lff, cmMakefile &mf,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// restore the variable to its prior value
|
// restore the variable to its prior value
|
||||||
mf.AddDefinition(this->Args[0],oldDef.c_str());
|
mf.AddDefinition(this->Args[0],oldDef.c_str());
|
||||||
return true;
|
return true;
|
||||||
|
@ -199,6 +202,8 @@ bool cmForEachCommand
|
||||||
}
|
}
|
||||||
this->Makefile->AddFunctionBlocker(f);
|
this->Makefile->AddFunctionBlocker(f);
|
||||||
|
|
||||||
|
this->Makefile->PushLoopBlock();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -242,5 +247,8 @@ bool cmForEachCommand::HandleInMode(std::vector<std::string> const& args)
|
||||||
}
|
}
|
||||||
|
|
||||||
this->Makefile->AddFunctionBlocker(f.release()); // TODO: pass auto_ptr
|
this->Makefile->AddFunctionBlocker(f.release()); // TODO: pass auto_ptr
|
||||||
|
|
||||||
|
this->Makefile->PushLoopBlock();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -171,6 +171,9 @@ void cmMakefile::Initialize()
|
||||||
// Protect the directory-level policies.
|
// Protect the directory-level policies.
|
||||||
this->PushPolicyBarrier();
|
this->PushPolicyBarrier();
|
||||||
|
|
||||||
|
// push empty loop block
|
||||||
|
this->PushLoopBlockBarrier();
|
||||||
|
|
||||||
// By default the check is not done. It is enabled by
|
// By default the check is not done. It is enabled by
|
||||||
// cmListFileCache in the top level if necessary.
|
// cmListFileCache in the top level if necessary.
|
||||||
this->CheckCMP0000 = false;
|
this->CheckCMP0000 = false;
|
||||||
|
@ -3353,6 +3356,38 @@ void cmMakefile::PopFunctionBlockerBarrier(bool reportError)
|
||||||
this->FunctionBlockerBarriers.pop_back();
|
this->FunctionBlockerBarriers.pop_back();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
void cmMakefile::PushLoopBlock()
|
||||||
|
{
|
||||||
|
assert(!this->LoopBlockCounter.empty());
|
||||||
|
this->LoopBlockCounter.top()++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmMakefile::PopLoopBlock()
|
||||||
|
{
|
||||||
|
assert(!this->LoopBlockCounter.empty());
|
||||||
|
assert(this->LoopBlockCounter.top() > 0);
|
||||||
|
this->LoopBlockCounter.top()--;
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmMakefile::PushLoopBlockBarrier()
|
||||||
|
{
|
||||||
|
this->LoopBlockCounter.push(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmMakefile::PopLoopBlockBarrier()
|
||||||
|
{
|
||||||
|
assert(!this->LoopBlockCounter.empty());
|
||||||
|
assert(this->LoopBlockCounter.top() == 0);
|
||||||
|
this->LoopBlockCounter.pop();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool cmMakefile::IsLoopBlock() const
|
||||||
|
{
|
||||||
|
assert(!this->LoopBlockCounter.empty());
|
||||||
|
return !this->LoopBlockCounter.empty() && this->LoopBlockCounter.top() > 0;
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
bool cmMakefile::ExpandArguments(
|
bool cmMakefile::ExpandArguments(
|
||||||
std::vector<cmListFileArgument> const& inArgs,
|
std::vector<cmListFileArgument> const& inArgs,
|
||||||
|
@ -4487,10 +4522,14 @@ void cmMakefile::PushScope()
|
||||||
this->Internal->VarStack.push(cmDefinitions(parent));
|
this->Internal->VarStack.push(cmDefinitions(parent));
|
||||||
this->Internal->VarInitStack.push(init);
|
this->Internal->VarInitStack.push(init);
|
||||||
this->Internal->VarUsageStack.push(usage);
|
this->Internal->VarUsageStack.push(usage);
|
||||||
|
|
||||||
|
this->PushLoopBlockBarrier();
|
||||||
}
|
}
|
||||||
|
|
||||||
void cmMakefile::PopScope()
|
void cmMakefile::PopScope()
|
||||||
{
|
{
|
||||||
|
this->PopLoopBlockBarrier();
|
||||||
|
|
||||||
cmDefinitions* current = &this->Internal->VarStack.top();
|
cmDefinitions* current = &this->Internal->VarStack.top();
|
||||||
std::set<std::string> init = this->Internal->VarInitStack.top();
|
std::set<std::string> init = this->Internal->VarInitStack.top();
|
||||||
std::set<std::string> usage = this->Internal->VarUsageStack.top();
|
std::set<std::string> usage = this->Internal->VarUsageStack.top();
|
||||||
|
|
|
@ -34,6 +34,8 @@
|
||||||
# include <cmsys/hash_map.hxx>
|
# include <cmsys/hash_map.hxx>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <stack>
|
||||||
|
|
||||||
class cmFunctionBlocker;
|
class cmFunctionBlocker;
|
||||||
class cmCommand;
|
class cmCommand;
|
||||||
class cmInstallGenerator;
|
class cmInstallGenerator;
|
||||||
|
@ -123,6 +125,15 @@ public:
|
||||||
};
|
};
|
||||||
friend class LexicalPushPop;
|
friend class LexicalPushPop;
|
||||||
|
|
||||||
|
class LoopBlockPop
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LoopBlockPop(cmMakefile* mf) { this->Makefile = mf; }
|
||||||
|
~LoopBlockPop() { this->Makefile->PopLoopBlock(); }
|
||||||
|
private:
|
||||||
|
cmMakefile* Makefile;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Try running cmake and building a file. This is used for dynalically
|
* Try running cmake and building a file. This is used for dynalically
|
||||||
* loaded commands, not as part of the usual build process.
|
* loaded commands, not as part of the usual build process.
|
||||||
|
@ -900,6 +911,10 @@ public:
|
||||||
void PopScope();
|
void PopScope();
|
||||||
void RaiseScope(const std::string& var, const char *value);
|
void RaiseScope(const std::string& var, const char *value);
|
||||||
|
|
||||||
|
// push and pop loop scopes
|
||||||
|
void PushLoopBlockBarrier();
|
||||||
|
void PopLoopBlockBarrier();
|
||||||
|
|
||||||
/** Helper class to push and pop scopes automatically. */
|
/** Helper class to push and pop scopes automatically. */
|
||||||
class ScopePushPop
|
class ScopePushPop
|
||||||
{
|
{
|
||||||
|
@ -960,6 +975,10 @@ public:
|
||||||
void ClearMatches();
|
void ClearMatches();
|
||||||
void StoreMatches(cmsys::RegularExpression& re);
|
void StoreMatches(cmsys::RegularExpression& re);
|
||||||
|
|
||||||
|
void PushLoopBlock();
|
||||||
|
void PopLoopBlock();
|
||||||
|
bool IsLoopBlock() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// add link libraries and directories to the target
|
// add link libraries and directories to the target
|
||||||
void AddGlobalLinkInformation(const std::string& name, cmTarget& target);
|
void AddGlobalLinkInformation(const std::string& name, cmTarget& target);
|
||||||
|
@ -1054,6 +1073,8 @@ private:
|
||||||
void PushFunctionBlockerBarrier();
|
void PushFunctionBlockerBarrier();
|
||||||
void PopFunctionBlockerBarrier(bool reportError = true);
|
void PopFunctionBlockerBarrier(bool reportError = true);
|
||||||
|
|
||||||
|
std::stack<int> LoopBlockCounter;
|
||||||
|
|
||||||
typedef std::map<std::string, std::string> StringStringMap;
|
typedef std::map<std::string, std::string> StringStringMap;
|
||||||
StringStringMap MacrosMap;
|
StringStringMap MacrosMap;
|
||||||
|
|
||||||
|
|
|
@ -364,6 +364,11 @@ cmPolicies::cmPolicies()
|
||||||
CMP0054, "CMP0054",
|
CMP0054, "CMP0054",
|
||||||
"Only interpret if() arguments as variables or keywords when unquoted.",
|
"Only interpret if() arguments as variables or keywords when unquoted.",
|
||||||
3,1,0, cmPolicies::WARN);
|
3,1,0, cmPolicies::WARN);
|
||||||
|
|
||||||
|
this->DefinePolicy(
|
||||||
|
CMP0055, "CMP0055",
|
||||||
|
"Strict checking for break() command.",
|
||||||
|
3,2,0, cmPolicies::WARN);
|
||||||
}
|
}
|
||||||
|
|
||||||
cmPolicies::~cmPolicies()
|
cmPolicies::~cmPolicies()
|
||||||
|
|
|
@ -111,6 +111,7 @@ public:
|
||||||
CMP0053, ///< Simplify variable reference and escape sequence evaluation
|
CMP0053, ///< Simplify variable reference and escape sequence evaluation
|
||||||
CMP0054, ///< Only interpret if() arguments as variables
|
CMP0054, ///< Only interpret if() arguments as variables
|
||||||
/// or keywords when unquoted.
|
/// or keywords when unquoted.
|
||||||
|
CMP0055, ///< Strict checking for break() command.
|
||||||
|
|
||||||
/** \brief Always the last entry.
|
/** \brief Always the last entry.
|
||||||
*
|
*
|
||||||
|
|
|
@ -27,6 +27,8 @@ 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)
|
||||||
{
|
{
|
||||||
|
cmMakefile::LoopBlockPop loopBlockPop(&mf);
|
||||||
|
|
||||||
// Remove the function blocker for this scope or bail.
|
// Remove the function blocker for this scope or bail.
|
||||||
cmsys::auto_ptr<cmFunctionBlocker>
|
cmsys::auto_ptr<cmFunctionBlocker>
|
||||||
fb(mf.RemoveFunctionBlocker(this, lff));
|
fb(mf.RemoveFunctionBlocker(this, lff));
|
||||||
|
@ -138,6 +140,8 @@ bool cmWhileCommand
|
||||||
f->Args = args;
|
f->Args = args;
|
||||||
this->Makefile->AddFunctionBlocker(f);
|
this->Makefile->AddFunctionBlocker(f);
|
||||||
|
|
||||||
|
this->Makefile->PushLoopBlock();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
1
|
|
@ -0,0 +1,4 @@
|
||||||
|
CMake Error at CMP0055-NEW-Out-of-Scope.cmake:4 \(break\):
|
||||||
|
A BREAK command was found outside of a proper FOREACH or WHILE loop scope.
|
||||||
|
Call Stack \(most recent call first\):
|
||||||
|
CMakeLists.txt:3 \(include\)
|
|
@ -0,0 +1,4 @@
|
||||||
|
|
||||||
|
cmake_policy(SET CMP0055 NEW)
|
||||||
|
|
||||||
|
break()
|
|
@ -0,0 +1 @@
|
||||||
|
1
|
|
@ -0,0 +1,4 @@
|
||||||
|
CMake Error at CMP0055-NEW-Reject-Arguments.cmake:5 \(break\):
|
||||||
|
The BREAK command does not accept any arguments.
|
||||||
|
Call Stack \(most recent call first\):
|
||||||
|
CMakeLists.txt:3 \(include\)
|
|
@ -0,0 +1,6 @@
|
||||||
|
|
||||||
|
cmake_policy(SET CMP0055 NEW)
|
||||||
|
|
||||||
|
foreach(i RANGE 1 2)
|
||||||
|
break(1)
|
||||||
|
endforeach()
|
|
@ -0,0 +1 @@
|
||||||
|
0
|
|
@ -0,0 +1 @@
|
||||||
|
^$
|
|
@ -0,0 +1,4 @@
|
||||||
|
|
||||||
|
cmake_policy(SET CMP0055 OLD)
|
||||||
|
|
||||||
|
break()
|
|
@ -0,0 +1 @@
|
||||||
|
0
|
|
@ -0,0 +1 @@
|
||||||
|
^$
|
|
@ -0,0 +1,6 @@
|
||||||
|
|
||||||
|
cmake_policy(SET CMP0055 OLD)
|
||||||
|
|
||||||
|
foreach(i RANGE 1 2)
|
||||||
|
break(1)
|
||||||
|
endforeach()
|
|
@ -0,0 +1 @@
|
||||||
|
0
|
|
@ -0,0 +1,9 @@
|
||||||
|
CMake Warning \(dev\) at CMP0055-WARN-Out-of-Scope.cmake:2 \(break\):
|
||||||
|
Policy CMP0055 is not set: Strict checking for break\(\) command. Run "cmake
|
||||||
|
--help-policy CMP0055" for policy details. Use the cmake_policy command to
|
||||||
|
set the policy and suppress this warning.
|
||||||
|
|
||||||
|
A BREAK command was found outside of a proper FOREACH or WHILE loop scope.
|
||||||
|
Call Stack \(most recent call first\):
|
||||||
|
CMakeLists.txt:3 \(include\)
|
||||||
|
This warning is for project developers. Use -Wno-dev to suppress it.
|
|
@ -0,0 +1,2 @@
|
||||||
|
|
||||||
|
break()
|
|
@ -0,0 +1 @@
|
||||||
|
0
|
|
@ -0,0 +1,9 @@
|
||||||
|
CMake Warning \(dev\) at CMP0055-WARN-Reject-Arguments.cmake:3 \(break\):
|
||||||
|
Policy CMP0055 is not set: Strict checking for break\(\) command. Run "cmake
|
||||||
|
--help-policy CMP0055" for policy details. Use the cmake_policy command to
|
||||||
|
set the policy and suppress this warning.
|
||||||
|
|
||||||
|
The BREAK command does not accept any arguments.
|
||||||
|
Call Stack \(most recent call first\):
|
||||||
|
CMakeLists.txt:3 \(include\)
|
||||||
|
This warning is for project developers. Use -Wno-dev to suppress it.
|
|
@ -0,0 +1,4 @@
|
||||||
|
|
||||||
|
foreach(i RANGE 1 2)
|
||||||
|
break(1)
|
||||||
|
endforeach()
|
|
@ -0,0 +1,3 @@
|
||||||
|
cmake_minimum_required(VERSION 3.1)
|
||||||
|
project(${RunCMake_TEST} NONE)
|
||||||
|
include(${RunCMake_TEST}.cmake)
|
|
@ -0,0 +1,9 @@
|
||||||
|
include(RunCMake)
|
||||||
|
|
||||||
|
run_cmake(CMP0055-OLD-Out-of-Scope)
|
||||||
|
run_cmake(CMP0055-NEW-Out-of-Scope)
|
||||||
|
run_cmake(CMP0055-WARN-Out-of-Scope)
|
||||||
|
|
||||||
|
run_cmake(CMP0055-OLD-Reject-Arguments)
|
||||||
|
run_cmake(CMP0055-NEW-Reject-Arguments)
|
||||||
|
run_cmake(CMP0055-WARN-Reject-Arguments)
|
|
@ -49,6 +49,7 @@ add_RunCMake_test(CMP0050)
|
||||||
add_RunCMake_test(CMP0051)
|
add_RunCMake_test(CMP0051)
|
||||||
add_RunCMake_test(CMP0053)
|
add_RunCMake_test(CMP0053)
|
||||||
add_RunCMake_test(CMP0054)
|
add_RunCMake_test(CMP0054)
|
||||||
|
add_RunCMake_test(CMP0055)
|
||||||
add_RunCMake_test(CTest)
|
add_RunCMake_test(CTest)
|
||||||
if(UNIX AND "${CMAKE_GENERATOR}" MATCHES "Unix Makefiles|Ninja")
|
if(UNIX AND "${CMAKE_GENERATOR}" MATCHES "Unix Makefiles|Ninja")
|
||||||
add_RunCMake_test(CompilerChange)
|
add_RunCMake_test(CompilerChange)
|
||||||
|
|
Loading…
Reference in New Issue