clang  19.0.0git
VariantValue.cpp
Go to the documentation of this file.
1 //===--- VariantValue.cpp - Polymorphic value type --------------*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 ///
9 /// \file
10 /// Polymorphic value type.
11 ///
12 //===----------------------------------------------------------------------===//
13 
15 #include "clang/Basic/LLVM.h"
16 #include "llvm/ADT/STLExtras.h"
17 #include <optional>
18 
19 namespace clang {
20 namespace ast_matchers {
21 namespace dynamic {
22 
23 std::string ArgKind::asString() const {
24  switch (getArgKind()) {
25  case AK_Matcher:
26  return (Twine("Matcher<") + NodeKind.asStringRef() + ">").str();
27  case AK_Node:
28  return NodeKind.asStringRef().str();
29  case AK_Boolean:
30  return "boolean";
31  case AK_Double:
32  return "double";
33  case AK_Unsigned:
34  return "unsigned";
35  case AK_String:
36  return "string";
37  }
38  llvm_unreachable("unhandled ArgKind");
39 }
40 
41 bool ArgKind::isConvertibleTo(ArgKind To, unsigned *Specificity) const {
42  if (K != To.K)
43  return false;
44  if (K != AK_Matcher && K != AK_Node) {
45  if (Specificity)
46  *Specificity = 1;
47  return true;
48  }
49  unsigned Distance;
50  if (!NodeKind.isBaseOf(To.NodeKind, &Distance))
51  return false;
52 
53  if (Specificity)
54  *Specificity = 100 - Distance;
55  return true;
56 }
57 
58 bool
59 VariantMatcher::MatcherOps::canConstructFrom(const DynTypedMatcher &Matcher,
60  bool &IsExactMatch) const {
61  IsExactMatch = Matcher.getSupportedKind().isSame(NodeKind);
62  return Matcher.canConvertTo(NodeKind);
63 }
64 
65 DynTypedMatcher VariantMatcher::MatcherOps::convertMatcher(
66  const DynTypedMatcher &Matcher) const {
67  return Matcher.dynCastTo(NodeKind);
68 }
69 
70 std::optional<DynTypedMatcher>
71 VariantMatcher::MatcherOps::constructVariadicOperator(
72  DynTypedMatcher::VariadicOperator Op,
73  ArrayRef<VariantMatcher> InnerMatchers) const {
74  std::vector<DynTypedMatcher> DynMatchers;
75  for (const auto &InnerMatcher : InnerMatchers) {
76  // Abort if any of the inner matchers can't be converted to
77  // Matcher<T>.
78  if (!InnerMatcher.Value)
79  return std::nullopt;
80  std::optional<DynTypedMatcher> Inner =
81  InnerMatcher.Value->getTypedMatcher(*this);
82  if (!Inner)
83  return std::nullopt;
84  DynMatchers.push_back(*Inner);
85  }
86  return DynTypedMatcher::constructVariadic(Op, NodeKind, DynMatchers);
87 }
88 
89 VariantMatcher::Payload::~Payload() {}
90 
91 class VariantMatcher::SinglePayload : public VariantMatcher::Payload {
92 public:
93  SinglePayload(const DynTypedMatcher &Matcher) : Matcher(Matcher) {}
94 
95  std::optional<DynTypedMatcher> getSingleMatcher() const override {
96  return Matcher;
97  }
98 
99  std::string getTypeAsString() const override {
100  return (Twine("Matcher<") + Matcher.getSupportedKind().asStringRef() + ">")
101  .str();
102  }
103 
104  std::optional<DynTypedMatcher>
105  getTypedMatcher(const MatcherOps &Ops) const override {
106  bool Ignore;
107  if (Ops.canConstructFrom(Matcher, Ignore))
108  return Matcher;
109  return std::nullopt;
110  }
111 
112  bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override {
113  return ArgKind::MakeMatcherArg(Matcher.getSupportedKind())
115  }
116 
117 private:
118  const DynTypedMatcher Matcher;
119 };
120 
121 class VariantMatcher::PolymorphicPayload : public VariantMatcher::Payload {
122 public:
123  PolymorphicPayload(std::vector<DynTypedMatcher> MatchersIn)
124  : Matchers(std::move(MatchersIn)) {}
125 
126  ~PolymorphicPayload() override {}
127 
128  std::optional<DynTypedMatcher> getSingleMatcher() const override {
129  if (Matchers.size() != 1)
130  return std::nullopt;
131  return Matchers[0];
132  }
133 
134  std::string getTypeAsString() const override {
135  std::string Inner;
136  for (size_t i = 0, e = Matchers.size(); i != e; ++i) {
137  if (i != 0)
138  Inner += "|";
139  Inner += Matchers[i].getSupportedKind().asStringRef();
140  }
141  return (Twine("Matcher<") + Inner + ">").str();
142  }
143 
144  std::optional<DynTypedMatcher>
145  getTypedMatcher(const MatcherOps &Ops) const override {
146  bool FoundIsExact = false;
147  const DynTypedMatcher *Found = nullptr;
148  int NumFound = 0;
149  for (size_t i = 0, e = Matchers.size(); i != e; ++i) {
150  bool IsExactMatch;
151  if (Ops.canConstructFrom(Matchers[i], IsExactMatch)) {
152  if (Found) {
153  if (FoundIsExact) {
154  assert(!IsExactMatch && "We should not have two exact matches.");
155  continue;
156  }
157  }
158  Found = &Matchers[i];
159  FoundIsExact = IsExactMatch;
160  ++NumFound;
161  }
162  }
163  // We only succeed if we found exactly one, or if we found an exact match.
164  if (Found && (FoundIsExact || NumFound == 1))
165  return *Found;
166  return std::nullopt;
167  }
168 
169  bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override {
170  unsigned MaxSpecificity = 0;
171  for (const DynTypedMatcher &Matcher : Matchers) {
172  unsigned ThisSpecificity;
173  if (ArgKind::MakeMatcherArg(Matcher.getSupportedKind())
175  &ThisSpecificity)) {
176  MaxSpecificity = std::max(MaxSpecificity, ThisSpecificity);
177  }
178  }
179  if (Specificity)
180  *Specificity = MaxSpecificity;
181  return MaxSpecificity > 0;
182  }
183 
184  const std::vector<DynTypedMatcher> Matchers;
185 };
186 
187 class VariantMatcher::VariadicOpPayload : public VariantMatcher::Payload {
188 public:
189  VariadicOpPayload(DynTypedMatcher::VariadicOperator Op,
190  std::vector<VariantMatcher> Args)
191  : Op(Op), Args(std::move(Args)) {}
192 
193  std::optional<DynTypedMatcher> getSingleMatcher() const override {
194  return std::nullopt;
195  }
196 
197  std::string getTypeAsString() const override {
198  std::string Inner;
199  for (size_t i = 0, e = Args.size(); i != e; ++i) {
200  if (i != 0)
201  Inner += "&";
202  Inner += Args[i].getTypeAsString();
203  }
204  return Inner;
205  }
206 
207  std::optional<DynTypedMatcher>
208  getTypedMatcher(const MatcherOps &Ops) const override {
209  return Ops.constructVariadicOperator(Op, Args);
210  }
211 
212  bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override {
213  for (const VariantMatcher &Matcher : Args) {
214  if (!Matcher.isConvertibleTo(Kind, Specificity))
215  return false;
216  }
217  return true;
218  }
219 
220 private:
221  const DynTypedMatcher::VariadicOperator Op;
222  const std::vector<VariantMatcher> Args;
223 };
224 
226 
227 VariantMatcher VariantMatcher::SingleMatcher(const DynTypedMatcher &Matcher) {
228  return VariantMatcher(std::make_shared<SinglePayload>(Matcher));
229 }
230 
232 VariantMatcher::PolymorphicMatcher(std::vector<DynTypedMatcher> Matchers) {
233  return VariantMatcher(
234  std::make_shared<PolymorphicPayload>(std::move(Matchers)));
235 }
236 
238  DynTypedMatcher::VariadicOperator Op,
239  std::vector<VariantMatcher> Args) {
240  return VariantMatcher(
241  std::make_shared<VariadicOpPayload>(Op, std::move(Args)));
242 }
243 
244 std::optional<DynTypedMatcher> VariantMatcher::getSingleMatcher() const {
245  return Value ? Value->getSingleMatcher() : std::optional<DynTypedMatcher>();
246 }
247 
248 void VariantMatcher::reset() { Value.reset(); }
249 
250 std::string VariantMatcher::getTypeAsString() const {
251  if (Value) return Value->getTypeAsString();
252  return "<Nothing>";
253 }
254 
255 VariantValue::VariantValue(const VariantValue &Other) : Type(VT_Nothing) {
256  *this = Other;
257 }
258 
261 }
262 
263 VariantValue::VariantValue(double Double) : Type(VT_Nothing) {
264  setDouble(Double);
265 }
266 
267 VariantValue::VariantValue(unsigned Unsigned) : Type(VT_Nothing) {
269 }
270 
271 VariantValue::VariantValue(StringRef String) : Type(VT_Nothing) {
272  setString(String);
273 }
274 
277 }
278 
279 VariantValue::VariantValue(const VariantMatcher &Matcher) : Type(VT_Nothing) {
280  setMatcher(Matcher);
281 }
282 
284 
286  if (this == &Other) return *this;
287  reset();
288  switch (Other.Type) {
289  case VT_Boolean:
290  setBoolean(Other.getBoolean());
291  break;
292  case VT_Double:
293  setDouble(Other.getDouble());
294  break;
295  case VT_Unsigned:
296  setUnsigned(Other.getUnsigned());
297  break;
298  case VT_String:
299  setString(Other.getString());
300  break;
301  case VT_NodeKind:
302  setNodeKind(Other.getNodeKind());
303  break;
304  case VT_Matcher:
305  setMatcher(Other.getMatcher());
306  break;
307  case VT_Nothing:
308  Type = VT_Nothing;
309  break;
310  }
311  return *this;
312 }
313 
314 void VariantValue::reset() {
315  switch (Type) {
316  case VT_String:
317  delete Value.String;
318  break;
319  case VT_Matcher:
320  delete Value.Matcher;
321  break;
322  case VT_NodeKind:
323  delete Value.NodeKind;
324  break;
325  // Cases that do nothing.
326  case VT_Boolean:
327  case VT_Double:
328  case VT_Unsigned:
329  case VT_Nothing:
330  break;
331  }
332  Type = VT_Nothing;
333 }
334 
336  return Type == VT_Boolean;
337 }
338 
340  assert(isBoolean());
341  return Value.Boolean;
342 }
343 
344 void VariantValue::setBoolean(bool NewValue) {
345  reset();
346  Type = VT_Boolean;
347  Value.Boolean = NewValue;
348 }
349 
351  return Type == VT_Double;
352 }
353 
354 double VariantValue::getDouble() const {
355  assert(isDouble());
356  return Value.Double;
357 }
358 
359 void VariantValue::setDouble(double NewValue) {
360  reset();
361  Type = VT_Double;
362  Value.Double = NewValue;
363 }
364 
366  return Type == VT_Unsigned;
367 }
368 
369 unsigned VariantValue::getUnsigned() const {
370  assert(isUnsigned());
371  return Value.Unsigned;
372 }
373 
374 void VariantValue::setUnsigned(unsigned NewValue) {
375  reset();
376  Type = VT_Unsigned;
377  Value.Unsigned = NewValue;
378 }
379 
381  return Type == VT_String;
382 }
383 
384 const std::string &VariantValue::getString() const {
385  assert(isString());
386  return *Value.String;
387 }
388 
389 void VariantValue::setString(StringRef NewValue) {
390  reset();
391  Type = VT_String;
392  Value.String = new std::string(NewValue);
393 }
394 
395 bool VariantValue::isNodeKind() const { return Type == VT_NodeKind; }
396 
398  assert(isNodeKind());
399  return *Value.NodeKind;
400 }
401 
403  reset();
404  Type = VT_NodeKind;
405  Value.NodeKind = new ASTNodeKind(NewValue);
406 }
407 
409  return Type == VT_Matcher;
410 }
411 
413  assert(isMatcher());
414  return *Value.Matcher;
415 }
416 
418  reset();
419  Type = VT_Matcher;
420  Value.Matcher = new VariantMatcher(NewValue);
421 }
422 
423 bool VariantValue::isConvertibleTo(ArgKind Kind, unsigned *Specificity) const {
424  switch (Kind.getArgKind()) {
425  case ArgKind::AK_Boolean:
426  if (!isBoolean())
427  return false;
428  *Specificity = 1;
429  return true;
430 
431  case ArgKind::AK_Double:
432  if (!isDouble())
433  return false;
434  *Specificity = 1;
435  return true;
436 
438  if (!isUnsigned())
439  return false;
440  *Specificity = 1;
441  return true;
442 
443  case ArgKind::AK_String:
444  if (!isString())
445  return false;
446  *Specificity = 1;
447  return true;
448 
449  case ArgKind::AK_Node:
450  if (!isNodeKind())
451  return false;
452  return getMatcher().isConvertibleTo(Kind.getNodeKind(), Specificity);
453 
454  case ArgKind::AK_Matcher:
455  if (!isMatcher())
456  return false;
457  return getMatcher().isConvertibleTo(Kind.getMatcherKind(), Specificity);
458  }
459  llvm_unreachable("Invalid Type");
460 }
461 
463  unsigned *Specificity) const {
464  unsigned MaxSpecificity = 0;
465  for (const ArgKind& Kind : Kinds) {
466  unsigned ThisSpecificity;
467  if (!isConvertibleTo(Kind, &ThisSpecificity))
468  continue;
469  MaxSpecificity = std::max(MaxSpecificity, ThisSpecificity);
470  }
471  if (Specificity && MaxSpecificity > 0) {
472  *Specificity = MaxSpecificity;
473  }
474  return MaxSpecificity > 0;
475 }
476 
477 std::string VariantValue::getTypeAsString() const {
478  switch (Type) {
479  case VT_String: return "String";
480  case VT_Matcher: return getMatcher().getTypeAsString();
481  case VT_Boolean: return "Boolean";
482  case VT_Double: return "Double";
483  case VT_Unsigned: return "Unsigned";
484  case VT_NodeKind:
485  return getNodeKind().asStringRef().str();
486  case VT_Nothing: return "Nothing";
487  }
488  llvm_unreachable("Invalid Type");
489 }
490 
491 } // end namespace dynamic
492 } // end namespace ast_matchers
493 } // end namespace clang
Forward-declares and imports various common LLVM datatypes that clang wants to use unqualified.
Polymorphic value type.
__DEVICE__ int max(int __a, int __b)
Kind identifier.
Definition: ASTTypeTraits.h:51
StringRef asStringRef() const
String representation of the kind.
bool isBaseOf(ASTNodeKind Other) const
Returns true if this is a base kind of (or same as) Other.
The base class of the type hierarchy.
Definition: Type.h:1813
static ArgKind MakeMatcherArg(ASTNodeKind MatcherKind)
Constructor for matcher types.
Definition: VariantValue.h:48
std::string asString() const
String representation of the type.
bool isConvertibleTo(ArgKind To, unsigned *Specificity) const
Determines if this type can be converted to To.
bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override
std::optional< DynTypedMatcher > getTypedMatcher(const MatcherOps &Ops) const override
PolymorphicPayload(std::vector< DynTypedMatcher > MatchersIn)
std::optional< DynTypedMatcher > getSingleMatcher() const override
bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override
std::optional< DynTypedMatcher > getSingleMatcher() const override
std::optional< DynTypedMatcher > getTypedMatcher(const MatcherOps &Ops) const override
VariadicOpPayload(DynTypedMatcher::VariadicOperator Op, std::vector< VariantMatcher > Args)
std::optional< DynTypedMatcher > getTypedMatcher(const MatcherOps &Ops) const override
bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const override
std::optional< DynTypedMatcher > getSingleMatcher() const override
static VariantMatcher PolymorphicMatcher(std::vector< DynTypedMatcher > Matchers)
Clones the provided matchers.
void reset()
Makes the matcher the "null" matcher.
std::optional< DynTypedMatcher > getSingleMatcher() const
Return a single matcher, if there is no ambiguity.
std::string getTypeAsString() const
String representation of the type of the value.
bool isConvertibleTo(ASTNodeKind Kind, unsigned *Specificity) const
Determines if the contained matcher can be converted to Kind.
Definition: VariantValue.h:200
static VariantMatcher VariadicOperatorMatcher(DynTypedMatcher::VariadicOperator Op, std::vector< VariantMatcher > Args)
Creates a 'variadic' operator matcher.
static VariantMatcher SingleMatcher(const DynTypedMatcher &Matcher)
Clones the provided matcher.
bool isString() const
String value functions.
void setNodeKind(ASTNodeKind NodeKind)
const ASTNodeKind & getNodeKind() const
const std::string & getString() const
bool isUnsigned() const
Unsigned value functions.
std::string getTypeAsString() const
String representation of the type of the value.
bool isConvertibleTo(ArgKind Kind, unsigned *Specificity) const
Determines if the contained value can be converted to Kind.
const VariantMatcher & getMatcher() const
bool isMatcher() const
Matcher value functions.
bool isBoolean() const
Boolean value functions.
void setMatcher(const VariantMatcher &Matcher)
VariantValue & operator=(const VariantValue &Other)
bool isDouble() const
Double value functions.
NodeKind
A kind of a syntax node, used for implementing casts.
Definition: Nodes.h:32
The JSON file list parser is used to communicate input to InstallAPI.
@ Other
Other implicit parameter.
Definition: Format.h:5433