|
| 1 | +//===- ComparisonCategories.h - Three Way Comparison Data -------*- C++ -*-===// |
| 2 | +// |
| 3 | +// The LLVM Compiler Infrastructure |
| 4 | +// |
| 5 | +// This file is distributed under the University of Illinois Open Source |
| 6 | +// License. See LICENSE.TXT for details. |
| 7 | +// |
| 8 | +//===----------------------------------------------------------------------===// |
| 9 | +// |
| 10 | +// This file defines the Comparison Category enum and data types, which |
| 11 | +// store the types and expressions needed to support operator<=> |
| 12 | +// |
| 13 | +//===----------------------------------------------------------------------===// |
| 14 | + |
| 15 | +#ifndef LLVM_CLANG_AST_COMPARISONCATEGORIES_H |
| 16 | +#define LLVM_CLANG_AST_COMPARISONCATEGORIES_H |
| 17 | + |
| 18 | +#include "clang/Basic/LLVM.h" |
| 19 | +#include "llvm/ADT/APSInt.h" |
| 20 | +#include "llvm/ADT/DenseMap.h" |
| 21 | +#include <array> |
| 22 | +#include <cassert> |
| 23 | + |
| 24 | +namespace llvm { |
| 25 | + class StringRef; |
| 26 | + class APSInt; |
| 27 | +} |
| 28 | + |
| 29 | +namespace clang { |
| 30 | + |
| 31 | +class ASTContext; |
| 32 | +class VarDecl; |
| 33 | +class CXXRecordDecl; |
| 34 | +class Sema; |
| 35 | +class QualType; |
| 36 | +class NamespaceDecl; |
| 37 | + |
| 38 | +/// \brief An enumeration representing the different comparison categories |
| 39 | +/// types. |
| 40 | +/// |
| 41 | +/// C++2a [cmp.categories.pre] The types weak_equality, strong_equality, |
| 42 | +/// partial_ordering, weak_ordering, and strong_ordering are collectively |
| 43 | +/// termed the comparison category types. |
| 44 | +enum class ComparisonCategoryType : unsigned char { |
| 45 | + WeakEquality, |
| 46 | + StrongEquality, |
| 47 | + PartialOrdering, |
| 48 | + WeakOrdering, |
| 49 | + StrongOrdering, |
| 50 | + First = WeakEquality, |
| 51 | + Last = StrongOrdering |
| 52 | +}; |
| 53 | + |
| 54 | +/// \brief An enumeration representing the possible results of a three-way |
| 55 | +/// comparison. These values map onto instances of comparison category types |
| 56 | +/// defined in the standard library. i.e. 'std::strong_ordering::less'. |
| 57 | +enum class ComparisonCategoryResult : unsigned char { |
| 58 | + Equal, |
| 59 | + Equivalent, |
| 60 | + Nonequivalent, |
| 61 | + Nonequal, |
| 62 | + Less, |
| 63 | + Greater, |
| 64 | + Unordered, |
| 65 | + Last = Unordered |
| 66 | +}; |
| 67 | + |
| 68 | +class ComparisonCategoryInfo { |
| 69 | + friend class ComparisonCategories; |
| 70 | + friend class Sema; |
| 71 | + |
| 72 | +public: |
| 73 | + ComparisonCategoryInfo(const ASTContext &Ctx, CXXRecordDecl *RD, |
| 74 | + ComparisonCategoryType Kind) |
| 75 | + : Ctx(Ctx), Record(RD), Kind(Kind) {} |
| 76 | + |
| 77 | + struct ValueInfo { |
| 78 | + ComparisonCategoryResult Kind; |
| 79 | + VarDecl *VD; |
| 80 | + |
| 81 | + ValueInfo(ComparisonCategoryResult Kind, VarDecl *VD) |
| 82 | + : Kind(Kind), VD(VD), HasValue(false) {} |
| 83 | + |
| 84 | + /// \brief True iff we've successfully evaluated the variable as a constant |
| 85 | + /// expression and extracted its integer value. |
| 86 | + bool hasValidIntValue() const { return HasValue; } |
| 87 | + |
| 88 | + /// \brief Get the constant integer value used by this variable to represent |
| 89 | + /// the comparison category result type. |
| 90 | + llvm::APSInt getIntValue() const { |
| 91 | + assert(hasValidIntValue()); |
| 92 | + return IntValue; |
| 93 | + } |
| 94 | + |
| 95 | + void setIntValue(llvm::APSInt Val) { |
| 96 | + IntValue = Val; |
| 97 | + HasValue = true; |
| 98 | + } |
| 99 | + |
| 100 | + private: |
| 101 | + friend class ComparisonCategoryInfo; |
| 102 | + llvm::APSInt IntValue; |
| 103 | + bool HasValue : 1; |
| 104 | + }; |
| 105 | +private: |
| 106 | + const ASTContext &Ctx; |
| 107 | + |
| 108 | + /// \brief A map containing the comparison category result decls from the |
| 109 | + /// standard library. The key is a value of ComparisonCategoryResult. |
| 110 | + mutable llvm::SmallVector< |
| 111 | + ValueInfo, static_cast<unsigned>(ComparisonCategoryResult::Last) + 1> |
| 112 | + Objects; |
| 113 | + |
| 114 | + /// \brief Lookup the ValueInfo struct for the specified ValueKind. If the |
| 115 | + /// VarDecl for the value cannot be found, nullptr is returned. |
| 116 | + /// |
| 117 | + /// If the ValueInfo does not have a valid integer value the variable |
| 118 | + /// is evaluated as a constant expression to determine that value. |
| 119 | + ValueInfo *lookupValueInfo(ComparisonCategoryResult ValueKind) const; |
| 120 | + |
| 121 | +public: |
| 122 | + /// \brief The declaration for the comparison category type from the |
| 123 | + /// standard library. |
| 124 | + // FIXME: Make this const |
| 125 | + CXXRecordDecl *Record = nullptr; |
| 126 | + |
| 127 | + /// \brief The Kind of the comparison category type |
| 128 | + ComparisonCategoryType Kind; |
| 129 | + |
| 130 | +public: |
| 131 | + QualType getType() const; |
| 132 | + |
| 133 | + const ValueInfo *getValueInfo(ComparisonCategoryResult ValueKind) const { |
| 134 | + ValueInfo *Info = lookupValueInfo(ValueKind); |
| 135 | + assert(Info && |
| 136 | + "comparison category does not contain the specified result kind"); |
| 137 | + assert(Info->hasValidIntValue() && |
| 138 | + "couldn't determine the integer constant for this value"); |
| 139 | + return Info; |
| 140 | + } |
| 141 | + |
| 142 | + /// \brief True iff the comparison category is an equality comparison. |
| 143 | + bool isEquality() const { return !isOrdered(); } |
| 144 | + |
| 145 | + /// \brief True iff the comparison category is a relational comparison. |
| 146 | + bool isOrdered() const { |
| 147 | + using CCK = ComparisonCategoryType; |
| 148 | + return Kind == CCK::PartialOrdering || Kind == CCK::WeakOrdering || |
| 149 | + Kind == CCK::StrongOrdering; |
| 150 | + } |
| 151 | + |
| 152 | + /// \brief True iff the comparison is "strong". i.e. it checks equality and |
| 153 | + /// not equivalence. |
| 154 | + bool isStrong() const { |
| 155 | + using CCK = ComparisonCategoryType; |
| 156 | + return Kind == CCK::StrongEquality || Kind == CCK::StrongOrdering; |
| 157 | + } |
| 158 | + |
| 159 | + /// \brief True iff the comparison is not totally ordered. |
| 160 | + bool isPartial() const { |
| 161 | + using CCK = ComparisonCategoryType; |
| 162 | + return Kind == CCK::PartialOrdering; |
| 163 | + } |
| 164 | + |
| 165 | + /// \brief Converts the specified result kind into the the correct result kind |
| 166 | + /// for this category. Specifically it lowers strong equality results to |
| 167 | + /// weak equivalence if needed. |
| 168 | + ComparisonCategoryResult makeWeakResult(ComparisonCategoryResult Res) const { |
| 169 | + using CCR = ComparisonCategoryResult; |
| 170 | + if (!isStrong()) { |
| 171 | + if (Res == CCR::Equal) |
| 172 | + return CCR::Equivalent; |
| 173 | + if (Res == CCR::Nonequal) |
| 174 | + return CCR::Nonequivalent; |
| 175 | + } |
| 176 | + return Res; |
| 177 | + } |
| 178 | + |
| 179 | + const ValueInfo *getEqualOrEquiv() const { |
| 180 | + return getValueInfo(makeWeakResult(ComparisonCategoryResult::Equal)); |
| 181 | + } |
| 182 | + const ValueInfo *getNonequalOrNonequiv() const { |
| 183 | + assert(isEquality()); |
| 184 | + return getValueInfo(makeWeakResult(ComparisonCategoryResult::Nonequal)); |
| 185 | + } |
| 186 | + const ValueInfo *getLess() const { |
| 187 | + assert(isOrdered()); |
| 188 | + return getValueInfo(ComparisonCategoryResult::Less); |
| 189 | + } |
| 190 | + const ValueInfo *getGreater() const { |
| 191 | + assert(isOrdered()); |
| 192 | + return getValueInfo(ComparisonCategoryResult::Greater); |
| 193 | + } |
| 194 | + const ValueInfo *getUnordered() const { |
| 195 | + assert(isPartial()); |
| 196 | + return getValueInfo(ComparisonCategoryResult::Unordered); |
| 197 | + } |
| 198 | +}; |
| 199 | + |
| 200 | +class ComparisonCategories { |
| 201 | +public: |
| 202 | + static StringRef getCategoryString(ComparisonCategoryType Kind); |
| 203 | + static StringRef getResultString(ComparisonCategoryResult Kind); |
| 204 | + |
| 205 | + /// \brief Return the list of results which are valid for the specified |
| 206 | + /// comparison category type. |
| 207 | + static std::vector<ComparisonCategoryResult> |
| 208 | + getPossibleResultsForType(ComparisonCategoryType Type); |
| 209 | + |
| 210 | + /// \brief Return the comparison category information for the category |
| 211 | + /// specified by 'Kind'. |
| 212 | + const ComparisonCategoryInfo &getInfo(ComparisonCategoryType Kind) const { |
| 213 | + const ComparisonCategoryInfo *Result = lookupInfo(Kind); |
| 214 | + assert(Result != nullptr && |
| 215 | + "information for specified comparison category has not been built"); |
| 216 | + return *Result; |
| 217 | + } |
| 218 | + |
| 219 | + /// \brief Return the comparison category information as specified by |
| 220 | + /// `getCategoryForType(Ty)`. If the information is not already cached, |
| 221 | + /// the declaration is looked up and a cache entry is created. |
| 222 | + /// NOTE: Lookup is expected to succeed. Use lookupInfo if failure is possible. |
| 223 | + const ComparisonCategoryInfo &getInfoForType(QualType Ty) const; |
| 224 | + |
| 225 | +public: |
| 226 | + /// \brief Return the cached comparison category information for the |
| 227 | + /// specified 'Kind'. If no cache entry is present the comparison category |
| 228 | + /// type is looked up. If lookup fails nullptr is returned. Otherwise, a |
| 229 | + /// new cache entry is created and returned |
| 230 | + const ComparisonCategoryInfo *lookupInfo(ComparisonCategoryType Kind) const; |
| 231 | + |
| 232 | + ComparisonCategoryInfo *lookupInfo(ComparisonCategoryType Kind) { |
| 233 | + const auto &This = *this; |
| 234 | + return const_cast<ComparisonCategoryInfo *>(This.lookupInfo(Kind)); |
| 235 | + } |
| 236 | + |
| 237 | +private: |
| 238 | + const ComparisonCategoryInfo *lookupInfoForType(QualType Ty) const; |
| 239 | + |
| 240 | +private: |
| 241 | + friend class ASTContext; |
| 242 | + |
| 243 | + explicit ComparisonCategories(const ASTContext &Ctx) : Ctx(Ctx) {} |
| 244 | + |
| 245 | + const ASTContext &Ctx; |
| 246 | + |
| 247 | + /// A map from the ComparisonCategoryType (represented as 'char') to the |
| 248 | + /// cached information for the specified category. |
| 249 | + mutable llvm::DenseMap<char, ComparisonCategoryInfo> Data; |
| 250 | + mutable NamespaceDecl *StdNS = nullptr; |
| 251 | +}; |
| 252 | + |
| 253 | +} // namespace clang |
| 254 | + |
| 255 | +#endif |
0 commit comments