/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #include "cmCMakeMinimumRequired.h" #include <cstdio> #include <sstream> #include "cmExecutionStatus.h" #include "cmMakefile.h" #include "cmMessageType.h" #include "cmSystemTools.h" #include "cmVersion.h" namespace { bool EnforceUnknownArguments(std::string const& version_max, std::vector<std::string> const& unknown_arguments, cmExecutionStatus& status); } // cmCMakeMinimumRequired bool cmCMakeMinimumRequired(std::vector<std::string> const& args, cmExecutionStatus& status) { // Process arguments. std::string version_string; bool doing_version = false; std::vector<std::string> unknown_arguments; for (std::string const& arg : args) { if (arg == "VERSION") { doing_version = true; } else if (arg == "FATAL_ERROR") { if (doing_version) { status.SetError("called with no value for VERSION."); return false; } doing_version = false; } else if (doing_version) { doing_version = false; version_string = arg; } else { unknown_arguments.push_back(arg); } } if (doing_version) { status.SetError("called with no value for VERSION."); return false; } // Make sure there was a version to check. if (version_string.empty()) { return EnforceUnknownArguments(std::string(), unknown_arguments, status); } // 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())) { std::ostringstream e; e << "VERSION \"" << version_string << R"(" does not have a version on both sides of "...".)"; status.SetError(e.str()); return false; } // Save the required version string. status.GetMakefile().AddDefinition("CMAKE_MINIMUM_REQUIRED_VERSION", version_min); // Get the current version number. unsigned int current_major = cmVersion::GetMajorVersion(); unsigned int current_minor = cmVersion::GetMinorVersion(); unsigned int current_patch = cmVersion::GetPatchVersion(); unsigned int current_tweak = cmVersion::GetTweakVersion(); // Parse at least two components of the version number. // Use zero for those not specified. unsigned int required_major = 0; unsigned int required_minor = 0; unsigned int required_patch = 0; unsigned int required_tweak = 0; if (sscanf(version_min.c_str(), "%u.%u.%u.%u", &required_major, &required_minor, &required_patch, &required_tweak) < 2) { std::ostringstream e; e << "could not parse VERSION \"" << version_min << "\"."; status.SetError(e.str()); return false; } // Compare the version numbers. if ((current_major < required_major) || (current_major == required_major && current_minor < required_minor) || (current_major == required_major && current_minor == required_minor && current_patch < required_patch) || (current_major == required_major && current_minor == required_minor && current_patch == required_patch && current_tweak < required_tweak)) { // The current version is too low. std::ostringstream e; e << "CMake " << version_min << " or higher is required. You are running version " << cmVersion::GetCMakeVersion(); status.GetMakefile().IssueMessage(MessageType::FATAL_ERROR, e.str()); cmSystemTools::SetFatalErrorOccured(); return true; } // The version is not from the future, so enforce unknown arguments. if (!EnforceUnknownArguments(version_max, unknown_arguments, status)) { return false; } if (required_major < 2 || (required_major == 2 && required_minor < 4)) { status.GetMakefile().IssueMessage( MessageType::AUTHOR_WARNING, "Compatibility with CMake < 2.4 is not supported by CMake >= 3.0."); status.GetMakefile().SetPolicyVersion("2.4", version_max); } else { status.GetMakefile().SetPolicyVersion(version_min, version_max); } return true; } namespace { bool EnforceUnknownArguments(std::string const& version_max, std::vector<std::string> const& unknown_arguments, cmExecutionStatus& status) { if (unknown_arguments.empty()) { return true; } // Consider the max version if at least two components were given. unsigned int max_major = 0; unsigned int max_minor = 0; unsigned int max_patch = 0; unsigned int max_tweak = 0; if (sscanf(version_max.c_str(), "%u.%u.%u.%u", &max_major, &max_minor, &max_patch, &max_tweak) >= 2) { unsigned int current_major = cmVersion::GetMajorVersion(); unsigned int current_minor = cmVersion::GetMinorVersion(); unsigned int current_patch = cmVersion::GetPatchVersion(); unsigned int current_tweak = cmVersion::GetTweakVersion(); if ((current_major < max_major) || (current_major == max_major && current_minor < max_minor) || (current_major == max_major && current_minor == max_minor && current_patch < max_patch) || (current_major == max_major && current_minor == max_minor && current_patch == max_patch && current_tweak < max_tweak)) { // A ...<max> version was given that is larger than the current // version of CMake, so tolerate unknown arguments. return true; } } std::ostringstream e; e << "called with unknown argument \"" << unknown_arguments[0] << "\"."; status.SetError(e.str()); return false; } }