226 lines
6.8 KiB
226 lines
6.8 KiB
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
#include "cmCMakePolicyCommand.h"
|
|
|
|
#include "cmExecutionStatus.h"
|
|
#include "cmMakefile.h"
|
|
#include "cmMessageType.h"
|
|
#include "cmPolicies.h"
|
|
#include "cmState.h"
|
|
#include "cmStateTypes.h"
|
|
#include "cmStringAlgorithms.h"
|
|
#include "cmValue.h"
|
|
|
|
namespace {
|
|
bool HandleSetMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status);
|
|
bool HandleGetMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status);
|
|
bool HandleVersionMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status);
|
|
bool HandleGetWarningMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status);
|
|
}
|
|
|
|
// cmCMakePolicyCommand
|
|
bool cmCMakePolicyCommand(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status)
|
|
{
|
|
if (args.empty()) {
|
|
status.SetError("requires at least one argument.");
|
|
return false;
|
|
}
|
|
|
|
if (args[0] == "SET") {
|
|
return HandleSetMode(args, status);
|
|
}
|
|
if (args[0] == "GET") {
|
|
return HandleGetMode(args, status);
|
|
}
|
|
if (args[0] == "PUSH") {
|
|
if (args.size() > 1) {
|
|
status.SetError("PUSH may not be given additional arguments.");
|
|
return false;
|
|
}
|
|
status.GetMakefile().PushPolicy();
|
|
return true;
|
|
}
|
|
if (args[0] == "POP") {
|
|
if (args.size() > 1) {
|
|
status.SetError("POP may not be given additional arguments.");
|
|
return false;
|
|
}
|
|
status.GetMakefile().PopPolicy();
|
|
return true;
|
|
}
|
|
if (args[0] == "VERSION") {
|
|
return HandleVersionMode(args, status);
|
|
}
|
|
if (args[0] == "GET_WARNING") {
|
|
return HandleGetWarningMode(args, status);
|
|
}
|
|
|
|
status.SetError(cmStrCat("given unknown first argument \"", args[0], "\""));
|
|
return false;
|
|
}
|
|
|
|
namespace {
|
|
|
|
bool HandleSetMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status)
|
|
{
|
|
if (args.size() != 3) {
|
|
status.SetError("SET must be given exactly 2 additional arguments.");
|
|
return false;
|
|
}
|
|
|
|
cmPolicies::PolicyStatus policyStatus;
|
|
if (args[2] == "OLD") {
|
|
policyStatus = cmPolicies::OLD;
|
|
} else if (args[2] == "NEW") {
|
|
policyStatus = cmPolicies::NEW;
|
|
} else {
|
|
status.SetError(
|
|
cmStrCat("SET given unrecognized policy status \"", args[2], "\""));
|
|
return false;
|
|
}
|
|
|
|
if (!status.GetMakefile().SetPolicy(args[1].c_str(), policyStatus)) {
|
|
status.SetError("SET failed to set policy.");
|
|
return false;
|
|
}
|
|
if (args[1] == "CMP0001" &&
|
|
(policyStatus == cmPolicies::WARN || policyStatus == cmPolicies::OLD)) {
|
|
if (!(status.GetMakefile().GetState()->GetInitializedCacheValue(
|
|
"CMAKE_BACKWARDS_COMPATIBILITY"))) {
|
|
// Set it to 2.4 because that is the last version where the
|
|
// variable had meaning.
|
|
status.GetMakefile().AddCacheDefinition(
|
|
"CMAKE_BACKWARDS_COMPATIBILITY", "2.4",
|
|
"For backwards compatibility, what version of CMake "
|
|
"commands and "
|
|
"syntax should this version of CMake try to support.",
|
|
cmStateEnums::STRING);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool HandleGetMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status)
|
|
{
|
|
bool parent_scope = false;
|
|
if (args.size() == 4 && args[3] == "PARENT_SCOPE") {
|
|
// Undocumented PARENT_SCOPE option for use within CMake.
|
|
parent_scope = true;
|
|
} else if (args.size() != 3) {
|
|
status.SetError("GET must be given exactly 2 additional arguments.");
|
|
return false;
|
|
}
|
|
|
|
// Get arguments.
|
|
std::string const& id = args[1];
|
|
std::string const& var = args[2];
|
|
|
|
// Lookup the policy number.
|
|
cmPolicies::PolicyID pid;
|
|
if (!cmPolicies::GetPolicyID(id.c_str(), pid)) {
|
|
status.SetError(
|
|
cmStrCat("GET given policy \"", id,
|
|
"\" which is not known to this version of CMake."));
|
|
return false;
|
|
}
|
|
|
|
// Lookup the policy setting.
|
|
cmPolicies::PolicyStatus policyStatus =
|
|
status.GetMakefile().GetPolicyStatus(pid, parent_scope);
|
|
switch (policyStatus) {
|
|
case cmPolicies::OLD:
|
|
// Report that the policy is set to OLD.
|
|
status.GetMakefile().AddDefinition(var, "OLD");
|
|
break;
|
|
case cmPolicies::WARN:
|
|
// Report that the policy is not set.
|
|
status.GetMakefile().AddDefinition(var, "");
|
|
break;
|
|
case cmPolicies::NEW:
|
|
// Report that the policy is set to NEW.
|
|
status.GetMakefile().AddDefinition(var, "NEW");
|
|
break;
|
|
case cmPolicies::REQUIRED_IF_USED:
|
|
case cmPolicies::REQUIRED_ALWAYS:
|
|
// The policy is required to be set before anything needs it.
|
|
{
|
|
status.GetMakefile().IssueMessage(
|
|
MessageType::FATAL_ERROR,
|
|
cmStrCat(
|
|
cmPolicies::GetRequiredPolicyError(pid), "\n",
|
|
"The call to cmake_policy(GET ", id,
|
|
" ...) at which this "
|
|
"error appears requests the policy, and this version of CMake ",
|
|
"requires that the policy be set to NEW before it is checked."));
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool HandleVersionMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status)
|
|
{
|
|
if (args.size() <= 1) {
|
|
status.SetError("VERSION not given an argument");
|
|
return false;
|
|
}
|
|
if (args.size() >= 3) {
|
|
status.SetError("VERSION given too many arguments");
|
|
return false;
|
|
}
|
|
std::string const& version_string = args[1];
|
|
|
|
// Separate the <min> version and any trailing ...<max> component.
|
|
std::string::size_type const dd = version_string.find("...");
|
|
std::string const version_min = version_string.substr(0, dd);
|
|
std::string const version_max = dd != std::string::npos
|
|
? version_string.substr(dd + 3, std::string::npos)
|
|
: std::string();
|
|
if (dd != std::string::npos &&
|
|
(version_min.empty() || version_max.empty())) {
|
|
status.SetError(
|
|
cmStrCat("VERSION \"", version_string,
|
|
R"(" does not have a version on both sides of "...".)"));
|
|
return false;
|
|
}
|
|
|
|
return status.GetMakefile().SetPolicyVersion(version_min, version_max);
|
|
}
|
|
|
|
bool HandleGetWarningMode(std::vector<std::string> const& args,
|
|
cmExecutionStatus& status)
|
|
{
|
|
if (args.size() != 3) {
|
|
status.SetError(
|
|
"GET_WARNING must be given exactly 2 additional arguments.");
|
|
return false;
|
|
}
|
|
|
|
// Get arguments.
|
|
std::string const& id = args[1];
|
|
std::string const& var = args[2];
|
|
|
|
// Lookup the policy number.
|
|
cmPolicies::PolicyID pid;
|
|
if (!cmPolicies::GetPolicyID(id.c_str(), pid)) {
|
|
status.SetError(
|
|
cmStrCat("GET_WARNING given policy \"", id,
|
|
"\" which is not known to this version of CMake."));
|
|
return false;
|
|
}
|
|
|
|
// Lookup the policy warning.
|
|
status.GetMakefile().AddDefinition(var, cmPolicies::GetPolicyWarning(pid));
|
|
|
|
return true;
|
|
}
|
|
}
|