- //===- VersionTuple.h - Version Number Handling -----------------*- C++ -*-===// 
- // 
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 
- // See https://llvm.org/LICENSE.txt for license information. 
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 
- // 
- //===----------------------------------------------------------------------===// 
- /// 
- /// \file 
- /// Defines the llvm::VersionTuple class, which represents a version in 
- /// the form major[.minor[.subminor]]. 
- /// 
- //===----------------------------------------------------------------------===// 
- #ifndef LLVM_SUPPORT_VERSIONTUPLE_H 
- #define LLVM_SUPPORT_VERSIONTUPLE_H 
-   
- #include "llvm/ADT/DenseMapInfo.h" 
- #include "llvm/ADT/Hashing.h" 
- #include "llvm/Support/Endian.h" 
- #include <optional> 
- #include <string> 
- #include <tuple> 
-   
- namespace llvm { 
- template <typename HasherT, support::endianness Endianness> 
- class HashBuilderImpl; 
- class raw_ostream; 
- class StringRef; 
-   
- /// Represents a version number in the form major[.minor[.subminor[.build]]]. 
- class VersionTuple { 
-   unsigned Major : 32; 
-   
-   unsigned Minor : 31; 
-   unsigned HasMinor : 1; 
-   
-   unsigned Subminor : 31; 
-   unsigned HasSubminor : 1; 
-   
-   unsigned Build : 31; 
-   unsigned HasBuild : 1; 
-   
- public: 
-   constexpr VersionTuple() 
-       : Major(0), Minor(0), HasMinor(false), Subminor(0), HasSubminor(false), 
-         Build(0), HasBuild(false) {} 
-   
-   explicit constexpr VersionTuple(unsigned Major) 
-       : Major(Major), Minor(0), HasMinor(false), Subminor(0), 
-         HasSubminor(false), Build(0), HasBuild(false) {} 
-   
-   explicit constexpr VersionTuple(unsigned Major, unsigned Minor) 
-       : Major(Major), Minor(Minor), HasMinor(true), Subminor(0), 
-         HasSubminor(false), Build(0), HasBuild(false) {} 
-   
-   explicit constexpr VersionTuple(unsigned Major, unsigned Minor, 
-                                   unsigned Subminor) 
-       : Major(Major), Minor(Minor), HasMinor(true), Subminor(Subminor), 
-         HasSubminor(true), Build(0), HasBuild(false) {} 
-   
-   explicit constexpr VersionTuple(unsigned Major, unsigned Minor, 
-                                   unsigned Subminor, unsigned Build) 
-       : Major(Major), Minor(Minor), HasMinor(true), Subminor(Subminor), 
-         HasSubminor(true), Build(Build), HasBuild(true) {} 
-   
-   /// Determine whether this version information is empty 
-   /// (e.g., all version components are zero). 
-   bool empty() const { 
-     return Major == 0 && Minor == 0 && Subminor == 0 && Build == 0; 
-   } 
-   
-   /// Retrieve the major version number. 
-   unsigned getMajor() const { return Major; } 
-   
-   /// Retrieve the minor version number, if provided. 
-   std::optional<unsigned> getMinor() const { 
-     if (!HasMinor) 
-       return std::nullopt; 
-     return Minor; 
-   } 
-   
-   /// Retrieve the subminor version number, if provided. 
-   std::optional<unsigned> getSubminor() const { 
-     if (!HasSubminor) 
-       return std::nullopt; 
-     return Subminor; 
-   } 
-   
-   /// Retrieve the build version number, if provided. 
-   std::optional<unsigned> getBuild() const { 
-     if (!HasBuild) 
-       return std::nullopt; 
-     return Build; 
-   } 
-   
-   /// Return a version tuple that contains only the first 3 version components. 
-   VersionTuple withoutBuild() const { 
-     if (HasBuild) 
-       return VersionTuple(Major, Minor, Subminor); 
-     return *this; 
-   } 
-   
-   /// Return a version tuple that contains a different major version but 
-   /// everything else is the same. 
-   VersionTuple withMajorReplaced(unsigned NewMajor) const { 
-     return VersionTuple(NewMajor, Minor, Subminor, Build); 
-   } 
-   
-   /// Return a version tuple that contains only components that are non-zero. 
-   VersionTuple normalize() const { 
-     VersionTuple Result = *this; 
-     if (Result.Build == 0) { 
-       Result.HasBuild = false; 
-       if (Result.Subminor == 0) { 
-         Result.HasSubminor = false; 
-         if (Result.Minor == 0) 
-           Result.HasMinor = false; 
-       } 
-     } 
-     return Result; 
-   } 
-   
-   /// Determine if two version numbers are equivalent. If not 
-   /// provided, minor and subminor version numbers are considered to be zero. 
-   friend bool operator==(const VersionTuple &X, const VersionTuple &Y) { 
-     return X.Major == Y.Major && X.Minor == Y.Minor && 
-            X.Subminor == Y.Subminor && X.Build == Y.Build; 
-   } 
-   
-   /// Determine if two version numbers are not equivalent. 
-   /// 
-   /// If not provided, minor and subminor version numbers are considered to be 
-   /// zero. 
-   friend bool operator!=(const VersionTuple &X, const VersionTuple &Y) { 
-     return !(X == Y); 
-   } 
-   
-   /// Determine whether one version number precedes another. 
-   /// 
-   /// If not provided, minor and subminor version numbers are considered to be 
-   /// zero. 
-   friend bool operator<(const VersionTuple &X, const VersionTuple &Y) { 
-     return std::tie(X.Major, X.Minor, X.Subminor, X.Build) < 
-            std::tie(Y.Major, Y.Minor, Y.Subminor, Y.Build); 
-   } 
-   
-   /// Determine whether one version number follows another. 
-   /// 
-   /// If not provided, minor and subminor version numbers are considered to be 
-   /// zero. 
-   friend bool operator>(const VersionTuple &X, const VersionTuple &Y) { 
-     return Y < X; 
-   } 
-   
-   /// Determine whether one version number precedes or is 
-   /// equivalent to another. 
-   /// 
-   /// If not provided, minor and subminor version numbers are considered to be 
-   /// zero. 
-   friend bool operator<=(const VersionTuple &X, const VersionTuple &Y) { 
-     return !(Y < X); 
-   } 
-   
-   /// Determine whether one version number follows or is 
-   /// equivalent to another. 
-   /// 
-   /// If not provided, minor and subminor version numbers are considered to be 
-   /// zero. 
-   friend bool operator>=(const VersionTuple &X, const VersionTuple &Y) { 
-     return !(X < Y); 
-   } 
-   
-   friend hash_code hash_value(const VersionTuple &VT) { 
-     return hash_combine(VT.Major, VT.Minor, VT.Subminor, VT.Build); 
-   } 
-   
-   template <typename HasherT, llvm::support::endianness Endianness> 
-   friend void addHash(HashBuilderImpl<HasherT, Endianness> &HBuilder, 
-                       const VersionTuple &VT) { 
-     HBuilder.add(VT.Major, VT.Minor, VT.Subminor, VT.Build); 
-   } 
-   
-   /// Retrieve a string representation of the version number. 
-   std::string getAsString() const; 
-   
-   /// Try to parse the given string as a version number. 
-   /// \returns \c true if the string does not match the regular expression 
-   ///   [0-9]+(\.[0-9]+){0,3} 
-   bool tryParse(StringRef string); 
- }; 
-   
- /// Print a version number. 
- raw_ostream &operator<<(raw_ostream &Out, const VersionTuple &V); 
-   
- // Provide DenseMapInfo for version tuples. 
- template <> struct DenseMapInfo<VersionTuple> { 
-   static inline VersionTuple getEmptyKey() { return VersionTuple(0x7FFFFFFF); } 
-   static inline VersionTuple getTombstoneKey() { 
-     return VersionTuple(0x7FFFFFFE); 
-   } 
-   static unsigned getHashValue(const VersionTuple &Value) { 
-     unsigned Result = Value.getMajor(); 
-     if (auto Minor = Value.getMinor()) 
-       Result = detail::combineHashValue(Result, *Minor); 
-     if (auto Subminor = Value.getSubminor()) 
-       Result = detail::combineHashValue(Result, *Subminor); 
-     if (auto Build = Value.getBuild()) 
-       Result = detail::combineHashValue(Result, *Build); 
-   
-     return Result; 
-   } 
-   
-   static bool isEqual(const VersionTuple &LHS, const VersionTuple &RHS) { 
-     return LHS == RHS; 
-   } 
- }; 
-   
- } // end namespace llvm 
- #endif // LLVM_SUPPORT_VERSIONTUPLE_H 
-