|  | //===--- VariantValue.cpp - Polymorphic value type -*- 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 | 
|  | /// Polymorphic value type. | 
|  | /// | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "clang/ASTMatchers/Dynamic/VariantValue.h" | 
|  | #include "clang/Basic/LLVM.h" | 
|  | #include "llvm/ADT/STLExtras.h" | 
|  |  | 
|  | namespace clang { | 
|  | namespace ast_matchers { | 
|  | namespace dynamic { | 
|  |  | 
|  | std::string ArgKind::asString() const { | 
|  | switch (getArgKind()) { | 
|  | case AK_Matcher: | 
|  | return (Twine("Matcher<") + MatcherKind.asStringRef() + ">").str(); | 
|  | case AK_Boolean: | 
|  | return "boolean"; | 
|  | case AK_Double: | 
|  | return "double"; | 
|  | case AK_Unsigned: | 
|  | return "unsigned"; | 
|  | case AK_String: | 
|  | return "string"; | 
|  | } | 
|  | llvm_unreachable("unhandled ArgKind"); | 
|  | } | 
|  |  | 
|  | bool ArgKind::isConvertibleTo(ArgKind To, unsigned *Specificity) const { | 
|  | if (K != To.K) | 
|  | return false; | 
|  | if (K != AK_Matcher) { | 
|  | if (Specificity) | 
|  | *Specificity = 1; | 
|  | return true; | 
|  | } | 
|  | unsigned Distance; | 
|  | if (!MatcherKind.isBaseOf(To.MatcherKind, &Distance)) | 
|  | return false; | 
|  |  | 
|  | if (Specificity) | 
|  | *Specificity = 100 - Distance; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool | 
|  | VariantMatcher::MatcherOps::canConstructFrom(const DynTypedMatcher &Matcher, | 
|  | bool &IsExactMatch) const { | 
|  | IsExactMatch = Matcher.getSupportedKind().isSame(NodeKind); | 
|  | return Matcher.canConvertTo(NodeKind); | 
|  | } | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> | 
|  | VariantMatcher::MatcherOps::constructVariadicOperator( | 
|  | DynTypedMatcher::VariadicOperator Op, | 
|  | ArrayRef<VariantMatcher> InnerMatchers) const { | 
|  | std::vector<DynTypedMatcher> DynMatchers; | 
|  | for (const auto &InnerMatcher : InnerMatchers) { | 
|  | // Abort if any of the inner matchers can't be converted to | 
|  | // Matcher<T>. | 
|  | if (!InnerMatcher.Value) | 
|  | return llvm::None; | 
|  | llvm::Optional<DynTypedMatcher> Inner = | 
|  | InnerMatcher.Value->getTypedMatcher(*this); | 
|  | if (!Inner) | 
|  | return llvm::None; | 
|  | DynMatchers.push_back(*Inner); | 
|  | } | 
|  | return DynTypedMatcher::constructVariadic(Op, NodeKind, DynMatchers); | 
|  | } | 
|  |  | 
|  | VariantMatcher::Payload::~Payload() {} | 
|  |  | 
|  | class VariantMatcher::SinglePayload : public VariantMatcher::Payload { | 
|  | public: | 
|  | SinglePayload(const DynTypedMatcher &Matcher) : Matcher(Matcher) {} | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> getSingleMatcher() const override { | 
|  | return Matcher; | 
|  | } | 
|  |  | 
|  | std::string getTypeAsString() const override { | 
|  | return (Twine("Matcher<") + Matcher.getSupportedKind().asStringRef() + ">") | 
|  | .str(); | 
|  | } | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> | 
|  | getTypedMatcher(const MatcherOps &Ops) const override { | 
|  | bool Ignore; | 
|  | if (Ops.canConstructFrom(Matcher, Ignore)) | 
|  | return Matcher; | 
|  | return llvm::None; | 
|  | } | 
|  |  | 
|  | bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override { | 
|  | return ArgKind(Matcher.getSupportedKind()) | 
|  | .isConvertibleTo(Kind, Specificity); | 
|  | } | 
|  |  | 
|  | private: | 
|  | const DynTypedMatcher Matcher; | 
|  | }; | 
|  |  | 
|  | class VariantMatcher::PolymorphicPayload : public VariantMatcher::Payload { | 
|  | public: | 
|  | PolymorphicPayload(std::vector<DynTypedMatcher> MatchersIn) | 
|  | : Matchers(std::move(MatchersIn)) {} | 
|  |  | 
|  | ~PolymorphicPayload() override {} | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> getSingleMatcher() const override { | 
|  | if (Matchers.size() != 1) | 
|  | return llvm::Optional<DynTypedMatcher>(); | 
|  | return Matchers[0]; | 
|  | } | 
|  |  | 
|  | std::string getTypeAsString() const override { | 
|  | std::string Inner; | 
|  | for (size_t i = 0, e = Matchers.size(); i != e; ++i) { | 
|  | if (i != 0) | 
|  | Inner += "|"; | 
|  | Inner += Matchers[i].getSupportedKind().asStringRef(); | 
|  | } | 
|  | return (Twine("Matcher<") + Inner + ">").str(); | 
|  | } | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> | 
|  | getTypedMatcher(const MatcherOps &Ops) const override { | 
|  | bool FoundIsExact = false; | 
|  | const DynTypedMatcher *Found = nullptr; | 
|  | int NumFound = 0; | 
|  | for (size_t i = 0, e = Matchers.size(); i != e; ++i) { | 
|  | bool IsExactMatch; | 
|  | if (Ops.canConstructFrom(Matchers[i], IsExactMatch)) { | 
|  | if (Found) { | 
|  | if (FoundIsExact) { | 
|  | assert(!IsExactMatch && "We should not have two exact matches."); | 
|  | continue; | 
|  | } | 
|  | } | 
|  | Found = &Matchers[i]; | 
|  | FoundIsExact = IsExactMatch; | 
|  | ++NumFound; | 
|  | } | 
|  | } | 
|  | // We only succeed if we found exactly one, or if we found an exact match. | 
|  | if (Found && (FoundIsExact || NumFound == 1)) | 
|  | return *Found; | 
|  | return llvm::None; | 
|  | } | 
|  |  | 
|  | bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override { | 
|  | unsigned MaxSpecificity = 0; | 
|  | for (const DynTypedMatcher &Matcher : Matchers) { | 
|  | unsigned ThisSpecificity; | 
|  | if (ArgKind(Matcher.getSupportedKind()) | 
|  | .isConvertibleTo(Kind, &ThisSpecificity)) { | 
|  | MaxSpecificity = std::max(MaxSpecificity, ThisSpecificity); | 
|  | } | 
|  | } | 
|  | if (Specificity) | 
|  | *Specificity = MaxSpecificity; | 
|  | return MaxSpecificity > 0; | 
|  | } | 
|  |  | 
|  | const std::vector<DynTypedMatcher> Matchers; | 
|  | }; | 
|  |  | 
|  | class VariantMatcher::VariadicOpPayload : public VariantMatcher::Payload { | 
|  | public: | 
|  | VariadicOpPayload(DynTypedMatcher::VariadicOperator Op, | 
|  | std::vector<VariantMatcher> Args) | 
|  | : Op(Op), Args(std::move(Args)) {} | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> getSingleMatcher() const override { | 
|  | return llvm::Optional<DynTypedMatcher>(); | 
|  | } | 
|  |  | 
|  | std::string getTypeAsString() const override { | 
|  | std::string Inner; | 
|  | for (size_t i = 0, e = Args.size(); i != e; ++i) { | 
|  | if (i != 0) | 
|  | Inner += "&"; | 
|  | Inner += Args[i].getTypeAsString(); | 
|  | } | 
|  | return Inner; | 
|  | } | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> | 
|  | getTypedMatcher(const MatcherOps &Ops) const override { | 
|  | return Ops.constructVariadicOperator(Op, Args); | 
|  | } | 
|  |  | 
|  | bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override { | 
|  | for (const VariantMatcher &Matcher : Args) { | 
|  | if (!Matcher.isConvertibleTo(Kind, Specificity)) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | const DynTypedMatcher::VariadicOperator Op; | 
|  | const std::vector<VariantMatcher> Args; | 
|  | }; | 
|  |  | 
|  | VariantMatcher::VariantMatcher() {} | 
|  |  | 
|  | VariantMatcher VariantMatcher::SingleMatcher(const DynTypedMatcher &Matcher) { | 
|  | return VariantMatcher(std::make_shared<SinglePayload>(Matcher)); | 
|  | } | 
|  |  | 
|  | VariantMatcher | 
|  | VariantMatcher::PolymorphicMatcher(std::vector<DynTypedMatcher> Matchers) { | 
|  | return VariantMatcher( | 
|  | std::make_shared<PolymorphicPayload>(std::move(Matchers))); | 
|  | } | 
|  |  | 
|  | VariantMatcher VariantMatcher::VariadicOperatorMatcher( | 
|  | DynTypedMatcher::VariadicOperator Op, | 
|  | std::vector<VariantMatcher> Args) { | 
|  | return VariantMatcher( | 
|  | std::make_shared<VariadicOpPayload>(Op, std::move(Args))); | 
|  | } | 
|  |  | 
|  | llvm::Optional<DynTypedMatcher> VariantMatcher::getSingleMatcher() const { | 
|  | return Value ? Value->getSingleMatcher() : llvm::Optional<DynTypedMatcher>(); | 
|  | } | 
|  |  | 
|  | void VariantMatcher::reset() { Value.reset(); } | 
|  |  | 
|  | std::string VariantMatcher::getTypeAsString() const { | 
|  | if (Value) return Value->getTypeAsString(); | 
|  | return "<Nothing>"; | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(const VariantValue &Other) : Type(VT_Nothing) { | 
|  | *this = Other; | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(bool Boolean) : Type(VT_Nothing) { | 
|  | setBoolean(Boolean); | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(double Double) : Type(VT_Nothing) { | 
|  | setDouble(Double); | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(unsigned Unsigned) : Type(VT_Nothing) { | 
|  | setUnsigned(Unsigned); | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(StringRef String) : Type(VT_Nothing) { | 
|  | setString(String); | 
|  | } | 
|  |  | 
|  | VariantValue::VariantValue(const VariantMatcher &Matcher) : Type(VT_Nothing) { | 
|  | setMatcher(Matcher); | 
|  | } | 
|  |  | 
|  | VariantValue::~VariantValue() { reset(); } | 
|  |  | 
|  | VariantValue &VariantValue::operator=(const VariantValue &Other) { | 
|  | if (this == &Other) return *this; | 
|  | reset(); | 
|  | switch (Other.Type) { | 
|  | case VT_Boolean: | 
|  | setBoolean(Other.getBoolean()); | 
|  | break; | 
|  | case VT_Double: | 
|  | setDouble(Other.getDouble()); | 
|  | break; | 
|  | case VT_Unsigned: | 
|  | setUnsigned(Other.getUnsigned()); | 
|  | break; | 
|  | case VT_String: | 
|  | setString(Other.getString()); | 
|  | break; | 
|  | case VT_Matcher: | 
|  | setMatcher(Other.getMatcher()); | 
|  | break; | 
|  | case VT_Nothing: | 
|  | Type = VT_Nothing; | 
|  | break; | 
|  | } | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | void VariantValue::reset() { | 
|  | switch (Type) { | 
|  | case VT_String: | 
|  | delete Value.String; | 
|  | break; | 
|  | case VT_Matcher: | 
|  | delete Value.Matcher; | 
|  | break; | 
|  | // Cases that do nothing. | 
|  | case VT_Boolean: | 
|  | case VT_Double: | 
|  | case VT_Unsigned: | 
|  | case VT_Nothing: | 
|  | break; | 
|  | } | 
|  | Type = VT_Nothing; | 
|  | } | 
|  |  | 
|  | bool VariantValue::isBoolean() const { | 
|  | return Type == VT_Boolean; | 
|  | } | 
|  |  | 
|  | bool VariantValue::getBoolean() const { | 
|  | assert(isBoolean()); | 
|  | return Value.Boolean; | 
|  | } | 
|  |  | 
|  | void VariantValue::setBoolean(bool NewValue) { | 
|  | reset(); | 
|  | Type = VT_Boolean; | 
|  | Value.Boolean = NewValue; | 
|  | } | 
|  |  | 
|  | bool VariantValue::isDouble() const { | 
|  | return Type == VT_Double; | 
|  | } | 
|  |  | 
|  | double VariantValue::getDouble() const { | 
|  | assert(isDouble()); | 
|  | return Value.Double; | 
|  | } | 
|  |  | 
|  | void VariantValue::setDouble(double NewValue) { | 
|  | reset(); | 
|  | Type = VT_Double; | 
|  | Value.Double = NewValue; | 
|  | } | 
|  |  | 
|  | bool VariantValue::isUnsigned() const { | 
|  | return Type == VT_Unsigned; | 
|  | } | 
|  |  | 
|  | unsigned VariantValue::getUnsigned() const { | 
|  | assert(isUnsigned()); | 
|  | return Value.Unsigned; | 
|  | } | 
|  |  | 
|  | void VariantValue::setUnsigned(unsigned NewValue) { | 
|  | reset(); | 
|  | Type = VT_Unsigned; | 
|  | Value.Unsigned = NewValue; | 
|  | } | 
|  |  | 
|  | bool VariantValue::isString() const { | 
|  | return Type == VT_String; | 
|  | } | 
|  |  | 
|  | const std::string &VariantValue::getString() const { | 
|  | assert(isString()); | 
|  | return *Value.String; | 
|  | } | 
|  |  | 
|  | void VariantValue::setString(StringRef NewValue) { | 
|  | reset(); | 
|  | Type = VT_String; | 
|  | Value.String = new std::string(NewValue); | 
|  | } | 
|  |  | 
|  | bool VariantValue::isMatcher() const { | 
|  | return Type == VT_Matcher; | 
|  | } | 
|  |  | 
|  | const VariantMatcher &VariantValue::getMatcher() const { | 
|  | assert(isMatcher()); | 
|  | return *Value.Matcher; | 
|  | } | 
|  |  | 
|  | void VariantValue::setMatcher(const VariantMatcher &NewValue) { | 
|  | reset(); | 
|  | Type = VT_Matcher; | 
|  | Value.Matcher = new VariantMatcher(NewValue); | 
|  | } | 
|  |  | 
|  | bool VariantValue::isConvertibleTo(ArgKind Kind, unsigned *Specificity) const { | 
|  | switch (Kind.getArgKind()) { | 
|  | case ArgKind::AK_Boolean: | 
|  | if (!isBoolean()) | 
|  | return false; | 
|  | *Specificity = 1; | 
|  | return true; | 
|  |  | 
|  | case ArgKind::AK_Double: | 
|  | if (!isDouble()) | 
|  | return false; | 
|  | *Specificity = 1; | 
|  | return true; | 
|  |  | 
|  | case ArgKind::AK_Unsigned: | 
|  | if (!isUnsigned()) | 
|  | return false; | 
|  | *Specificity = 1; | 
|  | return true; | 
|  |  | 
|  | case ArgKind::AK_String: | 
|  | if (!isString()) | 
|  | return false; | 
|  | *Specificity = 1; | 
|  | return true; | 
|  |  | 
|  | case ArgKind::AK_Matcher: | 
|  | if (!isMatcher()) | 
|  | return false; | 
|  | return getMatcher().isConvertibleTo(Kind.getMatcherKind(), Specificity); | 
|  | } | 
|  | llvm_unreachable("Invalid Type"); | 
|  | } | 
|  |  | 
|  | bool VariantValue::isConvertibleTo(ArrayRef<ArgKind> Kinds, | 
|  | unsigned *Specificity) const { | 
|  | unsigned MaxSpecificity = 0; | 
|  | for (const ArgKind& Kind : Kinds) { | 
|  | unsigned ThisSpecificity; | 
|  | if (!isConvertibleTo(Kind, &ThisSpecificity)) | 
|  | continue; | 
|  | MaxSpecificity = std::max(MaxSpecificity, ThisSpecificity); | 
|  | } | 
|  | if (Specificity && MaxSpecificity > 0) { | 
|  | *Specificity = MaxSpecificity; | 
|  | } | 
|  | return MaxSpecificity > 0; | 
|  | } | 
|  |  | 
|  | std::string VariantValue::getTypeAsString() const { | 
|  | switch (Type) { | 
|  | case VT_String: return "String"; | 
|  | case VT_Matcher: return getMatcher().getTypeAsString(); | 
|  | case VT_Boolean: return "Boolean"; | 
|  | case VT_Double: return "Double"; | 
|  | case VT_Unsigned: return "Unsigned"; | 
|  | case VT_Nothing: return "Nothing"; | 
|  | } | 
|  | llvm_unreachable("Invalid Type"); | 
|  | } | 
|  |  | 
|  | } // end namespace dynamic | 
|  | } // end namespace ast_matchers | 
|  | } // end namespace clang |