157 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* 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 <sstream>
 | |
| #include <stdio.h>
 | |
| 
 | |
| #include "cmMakefile.h"
 | |
| #include "cmSystemTools.h"
 | |
| #include "cmVersion.h"
 | |
| #include "cmake.h"
 | |
| 
 | |
| class cmExecutionStatus;
 | |
| 
 | |
| // cmCMakeMinimumRequired
 | |
| bool cmCMakeMinimumRequired::InitialPass(std::vector<std::string> const& args,
 | |
|                                          cmExecutionStatus&)
 | |
| {
 | |
|   // Process arguments.
 | |
|   std::string version_string;
 | |
|   bool doing_version = false;
 | |
|   for (std::string const& arg : args) {
 | |
|     if (arg == "VERSION") {
 | |
|       doing_version = true;
 | |
|     } else if (arg == "FATAL_ERROR") {
 | |
|       if (doing_version) {
 | |
|         this->SetError("called with no value for VERSION.");
 | |
|         return false;
 | |
|       }
 | |
|       doing_version = false;
 | |
|     } else if (doing_version) {
 | |
|       doing_version = false;
 | |
|       version_string = arg;
 | |
|     } else {
 | |
|       this->UnknownArguments.push_back(arg);
 | |
|     }
 | |
|   }
 | |
|   if (doing_version) {
 | |
|     this->SetError("called with no value for VERSION.");
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Make sure there was a version to check.
 | |
|   if (version_string.empty()) {
 | |
|     return this->EnforceUnknownArguments(std::string());
 | |
|   }
 | |
| 
 | |
|   // 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
 | |
|       << "\" does not have a version on both sides of \"...\".";
 | |
|     this->SetError(e.str());
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Save the required version string.
 | |
|   this->Makefile->AddDefinition("CMAKE_MINIMUM_REQUIRED_VERSION",
 | |
|                                 version_min.c_str());
 | |
| 
 | |
|   // 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 << "\".";
 | |
|     this->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();
 | |
|     this->Makefile->IssueMessage(cmake::FATAL_ERROR, e.str());
 | |
|     cmSystemTools::SetFatalErrorOccured();
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   // The version is not from the future, so enforce unknown arguments.
 | |
|   if (!this->EnforceUnknownArguments(version_max)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (required_major < 2 || (required_major == 2 && required_minor < 4)) {
 | |
|     this->Makefile->IssueMessage(
 | |
|       cmake::AUTHOR_WARNING,
 | |
|       "Compatibility with CMake < 2.4 is not supported by CMake >= 3.0.");
 | |
|     this->Makefile->SetPolicyVersion("2.4", version_max);
 | |
|   } else {
 | |
|     this->Makefile->SetPolicyVersion(version_min, version_max);
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool cmCMakeMinimumRequired::EnforceUnknownArguments(
 | |
|   std::string const& version_max)
 | |
| {
 | |
|   if (this->UnknownArguments.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 \"" << this->UnknownArguments[0] << "\".";
 | |
|   this->SetError(e.str());
 | |
|   return false;
 | |
| }
 |