485 lines
12 KiB
C++
485 lines
12 KiB
C++
|
//===--- 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);
|
||
|
}
|
||
|
|
||
|
DynTypedMatcher VariantMatcher::MatcherOps::convertMatcher(
|
||
|
const DynTypedMatcher &Matcher) const {
|
||
|
return Matcher.dynCastTo(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(ASTNodeKind NodeKind) : Type(VT_Nothing) {
|
||
|
setNodeKind(NodeKind);
|
||
|
}
|
||
|
|
||
|
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_NodeKind:
|
||
|
setNodeKind(Other.getNodeKind());
|
||
|
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;
|
||
|
case VT_NodeKind:
|
||
|
delete Value.NodeKind;
|
||
|
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::isNodeKind() const { return Type == VT_NodeKind; }
|
||
|
|
||
|
const ASTNodeKind &VariantValue::getNodeKind() const {
|
||
|
assert(isNodeKind());
|
||
|
return *Value.NodeKind;
|
||
|
}
|
||
|
|
||
|
void VariantValue::setNodeKind(ASTNodeKind NewValue) {
|
||
|
reset();
|
||
|
Type = VT_NodeKind;
|
||
|
Value.NodeKind = new ASTNodeKind(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_NodeKind:
|
||
|
return getNodeKind().asStringRef().str();
|
||
|
case VT_Nothing: return "Nothing";
|
||
|
}
|
||
|
llvm_unreachable("Invalid Type");
|
||
|
}
|
||
|
|
||
|
} // end namespace dynamic
|
||
|
} // end namespace ast_matchers
|
||
|
} // end namespace clang
|