clang 22.0.0git
RecursiveASTVisitor.h
Go to the documentation of this file.
1//===--- RecursiveASTVisitor.h - Recursive AST Visitor ----------*- 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// This file defines the RecursiveASTVisitor interface, which recursively
10// traverses the entire AST.
11//
12//===----------------------------------------------------------------------===//
13#ifndef LLVM_CLANG_AST_RECURSIVEASTVISITOR_H
14#define LLVM_CLANG_AST_RECURSIVEASTVISITOR_H
15
17#include "clang/AST/Attr.h"
18#include "clang/AST/Decl.h"
19#include "clang/AST/DeclBase.h"
20#include "clang/AST/DeclCXX.h"
22#include "clang/AST/DeclObjC.h"
27#include "clang/AST/Expr.h"
28#include "clang/AST/ExprCXX.h"
30#include "clang/AST/ExprObjC.h"
36#include "clang/AST/Stmt.h"
37#include "clang/AST/StmtCXX.h"
38#include "clang/AST/StmtObjC.h"
41#include "clang/AST/StmtSYCL.h"
44#include "clang/AST/Type.h"
45#include "clang/AST/TypeLoc.h"
46#include "clang/Basic/LLVM.h"
49#include "llvm/ADT/PointerIntPair.h"
50#include "llvm/ADT/SmallVector.h"
51#include "llvm/Support/Casting.h"
52#include <algorithm>
53#include <cstddef>
54#include <type_traits>
55
56namespace clang {
57
58// A helper macro to implement short-circuiting when recursing. It
59// invokes CALL_EXPR, which must be a method call, on the derived
60// object (s.t. a user of RecursiveASTVisitor can override the method
61// in CALL_EXPR).
62#define TRY_TO(CALL_EXPR) \
63 do { \
64 if (!getDerived().CALL_EXPR) \
65 return false; \
66 } while (false)
67
68namespace detail {
69
70template <typename T, typename U>
71struct has_same_member_pointer_type : std::false_type {};
72template <typename T, typename U, typename R, typename... P>
73struct has_same_member_pointer_type<R (T::*)(P...), R (U::*)(P...)>
74 : std::true_type {};
75
76/// Returns true if and only if \p FirstMethodPtr and \p SecondMethodPtr
77/// are pointers to the same non-static member function.
78template <typename FirstMethodPtrTy, typename SecondMethodPtrTy>
79LLVM_ATTRIBUTE_ALWAYS_INLINE LLVM_ATTRIBUTE_NODEBUG auto
80isSameMethod([[maybe_unused]] FirstMethodPtrTy FirstMethodPtr,
81 [[maybe_unused]] SecondMethodPtrTy SecondMethodPtr)
82 -> bool {
83 if constexpr (has_same_member_pointer_type<FirstMethodPtrTy,
84 SecondMethodPtrTy>::value)
85 return FirstMethodPtr == SecondMethodPtr;
86 return false;
87}
88
89} // end namespace detail
90
91/// A class that does preorder or postorder
92/// depth-first traversal on the entire Clang AST and visits each node.
93///
94/// This class performs three distinct tasks:
95/// 1. traverse the AST (i.e. go to each node);
96/// 2. at a given node, walk up the class hierarchy, starting from
97/// the node's dynamic type, until the top-most class (e.g. Stmt,
98/// Decl, or Type) is reached.
99/// 3. given a (node, class) combination, where 'class' is some base
100/// class of the dynamic type of 'node', call a user-overridable
101/// function to actually visit the node.
102///
103/// These tasks are done by three groups of methods, respectively:
104/// 1. TraverseDecl(Decl *x) does task #1. It is the entry point
105/// for traversing an AST rooted at x. This method simply
106/// dispatches (i.e. forwards) to TraverseFoo(Foo *x) where Foo
107/// is the dynamic type of *x, which calls WalkUpFromFoo(x) and
108/// then recursively visits the child nodes of x.
109/// TraverseStmt(Stmt *x) and TraverseType(QualType x) work
110/// similarly.
111/// 2. WalkUpFromFoo(Foo *x) does task #2. It does not try to visit
112/// any child node of x. Instead, it first calls WalkUpFromBar(x)
113/// where Bar is the direct parent class of Foo (unless Foo has
114/// no parent), and then calls VisitFoo(x) (see the next list item).
115/// 3. VisitFoo(Foo *x) does task #3.
116///
117/// These three method groups are tiered (Traverse* > WalkUpFrom* >
118/// Visit*). A method (e.g. Traverse*) may call methods from the same
119/// tier (e.g. other Traverse*) or one tier lower (e.g. WalkUpFrom*).
120/// It may not call methods from a higher tier.
121///
122/// Note that since WalkUpFromFoo() calls WalkUpFromBar() (where Bar
123/// is Foo's super class) before calling VisitFoo(), the result is
124/// that the Visit*() methods for a given node are called in the
125/// top-down order (e.g. for a node of type NamespaceDecl, the order will
126/// be VisitDecl(), VisitNamedDecl(), and then VisitNamespaceDecl()).
127///
128/// This scheme guarantees that all Visit*() calls for the same AST
129/// node are grouped together. In other words, Visit*() methods for
130/// different nodes are never interleaved.
131///
132/// Clients of this visitor should subclass the visitor (providing
133/// themselves as the template argument, using the curiously recurring
134/// template pattern) and override any of the Traverse*, WalkUpFrom*,
135/// and Visit* methods for declarations, types, statements,
136/// expressions, or other AST nodes where the visitor should customize
137/// behavior. Most users only need to override Visit*. Advanced
138/// users may override Traverse* and WalkUpFrom* to implement custom
139/// traversal strategies. Returning false from one of these overridden
140/// functions will abort the entire traversal.
141///
142/// By default, this visitor tries to visit every part of the explicit
143/// source code exactly once. The default policy towards templates
144/// is to descend into the 'pattern' class or function body, not any
145/// explicit or implicit instantiations. Explicit specializations
146/// are still visited, and the patterns of partial specializations
147/// are visited separately. This behavior can be changed by
148/// overriding shouldVisitTemplateInstantiations() in the derived class
149/// to return true, in which case all known implicit and explicit
150/// instantiations will be visited at the same time as the pattern
151/// from which they were produced.
152///
153/// By default, this visitor preorder traverses the AST. If postorder traversal
154/// is needed, the \c shouldTraversePostOrder method needs to be overridden
155/// to return \c true.
156template <typename Derived> class RecursiveASTVisitor {
157public:
158 /// A queue used for performing data recursion over statements.
159 /// Parameters involving this type are used to implement data
160 /// recursion over Stmts and Exprs within this class, and should
161 /// typically not be explicitly specified by derived classes.
162 /// The bool bit indicates whether the statement has been traversed or not.
165
166 /// Return a reference to the derived class.
167 Derived &getDerived() { return *static_cast<Derived *>(this); }
168
169 /// Return whether this visitor should recurse into
170 /// template instantiations.
171 bool shouldVisitTemplateInstantiations() const { return false; }
172
173 /// Return whether this visitor should recurse into the types of
174 /// TypeLocs.
175 bool shouldWalkTypesOfTypeLocs() const { return true; }
176
177 /// Return whether this visitor should recurse into implicit
178 /// code, e.g., implicit constructors and destructors.
179 bool shouldVisitImplicitCode() const { return false; }
180
181 /// Return whether this visitor should recurse into lambda body
182 bool shouldVisitLambdaBody() const { return true; }
183
184 /// Return whether this visitor should traverse post-order.
185 bool shouldTraversePostOrder() const { return false; }
186
187 /// Recursively visits an entire AST, starting from the TranslationUnitDecl.
188 /// \returns false if visitation was terminated early.
190 // Currently just an alias for TraverseDecl(TUDecl), but kept in case
191 // we change the implementation again.
192 return getDerived().TraverseDecl(AST.getTranslationUnitDecl());
193 }
194
195 /// Recursively visit a statement or expression, by
196 /// dispatching to Traverse*() based on the argument's dynamic type.
197 ///
198 /// \returns false if the visitation was terminated early, true
199 /// otherwise (including when the argument is nullptr).
200 bool TraverseStmt(Stmt *S, DataRecursionQueue *Queue = nullptr);
201
202 /// Invoked before visiting a statement or expression via data recursion.
203 ///
204 /// \returns false to skip visiting the node, true otherwise.
205 bool dataTraverseStmtPre(Stmt *S) { return true; }
206
207 /// Invoked after visiting a statement or expression via data recursion.
208 /// This is not invoked if the previously invoked \c dataTraverseStmtPre
209 /// returned false.
210 ///
211 /// \returns false if the visitation was terminated early, true otherwise.
212 bool dataTraverseStmtPost(Stmt *S) { return true; }
213
214 /// Recursively visit a type, by dispatching to
215 /// Traverse*Type() based on the argument's getTypeClass() property.
216 ///
217 /// \returns false if the visitation was terminated early, true
218 /// otherwise (including when the argument is a Null type).
219 bool TraverseType(QualType T, bool TraverseQualifier = true);
220
221 /// Recursively visit a type with location, by dispatching to
222 /// Traverse*TypeLoc() based on the argument type's getTypeClass() property.
223 ///
224 /// \returns false if the visitation was terminated early, true
225 /// otherwise (including when the argument is a Null type location).
226 bool TraverseTypeLoc(TypeLoc TL, bool TraverseQualifier = true);
227
228 /// Recursively visit an attribute, by dispatching to
229 /// Traverse*Attr() based on the argument's dynamic type.
230 ///
231 /// \returns false if the visitation was terminated early, true
232 /// otherwise (including when the argument is a Null type location).
234
235 /// Recursively visit a declaration, by dispatching to
236 /// Traverse*Decl() based on the argument's dynamic type.
237 ///
238 /// \returns false if the visitation was terminated early, true
239 /// otherwise (including when the argument is NULL).
241
242 /// Recursively visit a C++ nested-name-specifier.
243 ///
244 /// \returns false if the visitation was terminated early, true otherwise.
246
247 /// Recursively visit a C++ nested-name-specifier with location
248 /// information.
249 ///
250 /// \returns false if the visitation was terminated early, true otherwise.
252
253 /// Recursively visit a name with its location information.
254 ///
255 /// \returns false if the visitation was terminated early, true otherwise.
257
258 /// Recursively visit a template name and dispatch to the
259 /// appropriate method.
260 ///
261 /// \returns false if the visitation was terminated early, true otherwise.
263
264 /// Recursively visit a template argument and dispatch to the
265 /// appropriate method for the argument type.
266 ///
267 /// \returns false if the visitation was terminated early, true otherwise.
268 // FIXME: migrate callers to TemplateArgumentLoc instead.
270
271 /// Recursively visit a template argument location and dispatch to the
272 /// appropriate method for the argument type.
273 ///
274 /// \returns false if the visitation was terminated early, true otherwise.
276
277 /// Recursively visit a set of template arguments.
278 /// This can be overridden by a subclass, but it's not expected that
279 /// will be needed -- this visitor always dispatches to another.
280 ///
281 /// \returns false if the visitation was terminated early, true otherwise.
282 // FIXME: take a TemplateArgumentLoc* (or TemplateArgumentListInfo) instead.
284
285 /// Recursively visit a base specifier. This can be overridden by a
286 /// subclass.
287 ///
288 /// \returns false if the visitation was terminated early, true otherwise.
290
291 /// Recursively visit a constructor initializer. This
292 /// automatically dispatches to another visitor for the initializer
293 /// expression, but not for the name of the initializer, so may
294 /// be overridden for clients that need access to the name.
295 ///
296 /// \returns false if the visitation was terminated early, true otherwise.
298
299 /// Recursively visit a lambda capture. \c Init is the expression that
300 /// will be used to initialize the capture.
301 ///
302 /// \returns false if the visitation was terminated early, true otherwise.
304 Expr *Init);
305
306 /// Recursively visit the syntactic or semantic form of an
307 /// initialization list.
308 ///
309 /// \returns false if the visitation was terminated early, true otherwise.
311 DataRecursionQueue *Queue = nullptr);
312
313 /// Recursively visit an Objective-C protocol reference with location
314 /// information.
315 ///
316 /// \returns false if the visitation was terminated early, true otherwise.
318
319 /// Recursively visit concept reference with location information.
320 ///
321 /// \returns false if the visitation was terminated early, true otherwise.
323
324 // Visit concept reference.
325 bool VisitConceptReference(ConceptReference *CR) { return true; }
326 // ---- Methods on Attrs ----
327
328 // Visit an attribute.
329 bool VisitAttr(Attr *A) { return true; }
330
331// Declare Traverse* and empty Visit* for all Attr classes.
332#define ATTR_VISITOR_DECLS_ONLY
333#include "clang/AST/AttrVisitor.inc"
334#undef ATTR_VISITOR_DECLS_ONLY
335
336// ---- Methods on Stmts ----
337
339
340private:
341 // Traverse the given statement. If the most-derived traverse function takes a
342 // data recursion queue, pass it on; otherwise, discard it. Note that the
343 // first branch of this conditional must compile whether or not the derived
344 // class can take a queue, so if we're taking the second arm, make the first
345 // arm call our function rather than the derived class version.
346#define TRAVERSE_STMT_BASE(NAME, CLASS, VAR, QUEUE) \
347 (::clang::detail::has_same_member_pointer_type< \
348 decltype(&RecursiveASTVisitor::Traverse##NAME), \
349 decltype(&Derived::Traverse##NAME)>::value \
350 ? static_cast<std::conditional_t< \
351 ::clang::detail::has_same_member_pointer_type< \
352 decltype(&RecursiveASTVisitor::Traverse##NAME), \
353 decltype(&Derived::Traverse##NAME)>::value, \
354 Derived &, RecursiveASTVisitor &>>(*this) \
355 .Traverse##NAME(static_cast<CLASS *>(VAR), QUEUE) \
356 : getDerived().Traverse##NAME(static_cast<CLASS *>(VAR)))
357
358// Try to traverse the given statement, or enqueue it if we're performing data
359// recursion in the middle of traversing another statement. Can only be called
360// from within a DEF_TRAVERSE_STMT body or similar context.
361#define TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S) \
362 do { \
363 if (!TRAVERSE_STMT_BASE(Stmt, Stmt, S, Queue)) \
364 return false; \
365 } while (false)
366
367public:
368// Declare Traverse*() for all concrete Stmt classes.
369#define ABSTRACT_STMT(STMT)
370#define STMT(CLASS, PARENT) \
371 bool Traverse##CLASS(CLASS *S, DataRecursionQueue *Queue = nullptr);
372#include "clang/AST/StmtNodes.inc"
373 // The above header #undefs ABSTRACT_STMT and STMT upon exit.
374
375 // Define WalkUpFrom*() and empty Visit*() for all Stmt classes.
376 bool WalkUpFromStmt(Stmt *S) { return getDerived().VisitStmt(S); }
377 bool VisitStmt(Stmt *S) { return true; }
378#define STMT(CLASS, PARENT) \
379 bool WalkUpFrom##CLASS(CLASS *S) { \
380 TRY_TO(WalkUpFrom##PARENT(S)); \
381 TRY_TO(Visit##CLASS(S)); \
382 return true; \
383 } \
384 bool Visit##CLASS(CLASS *S) { return true; }
385#include "clang/AST/StmtNodes.inc"
386
387// ---- Methods on Types ----
388// FIXME: revamp to take TypeLoc's rather than Types.
389
390// Declare Traverse*() for all concrete Type classes.
391#define ABSTRACT_TYPE(CLASS, BASE)
392#define TYPE(CLASS, BASE) \
393 bool Traverse##CLASS##Type(CLASS##Type *T, bool TraverseQualifier);
394#include "clang/AST/TypeNodes.inc"
395 // The above header #undefs ABSTRACT_TYPE and TYPE upon exit.
396
397 // Define WalkUpFrom*() and empty Visit*() for all Type classes.
398 bool WalkUpFromType(Type *T) { return getDerived().VisitType(T); }
399 bool VisitType(Type *T) { return true; }
400#define TYPE(CLASS, BASE) \
401 bool WalkUpFrom##CLASS##Type(CLASS##Type *T) { \
402 TRY_TO(WalkUpFrom##BASE(T)); \
403 TRY_TO(Visit##CLASS##Type(T)); \
404 return true; \
405 } \
406 bool Visit##CLASS##Type(CLASS##Type *T) { return true; }
407#include "clang/AST/TypeNodes.inc"
408
409// ---- Methods on TypeLocs ----
410// FIXME: this currently just calls the matching Type methods
411
412// Declare Traverse*() for all concrete TypeLoc classes.
413#define ABSTRACT_TYPELOC(CLASS, BASE)
414#define TYPELOC(CLASS, BASE) \
415 bool Traverse##CLASS##TypeLoc(CLASS##TypeLoc TL, bool TraverseQualifier);
416#include "clang/AST/TypeLocNodes.def"
417 // The above header #undefs ABSTRACT_TYPELOC and TYPELOC upon exit.
418
419 // Define WalkUpFrom*() and empty Visit*() for all TypeLoc classes.
420 bool WalkUpFromTypeLoc(TypeLoc TL) { return getDerived().VisitTypeLoc(TL); }
421 bool VisitTypeLoc(TypeLoc TL) { return true; }
422
423 // QualifiedTypeLoc and UnqualTypeLoc are not declared in
424 // TypeNodes.inc and thus need to be handled specially.
426 return getDerived().VisitUnqualTypeLoc(TL.getUnqualifiedLoc());
427 }
428 bool VisitQualifiedTypeLoc(QualifiedTypeLoc TL) { return true; }
430 return getDerived().VisitUnqualTypeLoc(TL.getUnqualifiedLoc());
431 }
432 bool VisitUnqualTypeLoc(UnqualTypeLoc TL) { return true; }
433
434// Note that BASE includes trailing 'Type' which CLASS doesn't.
435#define TYPE(CLASS, BASE) \
436 bool WalkUpFrom##CLASS##TypeLoc(CLASS##TypeLoc TL) { \
437 TRY_TO(WalkUpFrom##BASE##Loc(TL)); \
438 TRY_TO(Visit##CLASS##TypeLoc(TL)); \
439 return true; \
440 } \
441 bool Visit##CLASS##TypeLoc(CLASS##TypeLoc TL) { return true; }
442#include "clang/AST/TypeNodes.inc"
443
444// ---- Methods on Decls ----
445
446// Declare Traverse*() for all concrete Decl classes.
447#define ABSTRACT_DECL(DECL)
448#define DECL(CLASS, BASE) bool Traverse##CLASS##Decl(CLASS##Decl *D);
449#include "clang/AST/DeclNodes.inc"
450 // The above header #undefs ABSTRACT_DECL and DECL upon exit.
451
452 // Define WalkUpFrom*() and empty Visit*() for all Decl classes.
453 bool WalkUpFromDecl(Decl *D) { return getDerived().VisitDecl(D); }
454 bool VisitDecl(Decl *D) { return true; }
455#define DECL(CLASS, BASE) \
456 bool WalkUpFrom##CLASS##Decl(CLASS##Decl *D) { \
457 TRY_TO(WalkUpFrom##BASE(D)); \
458 TRY_TO(Visit##CLASS##Decl(D)); \
459 return true; \
460 } \
461 bool Visit##CLASS##Decl(CLASS##Decl *D) { return true; }
462#include "clang/AST/DeclNodes.inc"
463
465
466#define DEF_TRAVERSE_TMPL_INST(TMPLDECLKIND) \
467 bool TraverseTemplateInstantiations(TMPLDECLKIND##TemplateDecl *D);
470 DEF_TRAVERSE_TMPL_INST(Function)
471#undef DEF_TRAVERSE_TMPL_INST
472
474
479
481
482private:
483 // These are helper methods used by more than one Traverse* method.
484 bool TraverseTemplateParameterListHelper(TemplateParameterList *TPL);
485
486 // Traverses template parameter lists of either a DeclaratorDecl or TagDecl.
487 template <typename T>
488 bool TraverseDeclTemplateParameterLists(T *D);
489
490 bool TraverseTemplateTypeParamDeclConstraints(const TemplateTypeParmDecl *D);
491
492 bool TraverseTemplateArgumentLocsHelper(const TemplateArgumentLoc *TAL,
493 unsigned Count);
494 bool TraverseArrayTypeLocHelper(ArrayTypeLoc TL);
495 bool TraverseSubstPackTypeHelper(SubstPackType *T);
496 bool TraverseSubstPackTypeLocHelper(SubstPackTypeLoc TL);
497 bool TraverseRecordHelper(RecordDecl *D);
498 bool TraverseCXXRecordHelper(CXXRecordDecl *D);
499 bool TraverseDeclaratorHelper(DeclaratorDecl *D);
500 bool TraverseDeclContextHelper(DeclContext *DC);
501 bool TraverseFunctionHelper(FunctionDecl *D);
502 bool TraverseVarHelper(VarDecl *D);
503 bool TraverseOMPExecutableDirective(OMPExecutableDirective *S);
504 bool TraverseOMPLoopDirective(OMPLoopDirective *S);
505 bool TraverseOMPClause(OMPClause *C);
506 bool TraverseTagType(TagType *T, bool TraverseQualifier);
507 bool TraverseTagTypeLoc(TagTypeLoc TL, bool TraverseQualifier);
508#define GEN_CLANG_CLAUSE_CLASS
509#define CLAUSE_CLASS(Enum, Str, Class) bool Visit##Class(Class *C);
510#include "llvm/Frontend/OpenMP/OMP.inc"
511 /// Process clauses with list of variables.
512 template <typename T> bool VisitOMPClauseList(T *Node);
513 /// Process clauses with pre-initis.
514 bool VisitOMPClauseWithPreInit(OMPClauseWithPreInit *Node);
515 bool VisitOMPClauseWithPostUpdate(OMPClauseWithPostUpdate *Node);
516
517 bool PostVisitStmt(Stmt *S);
518 bool TraverseOpenACCConstructStmt(OpenACCConstructStmt *S);
519 bool
520 TraverseOpenACCAssociatedStmtConstruct(OpenACCAssociatedStmtConstruct *S);
521 bool VisitOpenACCClauseList(ArrayRef<const OpenACCClause *>);
522 bool VisitOpenACCClause(const OpenACCClause *);
523};
524
525template <typename Derived>
527 const TypeConstraint *C) {
529 TRY_TO(TraverseConceptReference(C->getConceptReference()));
530 return true;
531 }
532 if (Expr *IDC = C->getImmediatelyDeclaredConstraint()) {
533 TRY_TO(TraverseStmt(IDC));
534 } else {
535 // Avoid traversing the ConceptReference in the TypeConstraint
536 // if we have an immediately-declared-constraint, otherwise
537 // we'll end up visiting the concept and the arguments in
538 // the TC twice.
539 TRY_TO(TraverseConceptReference(C->getConceptReference()));
540 }
541 return true;
542}
543
544template <typename Derived>
547 switch (R->getKind()) {
549 return getDerived().TraverseConceptTypeRequirement(
553 return getDerived().TraverseConceptExprRequirement(
556 return getDerived().TraverseConceptNestedRequirement(
558 }
559 llvm_unreachable("unexpected case");
560}
561
562template <typename Derived>
564 DataRecursionQueue *Queue) {
565 // Top switch stmt: dispatch to TraverseFooStmt for each concrete FooStmt.
566 switch (S->getStmtClass()) {
568 break;
569#define ABSTRACT_STMT(STMT)
570#define STMT(CLASS, PARENT) \
571 case Stmt::CLASS##Class: \
572 return TRAVERSE_STMT_BASE(CLASS, CLASS, S, Queue);
573#include "clang/AST/StmtNodes.inc"
574 }
575
576 return true;
577}
578
579#undef DISPATCH_STMT
580
581template <typename Derived>
584 if (R->isSubstitutionFailure())
585 return true;
586 return getDerived().TraverseTypeLoc(R->getType()->getTypeLoc());
587}
588
589template <typename Derived>
594 auto &RetReq = R->getReturnTypeRequirement();
595 if (RetReq.isTypeConstraint()) {
597 TRY_TO(TraverseTemplateParameterListHelper(
598 RetReq.getTypeConstraintTemplateParameterList()));
599 } else {
600 // Template parameter list is implicit, visit constraint directly.
601 TRY_TO(TraverseTypeConstraint(RetReq.getTypeConstraint()));
602 }
603 }
604 return true;
605}
606
607template <typename Derived>
614
615template <typename Derived>
616bool RecursiveASTVisitor<Derived>::PostVisitStmt(Stmt *S) {
617 // In pre-order traversal mode, each Traverse##STMT method is responsible for
618 // calling WalkUpFrom. Therefore, if the user overrides Traverse##STMT and
619 // does not call the default implementation, the WalkUpFrom callback is not
620 // called. Post-order traversal mode should provide the same behavior
621 // regarding method overrides.
622 //
623 // In post-order traversal mode the Traverse##STMT method, when it receives a
624 // DataRecursionQueue, can't call WalkUpFrom after traversing children because
625 // it only enqueues the children and does not traverse them. TraverseStmt
626 // traverses the enqueued children, and we call WalkUpFrom here.
627 //
628 // However, to make pre-order and post-order modes identical with regards to
629 // whether they call WalkUpFrom at all, we call WalkUpFrom if and only if the
630 // user did not override the Traverse##STMT method. We implement the override
631 // check with isSameMethod calls below.
632
633 switch (S->getStmtClass()) {
635 break;
636#define ABSTRACT_STMT(STMT)
637#define STMT(CLASS, PARENT) \
638 case Stmt::CLASS##Class: \
639 if (::clang::detail::isSameMethod(&RecursiveASTVisitor::Traverse##CLASS, \
640 &Derived::Traverse##CLASS)) { \
641 TRY_TO(WalkUpFrom##CLASS(static_cast<CLASS *>(S))); \
642 } \
643 break;
644#define INITLISTEXPR(CLASS, PARENT) \
645 case Stmt::CLASS##Class: \
646 if (::clang::detail::isSameMethod(&RecursiveASTVisitor::Traverse##CLASS, \
647 &Derived::Traverse##CLASS)) { \
648 auto ILE = static_cast<CLASS *>(S); \
649 if (auto Syn = ILE->isSemanticForm() ? ILE->getSyntacticForm() : ILE) \
650 TRY_TO(WalkUpFrom##CLASS(Syn)); \
651 if (auto Sem = ILE->isSemanticForm() ? ILE : ILE->getSemanticForm()) \
652 TRY_TO(WalkUpFrom##CLASS(Sem)); \
653 } \
654 break;
655#include "clang/AST/StmtNodes.inc"
656 }
657
658 return true;
659}
660
661#undef DISPATCH_STMT
662
663// Inlining this method can lead to large code size and compile-time increases
664// without any benefit to runtime performance.
665template <typename Derived>
666LLVM_ATTRIBUTE_NOINLINE bool
668 if (!S)
669 return true;
670
671 if (Queue) {
672 Queue->push_back({S, false});
673 return true;
674 }
675
677 LocalQueue.push_back({S, false});
678
679 while (!LocalQueue.empty()) {
680 auto &CurrSAndVisited = LocalQueue.back();
681 Stmt *CurrS = CurrSAndVisited.getPointer();
682 bool Visited = CurrSAndVisited.getInt();
683 if (Visited) {
684 LocalQueue.pop_back();
687 TRY_TO(PostVisitStmt(CurrS));
688 }
689 continue;
690 }
691
692 if (getDerived().dataTraverseStmtPre(CurrS)) {
693 CurrSAndVisited.setInt(true);
694 size_t N = LocalQueue.size();
695 TRY_TO(dataTraverseNode(CurrS, &LocalQueue));
696 // Process new children in the order they were added.
697 std::reverse(LocalQueue.begin() + N, LocalQueue.end());
698 } else {
699 LocalQueue.pop_back();
700 }
701 }
702
703 return true;
704}
705
706template <typename Derived>
708 bool TraverseQualifier) {
709 if (T.isNull())
710 return true;
711
712 switch (T->getTypeClass()) {
713#define ABSTRACT_TYPE(CLASS, BASE)
714#define TYPE(CLASS, BASE) \
715 case Type::CLASS: \
716 return getDerived().Traverse##CLASS##Type( \
717 static_cast<CLASS##Type *>(const_cast<Type *>(T.getTypePtr())), \
718 TraverseQualifier);
719#include "clang/AST/TypeNodes.inc"
720 }
721
722 return true;
723}
724
725template <typename Derived>
727 bool TraverseQualifier) {
728 if (TL.isNull())
729 return true;
730
731 switch (TL.getTypeLocClass()) {
732#define ABSTRACT_TYPELOC(CLASS, BASE)
733#define TYPELOC(CLASS, BASE) \
734 case TypeLoc::CLASS: \
735 return getDerived().Traverse##CLASS##TypeLoc(TL.castAs<CLASS##TypeLoc>(), \
736 TraverseQualifier);
737#include "clang/AST/TypeLocNodes.def"
738 }
739
740 return true;
741}
742
743// Define the Traverse*Attr(Attr* A) methods
744#define VISITORCLASS RecursiveASTVisitor
745#include "clang/AST/AttrVisitor.inc"
746#undef VISITORCLASS
747
748template <typename Derived>
750 if (!D)
751 return true;
752
753 // As a syntax visitor, by default we want to ignore declarations for
754 // implicit declarations (ones not typed explicitly by the user).
756 if (D->isImplicit()) {
757 // For an implicit template type parameter, its type constraints are not
758 // implicit and are not represented anywhere else. We still need to visit
759 // them.
760 if (auto *TTPD = dyn_cast<TemplateTypeParmDecl>(D))
761 return TraverseTemplateTypeParamDeclConstraints(TTPD);
762 return true;
763 }
764
765 // Deduction guides for alias templates are always synthesized, so they
766 // should not be traversed unless shouldVisitImplicitCode() returns true.
767 //
768 // It's important to note that checking the implicit bit is not efficient
769 // for the alias case. For deduction guides synthesized from explicit
770 // user-defined deduction guides, we must maintain the explicit bit to
771 // ensure correct overload resolution.
772 if (auto *FTD = dyn_cast<FunctionTemplateDecl>(D))
773 if (llvm::isa_and_present<TypeAliasTemplateDecl>(
774 FTD->getDeclName().getCXXDeductionGuideTemplate()))
775 return true;
776 }
777
778 switch (D->getKind()) {
779#define ABSTRACT_DECL(DECL)
780#define DECL(CLASS, BASE) \
781 case Decl::CLASS: \
782 if (!getDerived().Traverse##CLASS##Decl(static_cast<CLASS##Decl *>(D))) \
783 return false; \
784 break;
785#include "clang/AST/DeclNodes.inc"
786 }
787 return true;
788}
789
790template <typename Derived>
793 switch (NNS.getKind()) {
797 return true;
800 return true;
802 auto *T = const_cast<Type *>(NNS.getAsType());
803 TRY_TO(TraverseNestedNameSpecifier(T->getPrefix()));
804 TRY_TO(TraverseType(QualType(T, 0), /*TraverseQualifier=*/false));
805 return true;
806 }
807 }
808 llvm_unreachable("unhandled kind");
809}
810
811template <typename Derived>
833
834template <typename Derived>
862
863template <typename Derived>
865 if (DependentTemplateName *DTN = Template.getAsDependentTemplateName()) {
866 TRY_TO(TraverseNestedNameSpecifier(DTN->getQualifier()));
867 } else if (QualifiedTemplateName *QTN =
868 Template.getAsQualifiedTemplateName()) {
869 if (QTN->getQualifier()) {
870 TRY_TO(TraverseNestedNameSpecifier(QTN->getQualifier()));
871 }
872 }
873
874 return true;
875}
876
877template <typename Derived>
879 const TemplateArgument &Arg) {
880 switch (Arg.getKind()) {
886 return true;
887
889 return getDerived().TraverseType(Arg.getAsType());
890
893 return getDerived().TraverseTemplateName(
895
897 return getDerived().TraverseStmt(Arg.getAsExpr());
898
900 return getDerived().TraverseTemplateArguments(Arg.pack_elements());
901 }
902
903 return true;
904}
905
906// FIXME: no template name location?
907// FIXME: no source locations for a template argument pack?
908template <typename Derived>
910 const TemplateArgumentLoc &ArgLoc) {
911 const TemplateArgument &Arg = ArgLoc.getArgument();
912
913 switch (Arg.getKind()) {
919 return true;
920
922 // FIXME: how can TSI ever be NULL?
923 if (TypeSourceInfo *TSI = ArgLoc.getTypeSourceInfo())
924 return getDerived().TraverseTypeLoc(TSI->getTypeLoc());
925 else
926 return getDerived().TraverseType(Arg.getAsType());
927 }
928
931 if (ArgLoc.getTemplateQualifierLoc())
933 ArgLoc.getTemplateQualifierLoc()));
934 return getDerived().TraverseTemplateName(
936
938 return getDerived().TraverseStmt(ArgLoc.getSourceExpression());
939
941 return getDerived().TraverseTemplateArguments(Arg.pack_elements());
942 }
943
944 return true;
945}
946
947template <typename Derived>
950 for (const TemplateArgument &Arg : Args)
952
953 return true;
954}
955
956template <typename Derived>
959 if (TypeSourceInfo *TInfo = Init->getTypeSourceInfo())
960 TRY_TO(TraverseTypeLoc(TInfo->getTypeLoc()));
961
962 if (Init->isWritten() || getDerived().shouldVisitImplicitCode())
963 TRY_TO(TraverseStmt(Init->getInit()));
964
965 return true;
966}
967
968template <typename Derived>
969bool
971 const LambdaCapture *C,
972 Expr *Init) {
973 if (LE->isInitCapture(C))
974 TRY_TO(TraverseDecl(C->getCapturedVar()));
975 else
977 return true;
978}
979
980// ----------------- Type traversal -----------------
981
982// This macro makes available a variable T, the passed-in type.
983#define DEF_TRAVERSE_TYPE(TYPE, CODE) \
984 template <typename Derived> \
985 bool RecursiveASTVisitor<Derived>::Traverse##TYPE(TYPE *T, \
986 bool TraverseQualifier) { \
987 if (!getDerived().shouldTraversePostOrder()) \
988 TRY_TO(WalkUpFrom##TYPE(T)); \
989 { \
990 CODE; \
991 } \
992 if (getDerived().shouldTraversePostOrder()) \
993 TRY_TO(WalkUpFrom##TYPE(T)); \
994 return true; \
995 }
996
997DEF_TRAVERSE_TYPE(BuiltinType, {})
998
999DEF_TRAVERSE_TYPE(ComplexType, { TRY_TO(TraverseType(T->getElementType())); })
1000
1001DEF_TRAVERSE_TYPE(PointerType, { TRY_TO(TraverseType(T->getPointeeType())); })
1002
1004 { TRY_TO(TraverseType(T->getPointeeType())); })
1005
1006DEF_TRAVERSE_TYPE(LValueReferenceType,
1007 { TRY_TO(TraverseType(T->getPointeeType())); })
1008
1010 { TRY_TO(TraverseType(T->getPointeeType())); })
1011
1012DEF_TRAVERSE_TYPE(MemberPointerType, {
1013 NestedNameSpecifier Qualifier =
1014 T->isSugared() ? cast<MemberPointerType>(T->getCanonicalTypeUnqualified())
1015 ->getQualifier()
1016 : T->getQualifier();
1017 TRY_TO(TraverseNestedNameSpecifier(Qualifier));
1018 TRY_TO(TraverseType(T->getPointeeType()));
1019})
1020
1021DEF_TRAVERSE_TYPE(AdjustedType, { TRY_TO(TraverseType(T->getOriginalType())); })
1022
1023DEF_TRAVERSE_TYPE(DecayedType, { TRY_TO(TraverseType(T->getOriginalType())); })
1024
1026 TRY_TO(TraverseType(T->getElementType()));
1027 if (T->getSizeExpr())
1028 TRY_TO(TraverseStmt(const_cast<Expr*>(T->getSizeExpr())));
1029})
1030
1031DEF_TRAVERSE_TYPE(ArrayParameterType, {
1032 TRY_TO(TraverseType(T->getElementType()));
1033 if (T->getSizeExpr())
1034 TRY_TO(TraverseStmt(const_cast<Expr *>(T->getSizeExpr())));
1035})
1036
1038 { TRY_TO(TraverseType(T->getElementType())); })
1039
1040DEF_TRAVERSE_TYPE(VariableArrayType, {
1041 TRY_TO(TraverseType(T->getElementType()));
1042 TRY_TO(TraverseStmt(T->getSizeExpr()));
1043})
1044
1046 TRY_TO(TraverseType(T->getElementType()));
1047 if (T->getSizeExpr())
1048 TRY_TO(TraverseStmt(T->getSizeExpr()));
1049})
1050
1051DEF_TRAVERSE_TYPE(DependentAddressSpaceType, {
1052 TRY_TO(TraverseStmt(T->getAddrSpaceExpr()));
1053 TRY_TO(TraverseType(T->getPointeeType()));
1054})
1055
1057 if (T->getSizeExpr())
1058 TRY_TO(TraverseStmt(T->getSizeExpr()));
1059 TRY_TO(TraverseType(T->getElementType()));
1060})
1061
1062DEF_TRAVERSE_TYPE(DependentSizedExtVectorType, {
1063 if (T->getSizeExpr())
1064 TRY_TO(TraverseStmt(T->getSizeExpr()));
1065 TRY_TO(TraverseType(T->getElementType()));
1066})
1067
1068DEF_TRAVERSE_TYPE(VectorType, { TRY_TO(TraverseType(T->getElementType())); })
1069
1070DEF_TRAVERSE_TYPE(ExtVectorType, { TRY_TO(TraverseType(T->getElementType())); })
1071
1073 { TRY_TO(TraverseType(T->getElementType())); })
1074
1075DEF_TRAVERSE_TYPE(DependentSizedMatrixType, {
1076 if (T->getRowExpr())
1077 TRY_TO(TraverseStmt(T->getRowExpr()));
1078 if (T->getColumnExpr())
1079 TRY_TO(TraverseStmt(T->getColumnExpr()));
1080 TRY_TO(TraverseType(T->getElementType()));
1081})
1082
1084 { TRY_TO(TraverseType(T->getReturnType())); })
1085
1086DEF_TRAVERSE_TYPE(FunctionProtoType, {
1087 TRY_TO(TraverseType(T->getReturnType()));
1088
1089 for (const auto &A : T->param_types()) {
1090 TRY_TO(TraverseType(A));
1091 }
1092
1093 for (const auto &E : T->exceptions()) {
1094 TRY_TO(TraverseType(E));
1095 }
1096
1097 if (Expr *NE = T->getNoexceptExpr())
1098 TRY_TO(TraverseStmt(NE));
1099})
1100
1102 if (TraverseQualifier)
1103 TRY_TO(TraverseNestedNameSpecifier(T->getQualifier()));
1104})
1105DEF_TRAVERSE_TYPE(UnresolvedUsingType, {
1106 if (TraverseQualifier)
1107 TRY_TO(TraverseNestedNameSpecifier(T->getQualifier()));
1108})
1110 if (TraverseQualifier)
1111 TRY_TO(TraverseNestedNameSpecifier(T->getQualifier()));
1112})
1113
1114DEF_TRAVERSE_TYPE(TypeOfExprType,
1115 { TRY_TO(TraverseStmt(T->getUnderlyingExpr())); })
1116
1117DEF_TRAVERSE_TYPE(TypeOfType, { TRY_TO(TraverseType(T->getUnmodifiedType())); })
1118
1119DEF_TRAVERSE_TYPE(DecltypeType,
1120 { TRY_TO(TraverseStmt(T->getUnderlyingExpr())); })
1121
1122DEF_TRAVERSE_TYPE(PackIndexingType, {
1123 TRY_TO(TraverseType(T->getPattern()));
1124 TRY_TO(TraverseStmt(T->getIndexExpr()));
1125})
1126
1127DEF_TRAVERSE_TYPE(UnaryTransformType, {
1128 TRY_TO(TraverseType(T->getBaseType()));
1129 TRY_TO(TraverseType(T->getUnderlyingType()));
1130})
1131
1133 TRY_TO(TraverseType(T->getDeducedType()));
1134 if (T->isConstrained()) {
1135 TRY_TO(TraverseTemplateArguments(T->getTypeConstraintArguments()));
1136 }
1137})
1138
1139DEF_TRAVERSE_TYPE(TemplateTypeParmType, {})
1140DEF_TRAVERSE_TYPE(SubstTemplateTypeParmType, {
1141 TRY_TO(TraverseType(T->getReplacementType()));
1142})
1143DEF_TRAVERSE_TYPE(SubstTemplateTypeParmPackType,
1144 { TRY_TO(TraverseSubstPackTypeHelper(T)); })
1145DEF_TRAVERSE_TYPE(SubstBuiltinTemplatePackType,
1146 { TRY_TO(TraverseSubstPackTypeHelper(T)); })
1147
1148DEF_TRAVERSE_TYPE(AttributedType,
1149 { TRY_TO(TraverseType(T->getModifiedType())); })
1150
1151DEF_TRAVERSE_TYPE(CountAttributedType, {
1152 if (T->getCountExpr())
1153 TRY_TO(TraverseStmt(T->getCountExpr()));
1154 TRY_TO(TraverseType(T->desugar()));
1155})
1156
1157DEF_TRAVERSE_TYPE(BTFTagAttributedType,
1158 { TRY_TO(TraverseType(T->getWrappedType())); })
1159
1160DEF_TRAVERSE_TYPE(HLSLAttributedResourceType,
1161 { TRY_TO(TraverseType(T->getWrappedType())); })
1162
1163DEF_TRAVERSE_TYPE(HLSLInlineSpirvType, {
1164 for (auto &Operand : T->getOperands()) {
1165 if (Operand.isConstant() || Operand.isType()) {
1166 TRY_TO(TraverseType(Operand.getResultType()));
1167 }
1168 }
1169})
1170
1171DEF_TRAVERSE_TYPE(ParenType, { TRY_TO(TraverseType(T->getInnerType())); })
1172
1174 { TRY_TO(TraverseType(T->getUnderlyingType())); })
1175
1176template <typename Derived>
1177bool RecursiveASTVisitor<Derived>::TraverseTagType(TagType *T,
1178 bool TraverseQualifier) {
1179 if (TraverseQualifier)
1180 TRY_TO(TraverseNestedNameSpecifier(T->getQualifier()));
1181 return true;
1182}
1183
1184DEF_TRAVERSE_TYPE(EnumType, { TRY_TO(TraverseTagType(T, TraverseQualifier)); })
1185DEF_TRAVERSE_TYPE(RecordType,
1186 { TRY_TO(TraverseTagType(T, TraverseQualifier)); })
1187DEF_TRAVERSE_TYPE(InjectedClassNameType,
1188 { TRY_TO(TraverseTagType(T, TraverseQualifier)); })
1189
1190DEF_TRAVERSE_TYPE(DependentNameType, {
1191 if (TraverseQualifier)
1192 TRY_TO(TraverseNestedNameSpecifier(T->getQualifier()));
1193})
1194
1195DEF_TRAVERSE_TYPE(TemplateSpecializationType, {
1196 if (TraverseQualifier) {
1197 TRY_TO(TraverseTemplateName(T->getTemplateName()));
1198 } else {
1199 // FIXME: Try to preserve the rest of the template name.
1200 TRY_TO(TraverseTemplateName(TemplateName(
1201 T->getTemplateName().getAsTemplateDecl(/*IgnoreDeduced=*/true))));
1202 }
1203 TRY_TO(TraverseTemplateArguments(T->template_arguments()));
1204})
1205
1206DEF_TRAVERSE_TYPE(DeducedTemplateSpecializationType, {
1207 if (TraverseQualifier) {
1208 TRY_TO(TraverseTemplateName(T->getTemplateName()));
1209 } else {
1210 // FIXME: Try to preserve the rest of the template name.
1211 TRY_TO(TraverseTemplateName(TemplateName(
1212 T->getTemplateName().getAsTemplateDecl(/*IgnoreDeduced=*/true))));
1213 }
1214 TRY_TO(TraverseType(T->getDeducedType()));
1215})
1216
1217DEF_TRAVERSE_TYPE(PackExpansionType, { TRY_TO(TraverseType(T->getPattern())); })
1218
1219DEF_TRAVERSE_TYPE(ObjCTypeParamType, {})
1220
1222
1223DEF_TRAVERSE_TYPE(ObjCObjectType, {
1224 // We have to watch out here because an ObjCInterfaceType's base
1225 // type is itself.
1226 if (T->getBaseType().getTypePtr() != T)
1227 TRY_TO(TraverseType(T->getBaseType()));
1228 for (auto typeArg : T->getTypeArgsAsWritten()) {
1229 TRY_TO(TraverseType(typeArg));
1230 }
1231})
1232
1234 { TRY_TO(TraverseType(T->getPointeeType())); })
1235
1236DEF_TRAVERSE_TYPE(AtomicType, { TRY_TO(TraverseType(T->getValueType())); })
1237
1238DEF_TRAVERSE_TYPE(PipeType, { TRY_TO(TraverseType(T->getElementType())); })
1239
1240DEF_TRAVERSE_TYPE(BitIntType, {})
1242 { TRY_TO(TraverseStmt(T->getNumBitsExpr())); })
1243
1245
1246#undef DEF_TRAVERSE_TYPE
1247
1248// ----------------- TypeLoc traversal -----------------
1249
1250// This macro makes available a variable TL, the passed-in TypeLoc.
1251// If requested, it calls WalkUpFrom* for the Type in the given TypeLoc,
1252// in addition to WalkUpFrom* for the TypeLoc itself, such that existing
1253// clients that override the WalkUpFrom*Type() and/or Visit*Type() methods
1254// continue to work.
1256 template <typename Derived> \
1257 bool RecursiveASTVisitor<Derived>::Traverse##TYPE##Loc( \
1258 TYPE##Loc TL, bool TraverseQualifier) { \
1259 if (!getDerived().shouldTraversePostOrder()) { \
1260 TRY_TO(WalkUpFrom##TYPE##Loc(TL)); \
1261 if (getDerived().shouldWalkTypesOfTypeLocs()) \
1262 TRY_TO(WalkUpFrom##TYPE(const_cast<TYPE *>(TL.getTypePtr()))); \
1263 } \
1264 { \
1265 CODE; \
1266 } \
1267 if (getDerived().shouldTraversePostOrder()) { \
1268 TRY_TO(WalkUpFrom##TYPE##Loc(TL)); \
1269 if (getDerived().shouldWalkTypesOfTypeLocs()) \
1270 TRY_TO(WalkUpFrom##TYPE(const_cast<TYPE *>(TL.getTypePtr()))); \
1271 } \
1272 return true; \
1273 }
1274
1275template <typename Derived>
1277 QualifiedTypeLoc TL, bool TraverseQualifier) {
1278 assert(TraverseQualifier &&
1279 "Qualifiers should never occur within NestedNameSpecifiers");
1280 // Move this over to the 'main' typeloc tree. Note that this is a
1281 // move -- we pretend that we were really looking at the unqualified
1282 // typeloc all along -- rather than a recursion, so we don't follow
1283 // the normal CRTP plan of going through
1284 // getDerived().TraverseTypeLoc. If we did, we'd be traversing
1285 // twice for the same type (once as a QualifiedTypeLoc version of
1286 // the type, once as an UnqualifiedTypeLoc version of the type),
1287 // which in effect means we'd call VisitTypeLoc twice with the
1288 // 'same' type. This solves that problem, at the cost of never
1289 // seeing the qualified version of the type (unless the client
1290 // subclasses TraverseQualifiedTypeLoc themselves). It's not a
1291 // perfect solution. A perfect solution probably requires making
1292 // QualifiedTypeLoc a wrapper around TypeLoc -- like QualType is a
1293 // wrapper around Type* -- rather than being its own class in the
1294 // type hierarchy.
1295 return TraverseTypeLoc(TL.getUnqualifiedLoc());
1296}
1297
1299
1300// FIXME: ComplexTypeLoc is unfinished
1302 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1303})
1304
1305DEF_TRAVERSE_TYPELOC(PointerType,
1306 { TRY_TO(TraverseTypeLoc(TL.getPointeeLoc())); })
1307
1309 { TRY_TO(TraverseTypeLoc(TL.getPointeeLoc())); })
1310
1311DEF_TRAVERSE_TYPELOC(LValueReferenceType,
1312 { TRY_TO(TraverseTypeLoc(TL.getPointeeLoc())); })
1313
1315 { TRY_TO(TraverseTypeLoc(TL.getPointeeLoc())); })
1316
1317// We traverse this in the type case as well, but how is it not reached through
1318// the pointee type?
1319DEF_TRAVERSE_TYPELOC(MemberPointerType, {
1320 if (NestedNameSpecifierLoc QL = TL.getQualifierLoc())
1322 else
1323 TRY_TO(TraverseNestedNameSpecifier(TL.getTypePtr()->getQualifier()));
1324 TRY_TO(TraverseTypeLoc(TL.getPointeeLoc()));
1325})
1326
1328 { TRY_TO(TraverseTypeLoc(TL.getOriginalLoc())); })
1329
1330DEF_TRAVERSE_TYPELOC(DecayedType,
1331 { TRY_TO(TraverseTypeLoc(TL.getOriginalLoc())); })
1332
1333template <typename Derived>
1334bool RecursiveASTVisitor<Derived>::TraverseArrayTypeLocHelper(ArrayTypeLoc TL) {
1335 // This isn't available for ArrayType, but is for the ArrayTypeLoc.
1336 TRY_TO(TraverseStmt(TL.getSizeExpr()));
1337 return true;
1338}
1339
1341 TRY_TO(TraverseTypeLoc(TL.getElementLoc()));
1342 TRY_TO(TraverseArrayTypeLocHelper(TL));
1343})
1344
1345DEF_TRAVERSE_TYPELOC(ArrayParameterType, {
1346 TRY_TO(TraverseTypeLoc(TL.getElementLoc()));
1347 TRY_TO(TraverseArrayTypeLocHelper(TL));
1348})
1349
1351 TRY_TO(TraverseTypeLoc(TL.getElementLoc()));
1352 TRY_TO(TraverseArrayTypeLocHelper(TL));
1353})
1354
1355DEF_TRAVERSE_TYPELOC(VariableArrayType, {
1356 TRY_TO(TraverseTypeLoc(TL.getElementLoc()));
1357 TRY_TO(TraverseArrayTypeLocHelper(TL));
1358})
1359
1361 TRY_TO(TraverseTypeLoc(TL.getElementLoc()));
1362 TRY_TO(TraverseArrayTypeLocHelper(TL));
1363})
1364
1365DEF_TRAVERSE_TYPELOC(DependentAddressSpaceType, {
1366 TRY_TO(TraverseStmt(TL.getTypePtr()->getAddrSpaceExpr()));
1367 TRY_TO(TraverseType(TL.getTypePtr()->getPointeeType()));
1368})
1369
1370// FIXME: order? why not size expr first?
1371// FIXME: base VectorTypeLoc is unfinished
1373 if (TL.getTypePtr()->getSizeExpr())
1374 TRY_TO(TraverseStmt(TL.getTypePtr()->getSizeExpr()));
1375 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1376})
1377
1378// FIXME: VectorTypeLoc is unfinished
1379DEF_TRAVERSE_TYPELOC(VectorType, {
1380 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1381})
1382
1384 if (TL.getTypePtr()->getSizeExpr())
1385 TRY_TO(TraverseStmt(TL.getTypePtr()->getSizeExpr()));
1386 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1387})
1388
1389// FIXME: size and attributes
1390// FIXME: base VectorTypeLoc is unfinished
1391DEF_TRAVERSE_TYPELOC(ExtVectorType, {
1392 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1393})
1394
1396 TRY_TO(TraverseStmt(TL.getAttrRowOperand()));
1397 TRY_TO(TraverseStmt(TL.getAttrColumnOperand()));
1398 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1399})
1400
1401DEF_TRAVERSE_TYPELOC(DependentSizedMatrixType, {
1402 TRY_TO(TraverseStmt(TL.getAttrRowOperand()));
1403 TRY_TO(TraverseStmt(TL.getAttrColumnOperand()));
1404 TRY_TO(TraverseType(TL.getTypePtr()->getElementType()));
1405})
1406
1408 { TRY_TO(TraverseTypeLoc(TL.getReturnLoc())); })
1409
1410// FIXME: location of exception specifications (attributes?)
1411DEF_TRAVERSE_TYPELOC(FunctionProtoType, {
1412 TRY_TO(TraverseTypeLoc(TL.getReturnLoc()));
1413
1414 const FunctionProtoType *T = TL.getTypePtr();
1415
1416 for (unsigned I = 0, E = TL.getNumParams(); I != E; ++I) {
1417 if (TL.getParam(I)) {
1418 TRY_TO(TraverseDecl(TL.getParam(I)));
1419 } else if (I < T->getNumParams()) {
1420 TRY_TO(TraverseType(T->getParamType(I)));
1421 }
1422 }
1423
1424 for (const auto &E : T->exceptions()) {
1425 TRY_TO(TraverseType(E));
1426 }
1427
1428 if (Expr *NE = T->getNoexceptExpr())
1429 TRY_TO(TraverseStmt(NE));
1430})
1431
1433 if (NestedNameSpecifierLoc QualifierLoc = TL.getQualifierLoc();
1434 TraverseQualifier && QualifierLoc)
1436})
1437DEF_TRAVERSE_TYPELOC(UnresolvedUsingType, {
1438 if (NestedNameSpecifierLoc QualifierLoc = TL.getQualifierLoc();
1439 TraverseQualifier && QualifierLoc)
1441})
1443 if (NestedNameSpecifierLoc QualifierLoc = TL.getQualifierLoc();
1444 TraverseQualifier && QualifierLoc)
1446})
1447
1448DEF_TRAVERSE_TYPELOC(TypeOfExprType,
1449 { TRY_TO(TraverseStmt(TL.getUnderlyingExpr())); })
1450
1452 TRY_TO(TraverseTypeLoc(TL.getUnmodifiedTInfo()->getTypeLoc()));
1453})
1454
1455// FIXME: location of underlying expr
1456DEF_TRAVERSE_TYPELOC(DecltypeType, {
1457 TRY_TO(TraverseStmt(TL.getTypePtr()->getUnderlyingExpr()));
1458})
1459
1460DEF_TRAVERSE_TYPELOC(PackIndexingType, {
1461 TRY_TO(TraverseType(TL.getPattern()));
1462 TRY_TO(TraverseStmt(TL.getTypePtr()->getIndexExpr()));
1463})
1464
1465DEF_TRAVERSE_TYPELOC(UnaryTransformType, {
1466 TRY_TO(TraverseTypeLoc(TL.getUnderlyingTInfo()->getTypeLoc()));
1467})
1468
1470 TRY_TO(TraverseType(TL.getTypePtr()->getDeducedType()));
1471 if (TL.isConstrained()) {
1472 TRY_TO(TraverseConceptReference(TL.getConceptReference()));
1473 }
1474})
1475
1476DEF_TRAVERSE_TYPELOC(TemplateTypeParmType, {})
1477DEF_TRAVERSE_TYPELOC(SubstTemplateTypeParmType, {
1478 TRY_TO(TraverseType(TL.getTypePtr()->getReplacementType()));
1479})
1480
1481template <typename Derived>
1482bool RecursiveASTVisitor<Derived>::TraverseSubstPackTypeLocHelper(
1483 SubstPackTypeLoc TL) {
1484 TRY_TO(TraverseTemplateArgument(TL.getTypePtr()->getArgumentPack()));
1485 return true;
1486}
1487
1488template <typename Derived>
1489bool RecursiveASTVisitor<Derived>::TraverseSubstPackTypeHelper(
1490 SubstPackType *T) {
1491 TRY_TO(TraverseTemplateArgument(T->getArgumentPack()));
1492 return true;
1493}
1494
1495DEF_TRAVERSE_TYPELOC(SubstTemplateTypeParmPackType,
1496 { TRY_TO(TraverseSubstPackTypeLocHelper(TL)); })
1497
1498DEF_TRAVERSE_TYPELOC(SubstBuiltinTemplatePackType,
1499 { TRY_TO(TraverseSubstPackTypeLocHelper(TL)); })
1500
1501DEF_TRAVERSE_TYPELOC(ParenType, { TRY_TO(TraverseTypeLoc(TL.getInnerLoc())); })
1502
1503DEF_TRAVERSE_TYPELOC(MacroQualifiedType,
1504 { TRY_TO(TraverseTypeLoc(TL.getInnerLoc())); })
1505
1507 { TRY_TO(TraverseTypeLoc(TL.getModifiedLoc())); })
1508
1509DEF_TRAVERSE_TYPELOC(CountAttributedType,
1510 { TRY_TO(TraverseTypeLoc(TL.getInnerLoc())); })
1511
1512DEF_TRAVERSE_TYPELOC(BTFTagAttributedType,
1513 { TRY_TO(TraverseTypeLoc(TL.getWrappedLoc())); })
1514
1515DEF_TRAVERSE_TYPELOC(HLSLAttributedResourceType,
1516 { TRY_TO(TraverseTypeLoc(TL.getWrappedLoc())); })
1517
1518DEF_TRAVERSE_TYPELOC(HLSLInlineSpirvType,
1519 { TRY_TO(TraverseType(TL.getType())); })
1520
1521template <typename Derived>
1522bool RecursiveASTVisitor<Derived>::TraverseTagTypeLoc(TagTypeLoc TL,
1523 bool TraverseQualifier) {
1524 if (NestedNameSpecifierLoc QualifierLoc = TL.getQualifierLoc();
1525 TraverseQualifier && QualifierLoc)
1527 return true;
1528}
1529
1531 { TRY_TO(TraverseTagTypeLoc(TL, TraverseQualifier)); })
1532DEF_TRAVERSE_TYPELOC(RecordType,
1533 { TRY_TO(TraverseTagTypeLoc(TL, TraverseQualifier)); })
1534DEF_TRAVERSE_TYPELOC(InjectedClassNameType,
1535 { TRY_TO(TraverseTagTypeLoc(TL, TraverseQualifier)); })
1536
1537DEF_TRAVERSE_TYPELOC(DependentNameType, {
1538 if (TraverseQualifier)
1539 TRY_TO(TraverseNestedNameSpecifierLoc(TL.getQualifierLoc()));
1540})
1541
1542DEF_TRAVERSE_TYPELOC(TemplateSpecializationType, {
1543 if (TraverseQualifier)
1544 TRY_TO(TraverseNestedNameSpecifierLoc(TL.getQualifierLoc()));
1545
1546 // FIXME: Try to preserve the rest of the template name.
1547 TRY_TO(TraverseTemplateName(
1548 TemplateName(TL.getTypePtr()->getTemplateName().getAsTemplateDecl(
1549 /*IgnoreDeduced=*/true))));
1550
1551 for (unsigned I = 0, E = TL.getNumArgs(); I != E; ++I) {
1552 TRY_TO(TraverseTemplateArgumentLoc(TL.getArgLoc(I)));
1553 }
1554})
1555
1556DEF_TRAVERSE_TYPELOC(DeducedTemplateSpecializationType, {
1557 if (TraverseQualifier)
1558 TRY_TO(TraverseNestedNameSpecifierLoc(TL.getQualifierLoc()));
1559
1560 const auto *T = TL.getTypePtr();
1561 // FIXME: Try to preserve the rest of the template name.
1563 TraverseTemplateName(TemplateName(T->getTemplateName().getAsTemplateDecl(
1564 /*IgnoreDeduced=*/true))));
1565
1566 TRY_TO(TraverseType(T->getDeducedType()));
1567})
1568
1569DEF_TRAVERSE_TYPELOC(PackExpansionType,
1570 { TRY_TO(TraverseTypeLoc(TL.getPatternLoc())); })
1571
1572DEF_TRAVERSE_TYPELOC(ObjCTypeParamType, {
1573 for (unsigned I = 0, N = TL.getNumProtocols(); I != N; ++I) {
1574 ObjCProtocolLoc ProtocolLoc(TL.getProtocol(I), TL.getProtocolLoc(I));
1575 TRY_TO(TraverseObjCProtocolLoc(ProtocolLoc));
1576 }
1577})
1578
1580
1581DEF_TRAVERSE_TYPELOC(ObjCObjectType, {
1582 // We have to watch out here because an ObjCInterfaceType's base
1583 // type is itself.
1584 if (TL.getTypePtr()->getBaseType().getTypePtr() != TL.getTypePtr())
1585 TRY_TO(TraverseTypeLoc(TL.getBaseLoc()));
1586 for (unsigned i = 0, n = TL.getNumTypeArgs(); i != n; ++i)
1587 TRY_TO(TraverseTypeLoc(TL.getTypeArgTInfo(i)->getTypeLoc()));
1588 for (unsigned I = 0, N = TL.getNumProtocols(); I != N; ++I) {
1589 ObjCProtocolLoc ProtocolLoc(TL.getProtocol(I), TL.getProtocolLoc(I));
1590 TRY_TO(TraverseObjCProtocolLoc(ProtocolLoc));
1591 }
1592})
1593
1595 { TRY_TO(TraverseTypeLoc(TL.getPointeeLoc())); })
1596
1597DEF_TRAVERSE_TYPELOC(AtomicType, { TRY_TO(TraverseTypeLoc(TL.getValueLoc())); })
1598
1599DEF_TRAVERSE_TYPELOC(PipeType, { TRY_TO(TraverseTypeLoc(TL.getValueLoc())); })
1600
1603 TRY_TO(TraverseStmt(TL.getTypePtr()->getNumBitsExpr()));
1604})
1605
1607
1609
1610// ----------------- Decl traversal -----------------
1611//
1612// For a Decl, we automate (in the DEF_TRAVERSE_DECL macro) traversing
1613// the children that come from the DeclContext associated with it.
1614// Therefore each Traverse* only needs to worry about children other
1615// than those.
1616
1617template <typename Derived>
1619 const Decl *Child) {
1620 // BlockDecls are traversed through BlockExprs,
1621 // CapturedDecls are traversed through CapturedStmts.
1622 if (isa<BlockDecl>(Child) || isa<CapturedDecl>(Child))
1623 return true;
1624 // Lambda classes are traversed through LambdaExprs.
1625 if (const CXXRecordDecl* Cls = dyn_cast<CXXRecordDecl>(Child))
1626 return Cls->isLambda();
1627 return false;
1628}
1629
1630template <typename Derived>
1631bool RecursiveASTVisitor<Derived>::TraverseDeclContextHelper(DeclContext *DC) {
1632 if (!DC)
1633 return true;
1634
1635 for (auto *Child : DC->decls()) {
1636 if (!canIgnoreChildDeclWhileTraversingDeclContext(Child))
1637 TRY_TO(TraverseDecl(Child));
1638 }
1639
1640 return true;
1641}
1642
1643// This macro makes available a variable D, the passed-in decl.
1644#define DEF_TRAVERSE_DECL(DECL, CODE) \
1645 template <typename Derived> \
1646 bool RecursiveASTVisitor<Derived>::Traverse##DECL(DECL *D) { \
1647 bool ShouldVisitChildren = true; \
1648 bool ReturnValue = true; \
1649 if (!getDerived().shouldTraversePostOrder()) \
1650 TRY_TO(WalkUpFrom##DECL(D)); \
1651 { CODE; } \
1652 if (ReturnValue && ShouldVisitChildren) \
1653 TRY_TO(TraverseDeclContextHelper(dyn_cast<DeclContext>(D))); \
1654 if (ReturnValue) { \
1655 /* Visit any attributes attached to this declaration. */ \
1656 for (auto *I : D->attrs()) \
1657 TRY_TO(getDerived().TraverseAttr(I)); \
1658 } \
1659 if (ReturnValue && getDerived().shouldTraversePostOrder()) \
1660 TRY_TO(WalkUpFrom##DECL(D)); \
1661 return ReturnValue; \
1662 }
1663
1665
1667 if (TypeSourceInfo *TInfo = D->getSignatureAsWritten())
1668 TRY_TO(TraverseTypeLoc(TInfo->getTypeLoc()));
1669 TRY_TO(TraverseStmt(D->getBody()));
1670 for (const auto &I : D->captures()) {
1671 if (I.hasCopyExpr()) {
1672 TRY_TO(TraverseStmt(I.getCopyExpr()));
1673 }
1674 }
1675 ShouldVisitChildren = false;
1676})
1677
1679 TRY_TO(TraverseStmt(D->getBody()));
1680 ShouldVisitChildren = false;
1681})
1682
1684 TRY_TO(TraverseStmt(D->getBody()));
1685 ShouldVisitChildren = false;
1686})
1687
1689
1691
1693
1695 TRY_TO(TraverseStmt(D->getTemporaryExpr()));
1696})
1697
1699 { TRY_TO(TraverseStmt(D->getAsmStringExpr())); })
1700
1701DEF_TRAVERSE_DECL(TopLevelStmtDecl, { TRY_TO(TraverseStmt(D->getStmt())); })
1702
1704
1706 // Friend is either decl or a type.
1707 if (D->getFriendType()) {
1708 TRY_TO(TraverseTypeLoc(D->getFriendType()->getTypeLoc()));
1709 // Traverse any CXXRecordDecl owned by this type, since
1710 // it will not be in the parent context:
1711 if (auto *TT = D->getFriendType()->getType()->getAs<TagType>();
1712 TT && TT->isTagOwned())
1713 TRY_TO(TraverseDecl(TT->getOriginalDecl()));
1714 } else {
1715 TRY_TO(TraverseDecl(D->getFriendDecl()));
1716 }
1717})
1718
1720 if (D->getFriendType())
1721 TRY_TO(TraverseTypeLoc(D->getFriendType()->getTypeLoc()));
1722 else
1723 TRY_TO(TraverseDecl(D->getFriendDecl()));
1724 for (unsigned I = 0, E = D->getNumTemplateParameters(); I < E; ++I) {
1725 TemplateParameterList *TPL = D->getTemplateParameterList(I);
1726 for (TemplateParameterList::iterator ITPL = TPL->begin(), ETPL = TPL->end();
1727 ITPL != ETPL; ++ITPL) {
1728 TRY_TO(TraverseDecl(*ITPL));
1729 }
1730 }
1731})
1732
1734
1736
1737DEF_TRAVERSE_DECL(ObjCPropertyImplDecl, {// FIXME: implement this
1738 })
1739
1741 TRY_TO(TraverseStmt(D->getAssertExpr()));
1742 TRY_TO(TraverseStmt(D->getMessage()));
1743})
1744
1746 // Code in an unnamed namespace shows up automatically in
1747 // decls_begin()/decls_end(). Thus we don't need to recurse on
1748 // D->getAnonymousNamespace().
1749
1750 // If the traversal scope is set, then consider them to be the children of
1751 // the TUDecl, rather than traversing (and loading?) all top-level decls.
1752 auto Scope = D->getASTContext().getTraversalScope();
1753 bool HasLimitedScope =
1754 Scope.size() != 1 || !isa<TranslationUnitDecl>(Scope.front());
1755 if (HasLimitedScope) {
1756 ShouldVisitChildren = false; // we'll do that here instead
1757 for (auto *Child : Scope) {
1758 if (!canIgnoreChildDeclWhileTraversingDeclContext(Child))
1759 TRY_TO(TraverseDecl(Child));
1760 }
1761 }
1762})
1763
1765
1767
1769
1771 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
1772
1773 // We shouldn't traverse an aliased namespace, since it will be
1774 // defined (and, therefore, traversed) somewhere else.
1775 ShouldVisitChildren = false;
1776})
1777
1778DEF_TRAVERSE_DECL(LabelDecl, {// There is no code in a LabelDecl.
1779 })
1780
1783 {// Code in an unnamed namespace shows up automatically in
1784 // decls_begin()/decls_end(). Thus we don't need to recurse on
1785 // D->getAnonymousNamespace().
1786 })
1787
1788DEF_TRAVERSE_DECL(ObjCCompatibleAliasDecl, {// FIXME: implement
1789 })
1790
1792 if (ObjCTypeParamList *typeParamList = D->getTypeParamList()) {
1793 for (auto typeParam : *typeParamList) {
1794 TRY_TO(TraverseObjCTypeParamDecl(typeParam));
1795 }
1796 }
1797 for (auto It : llvm::zip(D->protocols(), D->protocol_locs())) {
1798 ObjCProtocolLoc ProtocolLoc(std::get<0>(It), std::get<1>(It));
1799 TRY_TO(TraverseObjCProtocolLoc(ProtocolLoc));
1800 }
1801})
1802
1803DEF_TRAVERSE_DECL(ObjCCategoryImplDecl, {// FIXME: implement
1804 })
1805
1806DEF_TRAVERSE_DECL(ObjCImplementationDecl, {// FIXME: implement
1807 })
1808
1810 if (ObjCTypeParamList *typeParamList = D->getTypeParamListAsWritten()) {
1811 for (auto typeParam : *typeParamList) {
1812 TRY_TO(TraverseObjCTypeParamDecl(typeParam));
1813 }
1814 }
1815
1816 if (TypeSourceInfo *superTInfo = D->getSuperClassTInfo()) {
1817 TRY_TO(TraverseTypeLoc(superTInfo->getTypeLoc()));
1818 }
1819 if (D->isThisDeclarationADefinition()) {
1820 for (auto It : llvm::zip(D->protocols(), D->protocol_locs())) {
1821 ObjCProtocolLoc ProtocolLoc(std::get<0>(It), std::get<1>(It));
1822 TRY_TO(TraverseObjCProtocolLoc(ProtocolLoc));
1823 }
1824 }
1825})
1826
1828 if (D->isThisDeclarationADefinition()) {
1829 for (auto It : llvm::zip(D->protocols(), D->protocol_locs())) {
1830 ObjCProtocolLoc ProtocolLoc(std::get<0>(It), std::get<1>(It));
1831 TRY_TO(TraverseObjCProtocolLoc(ProtocolLoc));
1832 }
1833 }
1834})
1835
1837 if (D->getReturnTypeSourceInfo()) {
1838 TRY_TO(TraverseTypeLoc(D->getReturnTypeSourceInfo()->getTypeLoc()));
1839 }
1840 for (ParmVarDecl *Parameter : D->parameters()) {
1841 TRY_TO(TraverseDecl(Parameter));
1842 }
1843 if (D->isThisDeclarationADefinition()) {
1844 TRY_TO(TraverseStmt(D->getBody()));
1845 }
1846 ShouldVisitChildren = false;
1847})
1848
1850 if (D->hasExplicitBound()) {
1851 TRY_TO(TraverseTypeLoc(D->getTypeSourceInfo()->getTypeLoc()));
1852 // We shouldn't traverse D->getTypeForDecl(); it's a result of
1853 // declaring the type alias, not something that was written in the
1854 // source.
1855 }
1856})
1857
1859 if (D->getTypeSourceInfo())
1860 TRY_TO(TraverseTypeLoc(D->getTypeSourceInfo()->getTypeLoc()));
1861 else
1862 TRY_TO(TraverseType(D->getType()));
1863 ShouldVisitChildren = false;
1864})
1865
1867 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
1868 TRY_TO(TraverseDeclarationNameInfo(D->getNameInfo()));
1869})
1870
1872 { TRY_TO(TraverseTypeLoc(D->getEnumTypeLoc())); })
1873
1875
1877 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
1878})
1879
1881
1883
1885 for (auto *I : D->varlist()) {
1886 TRY_TO(TraverseStmt(I));
1887 }
1888})
1889
1891 for (auto *C : D->clauselists()) {
1892 TRY_TO(TraverseOMPClause(C));
1893 }
1894})
1895
1897 TRY_TO(TraverseStmt(D->getCombiner()));
1898 if (auto *Initializer = D->getInitializer())
1899 TRY_TO(TraverseStmt(Initializer));
1900 TRY_TO(TraverseType(D->getType()));
1901 return true;
1902})
1903
1905 for (auto *C : D->clauselists())
1906 TRY_TO(TraverseOMPClause(C));
1907 TRY_TO(TraverseType(D->getType()));
1908 return true;
1909})
1910
1911DEF_TRAVERSE_DECL(OMPCapturedExprDecl, { TRY_TO(TraverseVarHelper(D)); })
1912
1914 for (auto *I : D->varlist())
1915 TRY_TO(TraverseStmt(I));
1916 for (auto *C : D->clauselists())
1917 TRY_TO(TraverseOMPClause(C));
1918})
1919
1921 { TRY_TO(VisitOpenACCClauseList(D->clauses())); })
1922
1924 TRY_TO(TraverseStmt(D->getFunctionReference()));
1925 TRY_TO(VisitOpenACCClauseList(D->clauses()));
1926})
1927
1928// A helper method for TemplateDecl's children.
1929template <typename Derived>
1930bool RecursiveASTVisitor<Derived>::TraverseTemplateParameterListHelper(
1931 TemplateParameterList *TPL) {
1932 if (TPL) {
1933 for (NamedDecl *D : *TPL) {
1934 TRY_TO(TraverseDecl(D));
1935 }
1936 if (Expr *RequiresClause = TPL->getRequiresClause()) {
1937 TRY_TO(TraverseStmt(RequiresClause));
1938 }
1939 }
1940 return true;
1941}
1942
1943template <typename Derived>
1944template <typename T>
1945bool RecursiveASTVisitor<Derived>::TraverseDeclTemplateParameterLists(T *D) {
1946 for (unsigned i = 0; i < D->getNumTemplateParameterLists(); i++) {
1947 TemplateParameterList *TPL = D->getTemplateParameterList(i);
1948 TraverseTemplateParameterListHelper(TPL);
1949 }
1950 return true;
1951}
1952
1953template <typename Derived>
1955 ClassTemplateDecl *D) {
1956 for (auto *SD : D->specializations()) {
1957 for (auto *RD : SD->redecls()) {
1958 assert(!cast<CXXRecordDecl>(RD)->isInjectedClassName());
1959 switch (
1960 cast<ClassTemplateSpecializationDecl>(RD)->getSpecializationKind()) {
1961 // Visit the implicit instantiations with the requested pattern.
1962 case TSK_Undeclared:
1964 TRY_TO(TraverseDecl(RD));
1965 break;
1966
1967 // We don't need to do anything on an explicit instantiation
1968 // or explicit specialization because there will be an explicit
1969 // node for it elsewhere.
1973 break;
1974 }
1975 }
1976 }
1977
1978 return true;
1979}
1980
1981template <typename Derived>
1983 VarTemplateDecl *D) {
1984 for (auto *SD : D->specializations()) {
1985 for (auto *RD : SD->redecls()) {
1986 switch (
1987 cast<VarTemplateSpecializationDecl>(RD)->getSpecializationKind()) {
1988 case TSK_Undeclared:
1990 TRY_TO(TraverseDecl(RD));
1991 break;
1992
1996 break;
1997 }
1998 }
1999 }
2000
2001 return true;
2002}
2003
2004// A helper method for traversing the instantiations of a
2005// function while skipping its specializations.
2006template <typename Derived>
2009 for (auto *FD : D->specializations()) {
2010 for (auto *RD : FD->redecls()) {
2011 switch (RD->getTemplateSpecializationKind()) {
2012 case TSK_Undeclared:
2014 // We don't know what kind of FunctionDecl this is.
2015 TRY_TO(TraverseDecl(RD));
2016 break;
2017
2018 // FIXME: For now traverse explicit instantiations here. Change that
2019 // once they are represented as dedicated nodes in the AST.
2022 TRY_TO(TraverseDecl(RD));
2023 break;
2024
2026 break;
2027 }
2028 }
2029 }
2030
2031 return true;
2032}
2033
2034// This macro unifies the traversal of class, variable and function
2035// template declarations.
2036#define DEF_TRAVERSE_TMPL_DECL(TMPLDECLKIND) \
2037 DEF_TRAVERSE_DECL(TMPLDECLKIND##TemplateDecl, { \
2038 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters())); \
2039 TRY_TO(TraverseDecl(D->getTemplatedDecl())); \
2040 \
2041 /* By default, we do not traverse the instantiations of \
2042 class templates since they do not appear in the user code. The \
2043 following code optionally traverses them. \
2044 \
2045 We only traverse the class instantiations when we see the canonical \
2046 declaration of the template, to ensure we only visit them once. */ \
2047 if (getDerived().shouldVisitTemplateInstantiations() && \
2048 D == D->getCanonicalDecl()) \
2049 TRY_TO(TraverseTemplateInstantiations(D)); \
2050 \
2051 /* Note that getInstantiatedFromMemberTemplate() is just a link \
2052 from a template instantiation back to the template from which \
2053 it was instantiated, and thus should not be traversed. */ \
2054 })
2055
2059
2061 // D is the "T" in something like
2062 // template <template <typename> class T> class container { };
2063 TRY_TO(TraverseDecl(D->getTemplatedDecl()));
2064 if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
2065 TRY_TO(TraverseTemplateArgumentLoc(D->getDefaultArgument()));
2066 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters()));
2067})
2068
2070 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters()));
2071})
2072
2073template <typename Derived>
2074bool RecursiveASTVisitor<Derived>::TraverseTemplateTypeParamDeclConstraints(
2075 const TemplateTypeParmDecl *D) {
2076 if (const auto *TC = D->getTypeConstraint())
2077 TRY_TO(TraverseTypeConstraint(TC));
2078 return true;
2079}
2080
2082 // D is the "T" in something like "template<typename T> class vector;"
2083 if (D->getTypeForDecl())
2084 TRY_TO(TraverseType(QualType(D->getTypeForDecl(), 0)));
2085 TRY_TO(TraverseTemplateTypeParamDeclConstraints(D));
2086 if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
2087 TRY_TO(TraverseTemplateArgumentLoc(D->getDefaultArgument()));
2088})
2089
2091 TRY_TO(TraverseTypeLoc(D->getTypeSourceInfo()->getTypeLoc()));
2092 // We shouldn't traverse D->getTypeForDecl(); it's a result of
2093 // declaring the typedef, not something that was written in the
2094 // source.
2095})
2096
2098 TRY_TO(TraverseTypeLoc(D->getTypeSourceInfo()->getTypeLoc()));
2099 // We shouldn't traverse D->getTypeForDecl(); it's a result of
2100 // declaring the type alias, not something that was written in the
2101 // source.
2102})
2103
2105 TRY_TO(TraverseDecl(D->getTemplatedDecl()));
2106 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters()));
2107})
2108
2110 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters()));
2111 TRY_TO(TraverseStmt(D->getConstraintExpr()));
2112})
2113
2115 // A dependent using declaration which was marked with 'typename'.
2116 // template<class T> class A : public B<T> { using typename B<T>::foo; };
2117 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2118 // We shouldn't traverse D->getTypeForDecl(); it's a result of
2119 // declaring the type, not something that was written in the
2120 // source.
2121})
2122
2124
2126 TRY_TO(TraverseDeclTemplateParameterLists(D));
2127
2128 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2129 if (auto *TSI = D->getIntegerTypeSourceInfo())
2130 TRY_TO(TraverseTypeLoc(TSI->getTypeLoc()));
2131 // The enumerators are already traversed by
2132 // decls_begin()/decls_end().
2133})
2134
2135// Helper methods for RecordDecl and its children.
2136template <typename Derived>
2137bool RecursiveASTVisitor<Derived>::TraverseRecordHelper(RecordDecl *D) {
2138 // We shouldn't traverse D->getTypeForDecl(); it's a result of
2139 // declaring the type, not something that was written in the source.
2140
2141 TRY_TO(TraverseDeclTemplateParameterLists(D));
2142 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2143 return true;
2144}
2145
2146template <typename Derived>
2148 const CXXBaseSpecifier &Base) {
2149 TRY_TO(TraverseTypeLoc(Base.getTypeSourceInfo()->getTypeLoc()));
2150 return true;
2151}
2152
2153template <typename Derived>
2154bool RecursiveASTVisitor<Derived>::TraverseCXXRecordHelper(CXXRecordDecl *D) {
2155 if (!TraverseRecordHelper(D))
2156 return false;
2157 if (D->isCompleteDefinition()) {
2158 for (const auto &I : D->bases()) {
2159 TRY_TO(TraverseCXXBaseSpecifier(I));
2160 }
2161 // We don't traverse the friends or the conversions, as they are
2162 // already in decls_begin()/decls_end().
2163 }
2164 return true;
2165}
2166
2167DEF_TRAVERSE_DECL(RecordDecl, { TRY_TO(TraverseRecordHelper(D)); })
2168
2169DEF_TRAVERSE_DECL(CXXRecordDecl, { TRY_TO(TraverseCXXRecordHelper(D)); })
2170
2171template <typename Derived>
2172bool RecursiveASTVisitor<Derived>::TraverseTemplateArgumentLocsHelper(
2173 const TemplateArgumentLoc *TAL, unsigned Count) {
2174 for (unsigned I = 0; I < Count; ++I) {
2175 TRY_TO(TraverseTemplateArgumentLoc(TAL[I]));
2176 }
2177 return true;
2178}
2179
2180#define DEF_TRAVERSE_TMPL_SPEC_DECL(TMPLDECLKIND, DECLKIND) \
2181 DEF_TRAVERSE_DECL(TMPLDECLKIND##TemplateSpecializationDecl, { \
2182 /* For implicit instantiations ("set<int> x;"), we don't want to \
2183 recurse at all, since the instatiated template isn't written in \
2184 the source code anywhere. (Note the instatiated *type* -- \
2185 set<int> -- is written, and will still get a callback of \
2186 TemplateSpecializationType). For explicit instantiations \
2187 ("template set<int>;"), we do need a callback, since this \
2188 is the only callback that's made for this instantiation. \
2189 We use getTemplateArgsAsWritten() to distinguish. */ \
2190 if (const auto *ArgsWritten = D->getTemplateArgsAsWritten()) { \
2191 /* The args that remains unspecialized. */ \
2192 TRY_TO(TraverseTemplateArgumentLocsHelper( \
2193 ArgsWritten->getTemplateArgs(), ArgsWritten->NumTemplateArgs)); \
2194 } \
2195 \
2196 if (getDerived().shouldVisitTemplateInstantiations() || \
2197 D->getTemplateSpecializationKind() == TSK_ExplicitSpecialization) { \
2198 /* Traverse base definition for explicit specializations */ \
2199 TRY_TO(Traverse##DECLKIND##Helper(D)); \
2200 } else { \
2201 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc())); \
2202 \
2203 /* Returning from here skips traversing the \
2204 declaration context of the *TemplateSpecializationDecl \
2205 (embedded in the DEF_TRAVERSE_DECL() macro) \
2206 which contains the instantiated members of the template. */ \
2207 return true; \
2208 } \
2209 })
2210
2213
2214#define DEF_TRAVERSE_TMPL_PART_SPEC_DECL(TMPLDECLKIND, DECLKIND) \
2215 DEF_TRAVERSE_DECL(TMPLDECLKIND##TemplatePartialSpecializationDecl, { \
2216 /* The partial specialization. */ \
2217 TRY_TO(TraverseTemplateParameterListHelper(D->getTemplateParameters())); \
2218 /* The args that remains unspecialized. */ \
2219 TRY_TO(TraverseTemplateArgumentLocsHelper( \
2220 D->getTemplateArgsAsWritten()->getTemplateArgs(), \
2221 D->getTemplateArgsAsWritten()->NumTemplateArgs)); \
2222 \
2223 /* Don't need the *TemplatePartialSpecializationHelper, even \
2224 though that's our parent class -- we already visit all the \
2225 template args here. */ \
2226 TRY_TO(Traverse##DECLKIND##Helper(D)); \
2227 \
2228 /* Instantiations will have been visited with the primary template. */ \
2229 })
2230
2233
2234DEF_TRAVERSE_DECL(EnumConstantDecl, { TRY_TO(TraverseStmt(D->getInitExpr())); })
2235
2237 // Like UnresolvedUsingTypenameDecl, but without the 'typename':
2238 // template <class T> Class A : public Base<T> { using Base<T>::foo; };
2239 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2240 TRY_TO(TraverseDeclarationNameInfo(D->getNameInfo()));
2241})
2242
2244
2245template <typename Derived>
2246bool RecursiveASTVisitor<Derived>::TraverseDeclaratorHelper(DeclaratorDecl *D) {
2247 TRY_TO(TraverseDeclTemplateParameterLists(D));
2248 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2249 if (D->getTypeSourceInfo())
2250 TRY_TO(TraverseTypeLoc(D->getTypeSourceInfo()->getTypeLoc()));
2251 else
2252 TRY_TO(TraverseType(D->getType()));
2253 return true;
2254}
2255
2257 TRY_TO(TraverseVarHelper(D));
2258 for (auto *Binding : D->bindings()) {
2259 TRY_TO(TraverseDecl(Binding));
2260 }
2261})
2262
2264 if (getDerived().shouldVisitImplicitCode()) {
2265 TRY_TO(TraverseStmt(D->getBinding()));
2266 if (const auto HoldingVar = D->getHoldingVar())
2267 TRY_TO(TraverseDecl(HoldingVar));
2268 }
2269})
2270
2271DEF_TRAVERSE_DECL(MSPropertyDecl, { TRY_TO(TraverseDeclaratorHelper(D)); })
2272
2275
2277
2279 TRY_TO(TraverseDeclaratorHelper(D));
2280 if (D->isBitField())
2281 TRY_TO(TraverseStmt(D->getBitWidth()));
2282 if (D->hasInClassInitializer())
2283 TRY_TO(TraverseStmt(D->getInClassInitializer()));
2284})
2285
2287 TRY_TO(TraverseDeclaratorHelper(D));
2288 if (D->isBitField())
2289 TRY_TO(TraverseStmt(D->getBitWidth()));
2290 // FIXME: implement the rest.
2291})
2292
2294 TRY_TO(TraverseDeclaratorHelper(D));
2295 if (D->isBitField())
2296 TRY_TO(TraverseStmt(D->getBitWidth()));
2297 // FIXME: implement the rest.
2298})
2299
2300template <typename Derived>
2301bool RecursiveASTVisitor<Derived>::TraverseFunctionHelper(FunctionDecl *D) {
2302 TRY_TO(TraverseDeclTemplateParameterLists(D));
2303 TRY_TO(TraverseNestedNameSpecifierLoc(D->getQualifierLoc()));
2304 TRY_TO(TraverseDeclarationNameInfo(D->getNameInfo()));
2305
2306 // If we're an explicit template specialization, iterate over the
2307 // template args that were explicitly specified. If we were doing
2308 // this in typing order, we'd do it between the return type and
2309 // the function args, but both are handled by the FunctionTypeLoc
2310 // above, so we have to choose one side. I've decided to do before.
2311 if (const FunctionTemplateSpecializationInfo *FTSI =
2312 D->getTemplateSpecializationInfo()) {
2313 if (FTSI->getTemplateSpecializationKind() != TSK_Undeclared &&
2314 FTSI->getTemplateSpecializationKind() != TSK_ImplicitInstantiation) {
2315 // A specialization might not have explicit template arguments if it has
2316 // a templated return type and concrete arguments.
2317 if (const ASTTemplateArgumentListInfo *TALI =
2318 FTSI->TemplateArgumentsAsWritten) {
2319 TRY_TO(TraverseTemplateArgumentLocsHelper(TALI->getTemplateArgs(),
2320 TALI->NumTemplateArgs));
2321 }
2322 }
2323 } else if (const DependentFunctionTemplateSpecializationInfo *DFSI =
2324 D->getDependentSpecializationInfo()) {
2325 if (const ASTTemplateArgumentListInfo *TALI =
2326 DFSI->TemplateArgumentsAsWritten) {
2327 TRY_TO(TraverseTemplateArgumentLocsHelper(TALI->getTemplateArgs(),
2328 TALI->NumTemplateArgs));
2329 }
2330 }
2331
2332 // Visit the function type itself, which can be either
2333 // FunctionNoProtoType or FunctionProtoType, or a typedef. This
2334 // also covers the return type and the function parameters,
2335 // including exception specifications.
2336 if (TypeSourceInfo *TSI = D->getTypeSourceInfo()) {
2337 TRY_TO(TraverseTypeLoc(TSI->getTypeLoc()));
2338 } else if (getDerived().shouldVisitImplicitCode()) {
2339 // Visit parameter variable declarations of the implicit function
2340 // if the traverser is visiting implicit code. Parameter variable
2341 // declarations do not have valid TypeSourceInfo, so to visit them
2342 // we need to traverse the declarations explicitly.
2343 for (ParmVarDecl *Parameter : D->parameters()) {
2344 TRY_TO(TraverseDecl(Parameter));
2345 }
2346 }
2347
2348 // Visit the trailing requires clause, if any.
2349 if (const AssociatedConstraint &TrailingRequiresClause =
2350 D->getTrailingRequiresClause()) {
2351 TRY_TO(TraverseStmt(
2352 const_cast<Expr *>(TrailingRequiresClause.ConstraintExpr)));
2353 }
2354
2355 if (CXXConstructorDecl *Ctor = dyn_cast<CXXConstructorDecl>(D)) {
2356 // Constructor initializers.
2357 for (auto *I : Ctor->inits()) {
2358 if (I->isWritten() || getDerived().shouldVisitImplicitCode())
2359 TRY_TO(TraverseConstructorInitializer(I));
2360 }
2361 }
2362
2363 bool VisitBody =
2364 D->isThisDeclarationADefinition() &&
2365 // Don't visit the function body if the function definition is generated
2366 // by clang.
2367 (!D->isDefaulted() || getDerived().shouldVisitImplicitCode());
2368
2369 if (const auto *MD = dyn_cast<CXXMethodDecl>(D)) {
2370 if (const CXXRecordDecl *RD = MD->getParent()) {
2371 if (RD->isLambda() &&
2372 declaresSameEntity(RD->getLambdaCallOperator(), MD)) {
2373 VisitBody = VisitBody && getDerived().shouldVisitLambdaBody();
2374 }
2375 }
2376 }
2377
2378 if (VisitBody) {
2379 TRY_TO(TraverseStmt(D->getBody()));
2380 // Body may contain using declarations whose shadows are parented to the
2381 // FunctionDecl itself.
2382 for (auto *Child : D->decls()) {
2383 if (isa<UsingShadowDecl>(Child))
2384 TRY_TO(TraverseDecl(Child));
2385 }
2386 }
2387 return true;
2388}
2389
2391 // We skip decls_begin/decls_end, which are already covered by
2392 // TraverseFunctionHelper().
2393 ShouldVisitChildren = false;
2394 ReturnValue = TraverseFunctionHelper(D);
2395})
2396
2398 // We skip decls_begin/decls_end, which are already covered by
2399 // TraverseFunctionHelper().
2400 ShouldVisitChildren = false;
2401 ReturnValue = TraverseFunctionHelper(D);
2402})
2403
2405 // We skip decls_begin/decls_end, which are already covered by
2406 // TraverseFunctionHelper().
2407 ShouldVisitChildren = false;
2408 ReturnValue = TraverseFunctionHelper(D);
2409})
2410
2412 // We skip decls_begin/decls_end, which are already covered by
2413 // TraverseFunctionHelper().
2414 ShouldVisitChildren = false;
2415 ReturnValue = TraverseFunctionHelper(D);
2416})
2417
2418// CXXConversionDecl is the declaration of a type conversion operator.
2419// It's not a cast expression.
2421 // We skip decls_begin/decls_end, which are already covered by
2422 // TraverseFunctionHelper().
2423 ShouldVisitChildren = false;
2424 ReturnValue = TraverseFunctionHelper(D);
2425})
2426
2428 // We skip decls_begin/decls_end, which are already covered by
2429 // TraverseFunctionHelper().
2430 ShouldVisitChildren = false;
2431 ReturnValue = TraverseFunctionHelper(D);
2432})
2433
2434template <typename Derived>
2435bool RecursiveASTVisitor<Derived>::TraverseVarHelper(VarDecl *D) {
2436 TRY_TO(TraverseDeclaratorHelper(D));
2437 // Default params are taken care of when we traverse the ParmVarDecl.
2438 if (!isa<ParmVarDecl>(D) &&
2439 (!D->isCXXForRangeDecl() || getDerived().shouldVisitImplicitCode()))
2440 TRY_TO(TraverseStmt(D->getInit()));
2441 return true;
2442}
2443
2444DEF_TRAVERSE_DECL(VarDecl, { TRY_TO(TraverseVarHelper(D)); })
2445
2446DEF_TRAVERSE_DECL(ImplicitParamDecl, { TRY_TO(TraverseVarHelper(D)); })
2447
2449 // A non-type template parameter, e.g. "S" in template<int S> class Foo ...
2450 TRY_TO(TraverseDeclaratorHelper(D));
2451 if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
2452 TRY_TO(TraverseTemplateArgumentLoc(D->getDefaultArgument()));
2453})
2454
2456 TRY_TO(TraverseVarHelper(D));
2457
2458 if (D->hasDefaultArg() && D->hasUninstantiatedDefaultArg() &&
2459 !D->hasUnparsedDefaultArg())
2460 TRY_TO(TraverseStmt(D->getUninstantiatedDefaultArg()));
2461
2462 if (D->hasDefaultArg() && !D->hasUninstantiatedDefaultArg() &&
2463 !D->hasUnparsedDefaultArg())
2464 TRY_TO(TraverseStmt(D->getDefaultArg()));
2465})
2466
2468
2470 TRY_TO(TraverseTemplateArguments(D->getTemplateArguments()));
2471})
2472
2473#undef DEF_TRAVERSE_DECL
2474
2475// ----------------- Stmt traversal -----------------
2476//
2477// For stmts, we automate (in the DEF_TRAVERSE_STMT macro) iterating
2478// over the children defined in children() (every stmt defines these,
2479// though sometimes the range is empty). Each individual Traverse*
2480// method only needs to worry about children other than those. To see
2481// what children() does for a given class, see, e.g.,
2482// http://clang.llvm.org/doxygen/Stmt_8cpp_source.html
2483
2484// This macro makes available a variable S, the passed-in stmt.
2485#define DEF_TRAVERSE_STMT(STMT, CODE) \
2486 template <typename Derived> \
2488 STMT *S, DataRecursionQueue *Queue) { \
2489 bool ShouldVisitChildren = true; \
2490 bool ReturnValue = true; \
2491 if (!getDerived().shouldTraversePostOrder()) \
2492 TRY_TO(WalkUpFrom##STMT(S)); \
2493 { CODE; } \
2494 if (ShouldVisitChildren) { \
2495 for (Stmt * SubStmt : getDerived().getStmtChildren(S)) { \
2496 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(SubStmt); \
2497 } \
2498 } \
2499 /* Call WalkUpFrom if TRY_TO_TRAVERSE_OR_ENQUEUE_STMT has traversed the \
2500 * children already. If TRY_TO_TRAVERSE_OR_ENQUEUE_STMT only enqueued the \
2501 * children, PostVisitStmt will call WalkUpFrom after we are done visiting \
2502 * children. */ \
2503 if (!Queue && ReturnValue && getDerived().shouldTraversePostOrder()) { \
2504 TRY_TO(WalkUpFrom##STMT(S)); \
2505 } \
2506 return ReturnValue; \
2507 }
2508
2510 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getAsmStringExpr());
2511 for (unsigned I = 0, E = S->getNumInputs(); I < E; ++I) {
2512 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getInputConstraintExpr(I));
2513 }
2514 for (unsigned I = 0, E = S->getNumOutputs(); I < E; ++I) {
2515 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getOutputConstraintExpr(I));
2516 }
2517 for (unsigned I = 0, E = S->getNumClobbers(); I < E; ++I) {
2518 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getClobberExpr(I));
2519 }
2520 // children() iterates over inputExpr and outputExpr.
2521})
2522
2524 MSAsmStmt,
2525 {// FIXME: MS Asm doesn't currently parse Constraints, Clobbers, etc. Once
2526 // added this needs to be implemented.
2527 })
2528
2530 TRY_TO(TraverseDecl(S->getExceptionDecl()));
2531 // children() iterates over the handler block.
2532})
2533
2535 for (auto *I : S->decls()) {
2536 TRY_TO(TraverseDecl(I));
2537 }
2538 // Suppress the default iteration over children() by
2539 // returning. Here's why: A DeclStmt looks like 'type var [=
2540 // initializer]'. The decls above already traverse over the
2541 // initializers, so we don't have to do it again (which
2542 // children() would do).
2543 ShouldVisitChildren = false;
2544})
2545
2546// These non-expr stmts (most of them), do not need any action except
2547// iterating over the children.
2569
2571 if (!getDerived().shouldVisitImplicitCode()) {
2572 if (S->getInit())
2573 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getInit());
2574 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getLoopVarStmt());
2575 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getRangeInit());
2576 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getBody());
2577 // Visit everything else only if shouldVisitImplicitCode().
2578 ShouldVisitChildren = false;
2579 }
2580})
2581
2583 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2584 TRY_TO(TraverseDeclarationNameInfo(S->getNameInfo()));
2585})
2586
2590
2592
2594 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2595 TRY_TO(TraverseDeclarationNameInfo(S->getMemberNameInfo()));
2596 if (S->hasExplicitTemplateArgs()) {
2597 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
2598 S->getNumTemplateArgs()));
2599 }
2600})
2601
2603 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2604 TRY_TO(TraverseDeclarationNameInfo(S->getNameInfo()));
2605 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
2606 S->getNumTemplateArgs()));
2607})
2608
2610 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2611 TRY_TO(TraverseDeclarationNameInfo(S->getNameInfo()));
2612 if (S->hasExplicitTemplateArgs()) {
2613 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
2614 S->getNumTemplateArgs()));
2615 }
2616})
2617
2619 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2620 TRY_TO(TraverseDeclarationNameInfo(S->getMemberNameInfo()));
2621 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
2622 S->getNumTemplateArgs()));
2623})
2624
2627 {// We don't traverse the cast type, as it's not written in the
2628 // source code.
2629 })
2630
2632 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2633})
2634
2636 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2637})
2638
2640 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2641})
2642
2644 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2645})
2646
2648 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2649})
2650
2652 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2653})
2654
2656 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2657})
2658
2660 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2661})
2662
2663template <typename Derived>
2665 InitListExpr *S, DataRecursionQueue *Queue) {
2666 if (S) {
2667 // Skip this if we traverse postorder. We will visit it later
2668 // in PostVisitStmt.
2669 if (!getDerived().shouldTraversePostOrder())
2670 TRY_TO(WalkUpFromInitListExpr(S));
2671
2672 // All we need are the default actions. FIXME: use a helper function.
2673 for (Stmt *SubStmt : S->children()) {
2675 }
2676
2677 if (!Queue && getDerived().shouldTraversePostOrder())
2678 TRY_TO(WalkUpFromInitListExpr(S));
2679 }
2680 return true;
2681}
2682
2683template <typename Derived>
2685 ObjCProtocolLoc ProtocolLoc) {
2686 return true;
2687}
2688
2689template <typename Derived>
2691 ConceptReference *CR) {
2692 if (!getDerived().shouldTraversePostOrder())
2693 TRY_TO(VisitConceptReference(CR));
2694 TRY_TO(TraverseNestedNameSpecifierLoc(CR->getNestedNameSpecifierLoc()));
2695 TRY_TO(TraverseDeclarationNameInfo(CR->getConceptNameInfo()));
2696 if (CR->hasExplicitTemplateArgs())
2697 TRY_TO(TraverseTemplateArgumentLocsHelper(
2698 CR->getTemplateArgsAsWritten()->getTemplateArgs(),
2699 CR->getTemplateArgsAsWritten()->NumTemplateArgs));
2700 if (getDerived().shouldTraversePostOrder())
2701 TRY_TO(VisitConceptReference(CR));
2702 return true;
2703}
2704
2705// If shouldVisitImplicitCode() returns false, this method traverses only the
2706// syntactic form of InitListExpr.
2707// If shouldVisitImplicitCode() return true, this method is called once for
2708// each pair of syntactic and semantic InitListExpr, and it traverses the
2709// subtrees defined by the two forms. This may cause some of the children to be
2710// visited twice, if they appear both in the syntactic and the semantic form.
2711//
2712// There is no guarantee about which form \p S takes when this method is called.
2713template <typename Derived>
2715 InitListExpr *S, DataRecursionQueue *Queue) {
2716 if (S->isSemanticForm() && S->isSyntacticForm()) {
2717 // `S` does not have alternative forms, traverse only once.
2718 TRY_TO(TraverseSynOrSemInitListExpr(S, Queue));
2719 return true;
2720 }
2721 TRY_TO(TraverseSynOrSemInitListExpr(
2722 S->isSemanticForm() ? S->getSyntacticForm() : S, Queue));
2723 if (getDerived().shouldVisitImplicitCode()) {
2724 // Only visit the semantic form if the clients are interested in implicit
2725 // compiler-generated.
2726 TRY_TO(TraverseSynOrSemInitListExpr(
2727 S->isSemanticForm() ? S : S->getSemanticForm(), Queue));
2728 }
2729 return true;
2730}
2731
2732// GenericSelectionExpr is a special case because the types and expressions
2733// are interleaved. We also need to watch out for null types (default
2734// generic associations).
2736 if (S->isExprPredicate())
2737 TRY_TO(TraverseStmt(S->getControllingExpr()));
2738 else
2739 TRY_TO(TraverseTypeLoc(S->getControllingType()->getTypeLoc()));
2740
2741 for (const GenericSelectionExpr::Association Assoc : S->associations()) {
2742 if (TypeSourceInfo *TSI = Assoc.getTypeSourceInfo())
2743 TRY_TO(TraverseTypeLoc(TSI->getTypeLoc()));
2744 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(Assoc.getAssociationExpr());
2745 }
2746 ShouldVisitChildren = false;
2747})
2748
2749// PseudoObjectExpr is a special case because of the weirdness with
2750// syntactic expressions and opaque values.
2752 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getSyntacticForm());
2753 for (PseudoObjectExpr::semantics_iterator i = S->semantics_begin(),
2754 e = S->semantics_end();
2755 i != e; ++i) {
2756 Expr *sub = *i;
2757 if (OpaqueValueExpr *OVE = dyn_cast<OpaqueValueExpr>(sub))
2758 sub = OVE->getSourceExpr();
2760 }
2761 ShouldVisitChildren = false;
2762})
2763
2765 // This is called for code like 'return T()' where T is a built-in
2766 // (i.e. non-class) type.
2767 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2768})
2769
2771 // The child-iterator will pick up the other arguments.
2772 TRY_TO(TraverseTypeLoc(S->getAllocatedTypeSourceInfo()->getTypeLoc()));
2773})
2774
2776 // The child-iterator will pick up the expression representing
2777 // the field.
2778 // FIMXE: for code like offsetof(Foo, a.b.c), should we get
2779 // making a MemberExpr callbacks for Foo.a, Foo.a.b, and Foo.a.b.c?
2780 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2781})
2782
2784 // The child-iterator will pick up the arg if it's an expression,
2785 // but not if it's a type.
2786 if (S->isArgumentType())
2787 TRY_TO(TraverseTypeLoc(S->getArgumentTypeInfo()->getTypeLoc()));
2788})
2789
2791 // The child-iterator will pick up the arg if it's an expression,
2792 // but not if it's a type.
2793 if (S->isTypeOperand())
2794 TRY_TO(TraverseTypeLoc(S->getTypeOperandSourceInfo()->getTypeLoc()));
2795})
2796
2798 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2799})
2800
2802
2804 // The child-iterator will pick up the arg if it's an expression,
2805 // but not if it's a type.
2806 if (S->isTypeOperand())
2807 TRY_TO(TraverseTypeLoc(S->getTypeOperandSourceInfo()->getTypeLoc()));
2808})
2809
2811 for (unsigned I = 0, N = S->getNumArgs(); I != N; ++I)
2812 TRY_TO(TraverseTypeLoc(S->getArg(I)->getTypeLoc()));
2813})
2814
2816 TRY_TO(TraverseTypeLoc(S->getQueriedTypeSourceInfo()->getTypeLoc()));
2817})
2818
2820 { TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getQueriedExpression()); })
2821
2823 // The child-iterator will pick up the expression argument.
2824 TRY_TO(TraverseTypeLoc(S->getWrittenTypeInfo()->getTypeLoc()));
2825})
2826
2828 // This is called for code like 'return T()' where T is a class type.
2829 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2830})
2831
2832// Walk only the visible parts of lambda expressions.
2834 // Visit the capture list.
2835 for (unsigned I = 0, N = S->capture_size(); I != N; ++I) {
2836 const LambdaCapture *C = S->capture_begin() + I;
2837 if (C->isExplicit() || getDerived().shouldVisitImplicitCode()) {
2838 TRY_TO(TraverseLambdaCapture(S, C, S->capture_init_begin()[I]));
2839 }
2840 }
2841
2842 if (getDerived().shouldVisitImplicitCode()) {
2843 // The implicit model is simple: everything else is in the lambda class.
2844 TRY_TO(TraverseDecl(S->getLambdaClass()));
2845 } else {
2846 // We need to poke around to find the bits that might be explicitly written.
2847 TypeLoc TL = S->getCallOperator()->getTypeSourceInfo()->getTypeLoc();
2849
2850 TRY_TO(TraverseTemplateParameterListHelper(S->getTemplateParameterList()));
2851 if (S->hasExplicitParameters()) {
2852 // Visit parameters.
2853 for (unsigned I = 0, N = Proto.getNumParams(); I != N; ++I)
2854 TRY_TO(TraverseDecl(Proto.getParam(I)));
2855 }
2856
2857 auto *T = Proto.getTypePtr();
2858 for (const auto &E : T->exceptions())
2859 TRY_TO(TraverseType(E));
2860
2861 if (Expr *NE = T->getNoexceptExpr())
2863
2864 if (S->hasExplicitResultType())
2865 TRY_TO(TraverseTypeLoc(Proto.getReturnLoc()));
2867 const_cast<Expr *>(S->getTrailingRequiresClause().ConstraintExpr));
2868
2869 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getBody());
2870 }
2871 ShouldVisitChildren = false;
2872})
2873
2875 // This is called for code like 'T()', where T is a template argument.
2876 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2877})
2878
2879// These expressions all might take explicit template arguments.
2880// We traverse those if so. FIXME: implement these.
2884
2885// These exprs (most of them), do not need any action except iterating
2886// over the children.
2893
2895 TRY_TO(TraverseDecl(S->getBlockDecl()));
2896 return true; // no child statements to loop through.
2897})
2898
2901 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2902})
2905
2907 if (getDerived().shouldVisitImplicitCode())
2908 TRY_TO(TraverseStmt(S->getExpr()));
2909})
2910
2912 if (getDerived().shouldVisitImplicitCode())
2913 TRY_TO(TraverseStmt(S->getExpr()));
2914})
2915
2921
2923 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
2924 if (TypeSourceInfo *ScopeInfo = S->getScopeTypeInfo())
2925 TRY_TO(TraverseTypeLoc(ScopeInfo->getTypeLoc()));
2926 if (TypeSourceInfo *DestroyedTypeInfo = S->getDestroyedTypeInfo())
2927 TRY_TO(TraverseTypeLoc(DestroyedTypeInfo->getTypeLoc()));
2928})
2929
2940 // FIXME: The source expression of the OVE should be listed as
2941 // a child of the ArrayInitLoopExpr.
2942 if (OpaqueValueExpr *OVE = S->getCommonExpr())
2943 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(OVE->getSourceExpr());
2944})
2947
2949 if (TypeSourceInfo *TInfo = S->getEncodedTypeSourceInfo())
2950 TRY_TO(TraverseTypeLoc(TInfo->getTypeLoc()));
2951})
2952
2955
2957 if (TypeSourceInfo *TInfo = S->getClassReceiverTypeInfo())
2958 TRY_TO(TraverseTypeLoc(TInfo->getTypeLoc()));
2959})
2960
2962 if (S->isClassReceiver()) {
2963 ObjCInterfaceDecl *IDecl = S->getClassReceiver();
2964 QualType Type = IDecl->getASTContext().getObjCInterfaceType(IDecl);
2966 Data.NameLoc = S->getReceiverLocation();
2967 Data.NameEndLoc = Data.NameLoc;
2968 TRY_TO(TraverseTypeLoc(TypeLoc(Type, &Data)));
2969 }
2970})
2975
2977 TRY_TO(TraverseTypeLoc(S->getTypeInfoAsWritten()->getTypeLoc()));
2978})
2979
2984 TRY_TO(TraverseTypeLoc(S->getTypeSourceInfo()->getTypeLoc()));
2985})
2993 for (IntegerLiteral *IL : S->underlying_data_elements()) {
2995 }
2996})
2997
2999 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
3000 if (S->hasExplicitTemplateArgs()) {
3001 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
3002 S->getNumTemplateArgs()));
3003 }
3004})
3005
3007 TRY_TO(TraverseNestedNameSpecifierLoc(S->getQualifierLoc()));
3008 if (S->hasExplicitTemplateArgs()) {
3009 TRY_TO(TraverseTemplateArgumentLocsHelper(S->getTemplateArgs(),
3010 S->getNumTemplateArgs()));
3011 }
3012})
3013
3018DEF_TRAVERSE_STMT(CapturedStmt, { TRY_TO(TraverseDecl(S->getCapturedDecl())); })
3019
3021 if (getDerived().shouldVisitImplicitCode()) {
3022 TRY_TO(TraverseStmt(S->getOriginalStmt()));
3023 TRY_TO(TraverseDecl(S->getOutlinedFunctionDecl()));
3024 ShouldVisitChildren = false;
3025 }
3026})
3027
3030 if (!getDerived().shouldVisitImplicitCode()) {
3032 S->getDecomposedForm();
3033 TRY_TO(TraverseStmt(const_cast<Expr*>(Decomposed.LHS)));
3034 TRY_TO(TraverseStmt(const_cast<Expr*>(Decomposed.RHS)));
3035 ShouldVisitChildren = false;
3036 }
3037})
3041
3042// These operators (all of them) do not need any action except
3043// iterating over the children.
3059
3061 if (S->getLifetimeExtendedTemporaryDecl()) {
3062 TRY_TO(TraverseLifetimeExtendedTemporaryDecl(
3063 S->getLifetimeExtendedTemporaryDecl()));
3064 ShouldVisitChildren = false;
3065 }
3066})
3067// For coroutines expressions, traverse either the operand
3068// as written or the implied calls, depending on what the
3069// derived class requests.
3071 if (!getDerived().shouldVisitImplicitCode()) {
3072 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getBody());
3073 ShouldVisitChildren = false;
3074 }
3075})
3077 if (!getDerived().shouldVisitImplicitCode()) {
3078 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getOperand());
3079 ShouldVisitChildren = false;
3080 }
3081})
3083 if (!getDerived().shouldVisitImplicitCode()) {
3084 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getOperand());
3085 ShouldVisitChildren = false;
3086 }
3087})
3089 if (!getDerived().shouldVisitImplicitCode()) {
3090 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getOperand());
3091 ShouldVisitChildren = false;
3092 }
3093})
3095 if (!getDerived().shouldVisitImplicitCode()) {
3096 TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S->getOperand());
3097 ShouldVisitChildren = false;
3098 }
3099})
3100
3102 TRY_TO(TraverseConceptReference(S->getConceptReference()));
3103})
3104
3106 TRY_TO(TraverseDecl(S->getBody()));
3107 for (ParmVarDecl *Parm : S->getLocalParameters())
3108 TRY_TO(TraverseDecl(Parm));
3109 for (concepts::Requirement *Req : S->getRequirements())
3110 TRY_TO(TraverseConceptRequirement(Req));
3111})
3112
3113// These literals (all of them) do not need any action.
3124
3125// Traverse OpenCL: AsType, Convert.
3127
3128// OpenMP directives.
3129template <typename Derived>
3130bool RecursiveASTVisitor<Derived>::TraverseOMPExecutableDirective(
3131 OMPExecutableDirective *S) {
3132 for (auto *C : S->clauses()) {
3133 TRY_TO(TraverseOMPClause(C));
3134 }
3135 return true;
3136}
3137
3138DEF_TRAVERSE_STMT(OMPCanonicalLoop, {
3139 if (!getDerived().shouldVisitImplicitCode()) {
3140 // Visit only the syntactical loop.
3141 TRY_TO(TraverseStmt(S->getLoopStmt()));
3142 ShouldVisitChildren = false;
3143 }
3144})
3145
3146template <typename Derived>
3147bool
3148RecursiveASTVisitor<Derived>::TraverseOMPLoopDirective(OMPLoopDirective *S) {
3149 return TraverseOMPExecutableDirective(S);
3150}
3151
3152DEF_TRAVERSE_STMT(OMPMetaDirective,
3153 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3154
3155DEF_TRAVERSE_STMT(OMPParallelDirective,
3156 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3157
3158DEF_TRAVERSE_STMT(OMPSimdDirective,
3159 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3160
3161DEF_TRAVERSE_STMT(OMPTileDirective,
3162 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3163
3164DEF_TRAVERSE_STMT(OMPStripeDirective,
3165 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3166
3167DEF_TRAVERSE_STMT(OMPUnrollDirective,
3168 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3169
3170DEF_TRAVERSE_STMT(OMPReverseDirective,
3171 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3172
3173DEF_TRAVERSE_STMT(OMPInterchangeDirective,
3174 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3175
3176DEF_TRAVERSE_STMT(OMPForDirective,
3177 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3178
3179DEF_TRAVERSE_STMT(OMPForSimdDirective,
3180 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3181
3182DEF_TRAVERSE_STMT(OMPSectionsDirective,
3183 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3184
3185DEF_TRAVERSE_STMT(OMPSectionDirective,
3186 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3187
3188DEF_TRAVERSE_STMT(OMPScopeDirective,
3189 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3190
3191DEF_TRAVERSE_STMT(OMPSingleDirective,
3192 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3193
3194DEF_TRAVERSE_STMT(OMPMasterDirective,
3195 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3196
3197DEF_TRAVERSE_STMT(OMPCriticalDirective, {
3198 TRY_TO(TraverseDeclarationNameInfo(S->getDirectiveName()));
3199 TRY_TO(TraverseOMPExecutableDirective(S));
3200})
3201
3202DEF_TRAVERSE_STMT(OMPParallelForDirective,
3203 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3204
3205DEF_TRAVERSE_STMT(OMPParallelForSimdDirective,
3206 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3207
3208DEF_TRAVERSE_STMT(OMPParallelMasterDirective,
3209 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3210
3211DEF_TRAVERSE_STMT(OMPParallelMaskedDirective,
3212 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3213
3214DEF_TRAVERSE_STMT(OMPParallelSectionsDirective,
3215 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3216
3217DEF_TRAVERSE_STMT(OMPTaskDirective,
3218 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3219
3220DEF_TRAVERSE_STMT(OMPTaskyieldDirective,
3221 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3222
3223DEF_TRAVERSE_STMT(OMPBarrierDirective,
3224 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3225
3226DEF_TRAVERSE_STMT(OMPTaskwaitDirective,
3227 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3228
3229DEF_TRAVERSE_STMT(OMPTaskgroupDirective,
3230 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3231
3232DEF_TRAVERSE_STMT(OMPCancellationPointDirective,
3233 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3234
3235DEF_TRAVERSE_STMT(OMPCancelDirective,
3236 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3237
3238DEF_TRAVERSE_STMT(OMPFlushDirective,
3239 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3240
3241DEF_TRAVERSE_STMT(OMPDepobjDirective,
3242 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3243
3244DEF_TRAVERSE_STMT(OMPScanDirective,
3245 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3246
3247DEF_TRAVERSE_STMT(OMPOrderedDirective,
3248 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3249
3250DEF_TRAVERSE_STMT(OMPAtomicDirective,
3251 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3252
3253DEF_TRAVERSE_STMT(OMPTargetDirective,
3254 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3255
3256DEF_TRAVERSE_STMT(OMPTargetDataDirective,
3257 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3258
3259DEF_TRAVERSE_STMT(OMPTargetEnterDataDirective,
3260 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3261
3262DEF_TRAVERSE_STMT(OMPTargetExitDataDirective,
3263 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3264
3265DEF_TRAVERSE_STMT(OMPTargetParallelDirective,
3266 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3267
3268DEF_TRAVERSE_STMT(OMPTargetParallelForDirective,
3269 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3270
3271DEF_TRAVERSE_STMT(OMPTeamsDirective,
3272 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3273
3274DEF_TRAVERSE_STMT(OMPTargetUpdateDirective,
3275 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3276
3277DEF_TRAVERSE_STMT(OMPTaskLoopDirective,
3278 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3279
3280DEF_TRAVERSE_STMT(OMPTaskLoopSimdDirective,
3281 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3282
3283DEF_TRAVERSE_STMT(OMPMasterTaskLoopDirective,
3284 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3285
3286DEF_TRAVERSE_STMT(OMPMasterTaskLoopSimdDirective,
3287 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3288
3289DEF_TRAVERSE_STMT(OMPParallelMasterTaskLoopDirective,
3290 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3291
3292DEF_TRAVERSE_STMT(OMPParallelMasterTaskLoopSimdDirective,
3293 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3294
3295DEF_TRAVERSE_STMT(OMPMaskedTaskLoopDirective,
3296 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3297
3298DEF_TRAVERSE_STMT(OMPMaskedTaskLoopSimdDirective,
3299 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3300
3301DEF_TRAVERSE_STMT(OMPParallelMaskedTaskLoopDirective,
3302 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3303
3304DEF_TRAVERSE_STMT(OMPParallelMaskedTaskLoopSimdDirective,
3305 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3306
3307DEF_TRAVERSE_STMT(OMPDistributeDirective,
3308 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3309
3310DEF_TRAVERSE_STMT(OMPDistributeParallelForDirective,
3311 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3312
3313DEF_TRAVERSE_STMT(OMPDistributeParallelForSimdDirective,
3314 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3315
3316DEF_TRAVERSE_STMT(OMPDistributeSimdDirective,
3317 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3318
3319DEF_TRAVERSE_STMT(OMPTargetParallelForSimdDirective,
3320 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3321
3322DEF_TRAVERSE_STMT(OMPTargetSimdDirective,
3323 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3324
3325DEF_TRAVERSE_STMT(OMPTeamsDistributeDirective,
3326 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3327
3328DEF_TRAVERSE_STMT(OMPTeamsDistributeSimdDirective,
3329 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3330
3331DEF_TRAVERSE_STMT(OMPTeamsDistributeParallelForSimdDirective,
3332 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3333
3334DEF_TRAVERSE_STMT(OMPTeamsDistributeParallelForDirective,
3335 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3336
3337DEF_TRAVERSE_STMT(OMPTargetTeamsDirective,
3338 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3339
3340DEF_TRAVERSE_STMT(OMPTargetTeamsDistributeDirective,
3341 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3342
3343DEF_TRAVERSE_STMT(OMPTargetTeamsDistributeParallelForDirective,
3344 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3345
3346DEF_TRAVERSE_STMT(OMPTargetTeamsDistributeParallelForSimdDirective,
3347 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3348
3349DEF_TRAVERSE_STMT(OMPTargetTeamsDistributeSimdDirective,
3350 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3351
3352DEF_TRAVERSE_STMT(OMPInteropDirective,
3353 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3354
3355DEF_TRAVERSE_STMT(OMPDispatchDirective,
3356 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3357
3358DEF_TRAVERSE_STMT(OMPMaskedDirective,
3359 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3360
3361DEF_TRAVERSE_STMT(OMPGenericLoopDirective,
3362 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3363
3364DEF_TRAVERSE_STMT(OMPTeamsGenericLoopDirective,
3365 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3366
3367DEF_TRAVERSE_STMT(OMPTargetTeamsGenericLoopDirective,
3368 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3369
3370DEF_TRAVERSE_STMT(OMPParallelGenericLoopDirective,
3371 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3372
3373DEF_TRAVERSE_STMT(OMPTargetParallelGenericLoopDirective,
3374 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3375
3376DEF_TRAVERSE_STMT(OMPAssumeDirective,
3377 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3378
3379DEF_TRAVERSE_STMT(OMPErrorDirective,
3380 { TRY_TO(TraverseOMPExecutableDirective(S)); })
3381
3382// OpenMP clauses.
3383template <typename Derived>
3384bool RecursiveASTVisitor<Derived>::TraverseOMPClause(OMPClause *C) {
3385 if (!C)
3386 return true;
3387 switch (C->getClauseKind()) {
3388#define GEN_CLANG_CLAUSE_CLASS
3389#define CLAUSE_CLASS(Enum, Str, Class) \
3390 case llvm::omp::Clause::Enum: \
3391 TRY_TO(Visit##Class(static_cast<Class *>(C))); \
3392 break;
3393#define CLAUSE_NO_CLASS(Enum, Str) \
3394 case llvm::omp::Clause::Enum: \
3395 break;
3396#include "llvm/Frontend/OpenMP/OMP.inc"
3397 }
3398 return true;
3399}
3400
3401template <typename Derived>
3402bool RecursiveASTVisitor<Derived>::VisitOMPClauseWithPreInit(
3403 OMPClauseWithPreInit *Node) {
3404 TRY_TO(TraverseStmt(Node->getPreInitStmt()));
3405 return true;
3406}
3407
3408template <typename Derived>
3409bool RecursiveASTVisitor<Derived>::VisitOMPClauseWithPostUpdate(
3411 TRY_TO(VisitOMPClauseWithPreInit(Node));
3412 TRY_TO(TraverseStmt(Node->getPostUpdateExpr()));
3413 return true;
3414}
3415
3416template <typename Derived>
3419 TRY_TO(TraverseStmt(C->getAllocator()));
3420 return true;
3421}
3422
3423template <typename Derived>
3425 TRY_TO(TraverseStmt(C->getAllocator()));
3426 TRY_TO(VisitOMPClauseList(C));
3427 return true;
3428}
3429
3430template <typename Derived>
3432 TRY_TO(VisitOMPClauseWithPreInit(C));
3433 TRY_TO(TraverseStmt(C->getCondition()));
3434 return true;
3435}
3436
3437template <typename Derived>
3439 TRY_TO(VisitOMPClauseWithPreInit(C));
3440 TRY_TO(TraverseStmt(C->getCondition()));
3441 return true;
3442}
3443
3444template <typename Derived>
3445bool
3447 TRY_TO(VisitOMPClauseWithPreInit(C));
3448 TRY_TO(TraverseStmt(C->getNumThreads()));
3449 return true;
3450}
3451
3452template <typename Derived>
3454 TRY_TO(TraverseStmt(C->getAlignment()));
3455 return true;
3456}
3457
3458template <typename Derived>
3460 TRY_TO(TraverseStmt(C->getSafelen()));
3461 return true;
3462}
3463
3464template <typename Derived>
3466 TRY_TO(TraverseStmt(C->getSimdlen()));
3467 return true;
3468}
3469
3470template <typename Derived>
3472 for (Expr *E : C->getSizesRefs())
3473 TRY_TO(TraverseStmt(E));
3474 return true;
3475}
3476
3477template <typename Derived>
3480 for (Expr *E : C->getArgsRefs())
3481 TRY_TO(TraverseStmt(E));
3482 return true;
3483}
3484
3485template <typename Derived>
3487 return true;
3488}
3489
3490template <typename Derived>
3492 TRY_TO(TraverseStmt(C->getFactor()));
3493 return true;
3494}
3495
3496template <typename Derived>
3497bool
3499 TRY_TO(TraverseStmt(C->getNumForLoops()));
3500 return true;
3501}
3502
3503template <typename Derived>
3505 return true;
3506}
3507
3508template <typename Derived>
3510 return true;
3511}
3512
3513template <typename Derived>
3516 return true;
3517}
3518
3519template <typename Derived>
3522 return true;
3523}
3524
3525template <typename Derived>
3528 return true;
3529}
3530
3531template <typename Derived>
3534 return true;
3535}
3536
3537template <typename Derived>
3540 return true;
3541}
3542
3543template <typename Derived>
3545 return true;
3546}
3547
3548template <typename Derived>
3550 return true;
3551}
3552
3553template <typename Derived>
3555 return true;
3556}
3557
3558template <typename Derived>
3560 TRY_TO(TraverseStmt(C->getMessageString()));
3561 return true;
3562}
3563
3564template <typename Derived>
3565bool
3567 TRY_TO(VisitOMPClauseWithPreInit(C));
3568 TRY_TO(TraverseStmt(C->getChunkSize()));
3569 return true;
3570}
3571
3572template <typename Derived>
3574 TRY_TO(TraverseStmt(C->getNumForLoops()));
3575 return true;
3576}
3577
3578template <typename Derived>
3580 return true;
3581}
3582
3583template <typename Derived>
3585 return true;
3586}
3587
3588template <typename Derived>
3589bool
3591 return true;
3592}
3593
3594template <typename Derived>
3596 return true;
3597}
3598
3599template <typename Derived>
3601 return true;
3602}
3603
3604template <typename Derived>
3606 return true;
3607}
3608
3609template <typename Derived>
3611 return true;
3612}
3613
3614template <typename Derived>
3616 return true;
3617}
3618
3619template <typename Derived>
3621 return true;
3622}
3623
3624template <typename Derived>
3626 return true;
3627}
3628
3629template <typename Derived>
3631 return true;
3632}
3633
3634template <typename Derived>
3636 return true;
3637}
3638
3639template <typename Derived>
3641 return true;
3642}
3643
3644template <typename Derived>
3646 return true;
3647}
3648
3649template <typename Derived>
3651 return true;
3652}
3653
3654template <typename Derived>
3657 return true;
3658}
3659
3660template <typename Derived>
3663 return true;
3664}
3665
3666template <typename Derived>
3669 return true;
3670}
3671
3672template <typename Derived>
3674 return true;
3675}
3676
3677template <typename Derived>
3679 return true;
3680}
3681
3682template <typename Derived>
3684 return true;
3685}
3686
3687template <typename Derived>
3689 return true;
3690}
3691
3692template <typename Derived>
3694 return true;
3695}
3696
3697template <typename Derived>
3699 return true;
3700}
3701
3702template <typename Derived>
3704 return true;
3705}
3706
3707template <typename Derived>
3709 TRY_TO(VisitOMPClauseList(C));
3710 return true;
3711}
3712
3713template <typename Derived>
3715 TRY_TO(TraverseStmt(C->getInteropVar()));
3716 return true;
3717}
3718
3719template <typename Derived>
3721 TRY_TO(TraverseStmt(C->getInteropVar()));
3722 return true;
3723}
3724
3725template <typename Derived>
3728 TRY_TO(VisitOMPClauseWithPreInit(C));
3729 TRY_TO(TraverseStmt(C->getCondition()));
3730 return true;
3731}
3732
3733template <typename Derived>
3736 TRY_TO(VisitOMPClauseWithPreInit(C));
3737 TRY_TO(TraverseStmt(C->getCondition()));
3738 return true;
3739}
3740
3741template <typename Derived>
3742template <typename T>
3743bool RecursiveASTVisitor<Derived>::VisitOMPClauseList(T *Node) {
3744 for (auto *E : Node->varlist()) {
3745 TRY_TO(TraverseStmt(E));
3746 }
3747 return true;
3748}
3749
3750template <typename Derived>
3753 TRY_TO(VisitOMPClauseList(C));
3754 return true;
3755}
3756
3757template <typename Derived>
3760 TRY_TO(VisitOMPClauseList(C));
3761 return true;
3762}
3763
3764template <typename Derived>
3766 TRY_TO(VisitOMPClauseList(C));
3767 for (auto *E : C->private_copies()) {
3768 TRY_TO(TraverseStmt(E));
3769 }
3770 return true;
3771}
3772
3773template <typename Derived>
3776 TRY_TO(VisitOMPClauseList(C));
3777 TRY_TO(VisitOMPClauseWithPreInit(C));
3778 for (auto *E : C->private_copies()) {
3779 TRY_TO(TraverseStmt(E));
3780 }
3781 for (auto *E : C->inits()) {
3782 TRY_TO(TraverseStmt(E));
3783 }
3784 return true;
3785}
3786
3787template <typename Derived>
3790 TRY_TO(VisitOMPClauseList(C));
3791 TRY_TO(VisitOMPClauseWithPostUpdate(C));
3792 for (auto *E : C->private_copies()) {
3793 TRY_TO(TraverseStmt(E));
3794 }
3795 for (auto *E : C->source_exprs()) {
3796 TRY_TO(TraverseStmt(E));
3797 }
3798 for (auto *E : C->destination_exprs()) {
3799 TRY_TO(TraverseStmt(E));
3800 }
3801 for (auto *E : C->assignment_ops()) {
3802 TRY_TO(TraverseStmt(E));
3803 }
3804 return true;
3805}
3806
3807template <typename Derived>
3809 TRY_TO(VisitOMPClauseList(C));
3810 return true;
3811}
3812
3813template <typename Derived>
3815 TRY_TO(TraverseStmt(C->getStep()));
3816 TRY_TO(TraverseStmt(C->getCalcStep()));
3817 TRY_TO(VisitOMPClauseList(C));
3818 TRY_TO(VisitOMPClauseWithPostUpdate(C));
3819 for (auto *E : C->privates()) {
3820 TRY_TO(TraverseStmt(E));
3821 }
3822 for (auto *E : C->inits()) {
3823 TRY_TO(TraverseStmt(E));
3824 }
3825 for (auto *E : C->updates()) {
3826 TRY_TO(TraverseStmt(E));
3827 }
3828 for (auto *E : C->finals()) {
3829 TRY_TO(TraverseStmt(E));
3830 }
3831 return true;
3832}
3833
3834template <typename Derived>
3836 TRY_TO(TraverseStmt(C->getAlignment()));
3837 TRY_TO(VisitOMPClauseList(C));
3838 return true;
3839}
3840
3841template <typename Derived>
3843 TRY_TO(VisitOMPClauseList(C));
3844 for (auto *E : C->source_exprs()) {
3845 TRY_TO(TraverseStmt(E));
3846 }
3847 for (auto *E : C->destination_exprs()) {
3848 TRY_TO(TraverseStmt(E));
3849 }
3850 for (auto *E : C->assignment_ops()) {
3851 TRY_TO(TraverseStmt(E));
3852 }
3853 return true;
3854}
3855
3856template <typename Derived>
3859 TRY_TO(VisitOMPClauseList(C));
3860 for (auto *E : C->source_exprs()) {
3861 TRY_TO(TraverseStmt(E));
3862 }
3863 for (auto *E : C->destination_exprs()) {
3864 TRY_TO(TraverseStmt(E));
3865 }
3866 for (auto *E : C->assignment_ops()) {
3867 TRY_TO(TraverseStmt(E));
3868 }
3869 return true;
3870}
3871
3872template <typename Derived>
3873bool
3875 TRY_TO(TraverseNestedNameSpecifierLoc(C->getQualifierLoc()));
3876 TRY_TO(TraverseDeclarationNameInfo(C->getNameInfo()));
3877 TRY_TO(VisitOMPClauseList(C));
3878 TRY_TO(VisitOMPClauseWithPostUpdate(C));
3879 for (auto *E : C->privates()) {
3880 TRY_TO(TraverseStmt(E));
3881 }
3882 for (auto *E : C->lhs_exprs()) {
3883 TRY_TO(TraverseStmt(E));
3884 }
3885 for (auto *E : C->rhs_exprs()) {
3886 TRY_TO(TraverseStmt(E));
3887 }
3888 for (auto *E : C->reduction_ops()) {
3889 TRY_TO(TraverseStmt(E));
3890 }
3891 if (C->getModifier() == OMPC_REDUCTION_inscan) {
3892 for (auto *E : C->copy_ops()) {
3893 TRY_TO(TraverseStmt(E));
3894 }
3895 for (auto *E : C->copy_array_temps()) {
3896 TRY_TO(TraverseStmt(E));
3897 }
3898 for (auto *E : C->copy_array_elems()) {
3899 TRY_TO(TraverseStmt(E));
3900 }
3901 }
3902 return true;
3903}
3904
3905template <typename Derived>
3908 TRY_TO(TraverseNestedNameSpecifierLoc(C->getQualifierLoc()));
3909 TRY_TO(TraverseDeclarationNameInfo(C->getNameInfo()));
3910 TRY_TO(VisitOMPClauseList(C));
3911 TRY_TO(VisitOMPClauseWithPostUpdate(C));
3912 for (auto *E : C->privates()) {
3913 TRY_TO(TraverseStmt(E));
3914 }
3915 for (auto *E : C->lhs_exprs()) {
3916 TRY_TO(TraverseStmt(E));
3917 }
3918 for (auto *E : C->rhs_exprs()) {
3919 TRY_TO(TraverseStmt(E));
3920 }
3921 for (auto *E : C->reduction_ops()) {
3922 TRY_TO(TraverseStmt(E));
3923 }
3924 return true;
3925}
3926
3927template <typename Derived>
3930 TRY_TO(TraverseNestedNameSpecifierLoc(C->getQualifierLoc()));
3931 TRY_TO(TraverseDeclarationNameInfo(C->getNameInfo()));
3932 TRY_TO(VisitOMPClauseList(C));
3933 TRY_TO(VisitOMPClauseWithPostUpdate(C));
3934 for (auto *E : C->privates()) {
3935 TRY_TO(TraverseStmt(E));
3936 }
3937 for (auto *E : C->lhs_exprs()) {
3938 TRY_TO(TraverseStmt(E));
3939 }
3940 for (auto *E : C->rhs_exprs()) {
3941 TRY_TO(TraverseStmt(E));
3942 }
3943 for (auto *E : C->reduction_ops()) {
3944 TRY_TO(TraverseStmt(E));
3945 }
3946 for (auto *E : C->taskgroup_descriptors())
3947 TRY_TO(TraverseStmt(E));
3948 return true;
3949}
3950
3951template <typename Derived>
3953 TRY_TO(VisitOMPClauseList(C));
3954 return true;
3955}
3956
3957template <typename Derived>
3959 TRY_TO(TraverseStmt(C->getDepobj()));
3960 return true;
3961}
3962
3963template <typename Derived>
3965 TRY_TO(VisitOMPClauseList(C));
3966 return true;
3967}
3968
3969template <typename Derived>
3971 TRY_TO(VisitOMPClauseWithPreInit(C));
3972 TRY_TO(TraverseStmt(C->getDevice()));
3973 return true;
3974}
3975
3976template <typename Derived>
3978 TRY_TO(VisitOMPClauseList(C));
3979 return true;
3980}
3981
3982template <typename Derived>
3985 TRY_TO(VisitOMPClauseList(C));
3986 TRY_TO(VisitOMPClauseWithPreInit(C));
3987 return true;
3988}
3989
3990template <typename Derived>
3993 TRY_TO(VisitOMPClauseList(C));
3994 TRY_TO(VisitOMPClauseWithPreInit(C));
3995 return true;
3996}
3997
3998template <typename Derived>
4001 TRY_TO(VisitOMPClauseWithPreInit(C));
4002 TRY_TO(TraverseStmt(C->getPriority()));
4003 return true;
4004}
4005
4006template <typename Derived>
4009 TRY_TO(VisitOMPClauseWithPreInit(C));
4010 TRY_TO(TraverseStmt(C->getGrainsize()));
4011 return true;
4012}
4013
4014template <typename Derived>
4017 TRY_TO(VisitOMPClauseWithPreInit(C));
4018 TRY_TO(TraverseStmt(C->getNumTasks()));
4019 return true;
4020}
4021
4022template <typename Derived>
4024 TRY_TO(TraverseStmt(C->getHint()));
4025 return true;
4026}
4027
4028template <typename Derived>
4031 TRY_TO(VisitOMPClauseWithPreInit(C));
4032 TRY_TO(TraverseStmt(C->getChunkSize()));
4033 return true;
4034}
4035
4036template <typename Derived>
4037bool
4039 return true;
4040}
4041
4042template <typename Derived>
4044 TRY_TO(VisitOMPClauseList(C));
4045 return true;
4046}
4047
4048template <typename Derived>
4050 TRY_TO(VisitOMPClauseList(C));
4051 return true;
4052}
4053
4054template <typename Derived>
4057 TRY_TO(VisitOMPClauseList(C));
4058 return true;
4059}
4060
4061template <typename Derived>
4064 TRY_TO(VisitOMPClauseList(C));
4065 return true;
4066}
4067
4068template <typename Derived>
4071 TRY_TO(VisitOMPClauseList(C));
4072 return true;
4073}
4074
4075template <typename Derived>
4078 TRY_TO(VisitOMPClauseList(C));
4079 return true;
4080}
4081
4082template <typename Derived>
4085 TRY_TO(VisitOMPClauseList(C));
4086 for (auto *E : C->private_refs()) {
4087 TRY_TO(TraverseStmt(E));
4088 }
4089 return true;
4090}
4091
4092template <typename Derived>
4094 return true;
4095}
4096
4097template <typename Derived>
4099 TRY_TO(TraverseStmt(C->getEventHandler()));
4100 return true;
4101}
4102
4103template <typename Derived>
4106 for (unsigned I = 0, E = C->getNumberOfAllocators(); I < E; ++I) {
4107 const OMPUsesAllocatorsClause::Data Data = C->getAllocatorData(I);
4108 TRY_TO(TraverseStmt(Data.Allocator));
4109 TRY_TO(TraverseStmt(Data.AllocatorTraits));
4110 }
4111 return true;
4112}
4113
4114template <typename Derived>
4117 TRY_TO(TraverseStmt(C->getModifier()));
4118 for (Expr *E : C->varlist())
4119 TRY_TO(TraverseStmt(E));
4120 return true;
4121}
4122
4123template <typename Derived>
4125 TRY_TO(VisitOMPClauseWithPreInit(C));
4126 TRY_TO(TraverseStmt(C->getThreadID()));
4127 return true;
4128}
4129
4130template <typename Derived>
4132 return true;
4133}
4134
4135template <typename Derived>
4138 TRY_TO(VisitOMPClauseWithPreInit(C));
4139 TRY_TO(TraverseStmt(C->getSize()));
4140 return true;
4141}
4142
4143template <typename Derived>
4146 TRY_TO(VisitOMPClauseList(C));
4147 return true;
4148}
4149
4150template <typename Derived>
4153 return true;
4154}
4155
4156template <typename Derived>
4158 return true;
4159}
4160
4161template <typename Derived>
4162bool RecursiveASTVisitor<Derived>::TraverseOpenACCConstructStmt(
4164 TRY_TO(VisitOpenACCClauseList(C->clauses()));
4165 return true;
4166}
4167
4168template <typename Derived>
4169bool RecursiveASTVisitor<Derived>::TraverseOpenACCAssociatedStmtConstruct(
4171 TRY_TO(TraverseOpenACCConstructStmt(S));
4172 TRY_TO(TraverseStmt(S->getAssociatedStmt()));
4173 return true;
4174}
4175
4176template <typename Derived>
4177bool RecursiveASTVisitor<Derived>::VisitOpenACCClause(const OpenACCClause *C) {
4178 for (const Stmt *Child : C->children())
4179 TRY_TO(TraverseStmt(const_cast<Stmt *>(Child)));
4180 return true;
4181}
4182
4183template <typename Derived>
4184bool RecursiveASTVisitor<Derived>::VisitOpenACCClauseList(
4186
4187 for (const auto *C : Clauses)
4188 TRY_TO(VisitOpenACCClause(C));
4189 return true;
4190}
4191
4193 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4194DEF_TRAVERSE_STMT(OpenACCLoopConstruct,
4195 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4196DEF_TRAVERSE_STMT(OpenACCCombinedConstruct,
4197 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4198DEF_TRAVERSE_STMT(OpenACCDataConstruct,
4199 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4200DEF_TRAVERSE_STMT(OpenACCEnterDataConstruct,
4201 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4202DEF_TRAVERSE_STMT(OpenACCExitDataConstruct,
4203 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4204DEF_TRAVERSE_STMT(OpenACCHostDataConstruct,
4205 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4206DEF_TRAVERSE_STMT(OpenACCWaitConstruct, {
4207 if (S->hasDevNumExpr())
4208 TRY_TO(TraverseStmt(S->getDevNumExpr()));
4209 for (auto *E : S->getQueueIdExprs())
4210 TRY_TO(TraverseStmt(E));
4211 TRY_TO(VisitOpenACCClauseList(S->clauses()));
4212})
4213DEF_TRAVERSE_STMT(OpenACCInitConstruct,
4214 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4215DEF_TRAVERSE_STMT(OpenACCShutdownConstruct,
4216 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4217DEF_TRAVERSE_STMT(OpenACCSetConstruct,
4218 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4219DEF_TRAVERSE_STMT(OpenACCUpdateConstruct,
4220 { TRY_TO(VisitOpenACCClauseList(S->clauses())); })
4221DEF_TRAVERSE_STMT(OpenACCAtomicConstruct,
4222 { TRY_TO(TraverseOpenACCAssociatedStmtConstruct(S)); })
4223DEF_TRAVERSE_STMT(OpenACCCacheConstruct, {
4224 for (auto *E : S->getVarList())
4225 TRY_TO(TraverseStmt(E));
4226})
4227
4228// Traverse HLSL: Out argument expression
4230
4231// FIXME: look at the following tricky-seeming exprs to see if we
4232// need to recurse on anything. These are ones that have methods
4233// returning decls or qualtypes or nestednamespecifier -- though I'm
4234// not sure if they own them -- or just seemed very complicated, or
4235// had lots of sub-types to explore.
4236//
4237// VisitOverloadExpr and its children: recurse on template args? etc?
4238
4239// FIXME: go through all the stmts and exprs again, and see which of them
4240// create new types, and recurse on the types (TypeLocs?) of those.
4241// Candidates:
4242//
4243// http://clang.llvm.org/doxygen/classclang_1_1CXXTypeidExpr.html
4244// http://clang.llvm.org/doxygen/classclang_1_1UnaryExprOrTypeTraitExpr.html
4245// http://clang.llvm.org/doxygen/classclang_1_1TypesCompatibleExpr.html
4246// Every class that has getQualifier.
4247
4248#undef DEF_TRAVERSE_STMT
4249#undef TRAVERSE_STMT
4250#undef TRAVERSE_STMT_BASE
4251
4252#undef TRY_TO
4253
4254} // end namespace clang
4255
4256#endif // LLVM_CLANG_AST_RECURSIVEASTVISITOR_H
This file provides AST data structures related to concepts.
#define STMT(DERIVED, BASE)
Definition ASTFwd.h:23
#define TYPE(DERIVED, BASE)
Definition ASTFwd.h:26
bool TraverseNestedNameSpecifierLoc(NestedNameSpecifierLoc QualifierLoc)
Defines the C++ Decl subclasses, other than those for templates (found in DeclTemplate....
This file defines OpenACC nodes for declarative directives.
This file defines OpenMP nodes for declarative directives.
Defines the C++ template declaration subclasses.
#define DEF_TRAVERSE_TMPL_INST(kind)
Defines the clang::Expr interface and subclasses for C++ expressions.
Defines Expressions and AST nodes for C++2a concepts.
Forward-declares and imports various common LLVM datatypes that clang wants to use unqualified.
Defines the LambdaCapture class.
This file defines OpenMP AST classes for clauses.
Defines some OpenMP-specific enums and functions.
#define DEF_TRAVERSE_TMPL_PART_SPEC_DECL(TMPLDECLKIND, DECLKIND)
#define TRAVERSE_STMT_BASE(NAME, CLASS, VAR, QUEUE)
#define DEF_TRAVERSE_TYPE(TYPE, CODE)
#define DEF_TRAVERSE_TYPELOC(TYPE, CODE)
#define TRY_TO_TRAVERSE_OR_ENQUEUE_STMT(S)
#define DEF_TRAVERSE_TMPL_SPEC_DECL(TMPLDECLKIND, DECLKIND)
#define DEF_TRAVERSE_DECL(DECL, CODE)
#define DEF_TRAVERSE_STMT(STMT, CODE)
#define DEF_TRAVERSE_TMPL_DECL(TMPLDECLKIND)
#define TRY_TO(CALL_EXPR)
Defines various enumerations that describe declaration and type specifiers.
Defines the Objective-C statement AST node classes.
This file defines OpenACC AST classes for statement-level contructs.
This file defines OpenMP AST classes for executable directives and clauses.
This file defines SYCL AST classes used to represent calls to SYCL kernels.
Defines the clang::TypeLoc interface and its subclasses.
C Language Family Type Representation.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
Definition ASTContext.h:188
TranslationUnitDecl * getTranslationUnitDecl() const
Represents an access specifier followed by colon ':'.
Definition DeclCXX.h:86
AddrLabelExpr - The GNU address of label extension, representing &&label.
Definition Expr.h:4486
Represents a type which was implicitly adjusted by the semantic engine for arbitrary reasons.
Definition TypeBase.h:3489
Represents the index of the current element of an array being initialized by an ArrayInitLoopExpr.
Definition Expr.h:5957
Represents a loop initializing the elements of an array.
Definition Expr.h:5904
This class represents BOTH the OpenMP Array Section and OpenACC 'subarray', with a boolean differenti...
Definition Expr.h:7092
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
Definition Expr.h:2723
Wrapper for source info for arrays.
Definition TypeLoc.h:1757
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
Definition ExprCXX.h:2990
AsTypeExpr - Clang builtin function __builtin_astype [OpenCL 6.2.4.2] This AST node provides support ...
Definition Expr.h:6621
AtomicExpr - Variadic atomic builtins: __atomic_exchange, __atomic_fetch_*, __atomic_load,...
Definition Expr.h:6816
Attr - This represents one attribute.
Definition Attr.h:44
Represents an attribute applied to a statement.
Definition Stmt.h:2203
BinaryConditionalOperator - The GNU extension to the conditional operator which allows the middle ope...
Definition Expr.h:4389
A builtin binary operation expression such as "x + y" or "x <= y".
Definition Expr.h:3974
A binding in a decomposition declaration.
Definition DeclCXX.h:4179
A fixed int type of a specified bitwidth.
Definition TypeBase.h:8137
Represents a block literal declaration, which is like an unnamed FunctionDecl.
Definition Decl.h:4634
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
Definition Expr.h:6560
Pointer to a block type.
Definition TypeBase.h:3540
BreakStmt - This represents a break.
Definition Stmt.h:3135
Represents a C++2a __builtin_bit_cast(T, v) expression.
Definition ExprCXX.h:5470
Represents the builtin template declaration which is used to implement __make_integer_seq and other b...
This class is used for builtin types like 'int'.
Definition TypeBase.h:3164
CStyleCastExpr - An explicit cast in C (C99 6.5.4) or a C-style cast in C++ (C++ [expr....
Definition Expr.h:3905
Represents a call to a CUDA kernel function.
Definition ExprCXX.h:234
A C++ addrspace_cast expression (currently only enabled for OpenCL).
Definition ExprCXX.h:604
Represents a base class of a C++ class.
Definition DeclCXX.h:146
Represents binding an expression to a temporary.
Definition ExprCXX.h:1494
A boolean literal, per ([C++ lex.bool] Boolean literals).
Definition ExprCXX.h:723
CXXCatchStmt - This represents a C++ catch block.
Definition StmtCXX.h:28
A C++ const_cast expression (C++ [expr.const.cast]).
Definition ExprCXX.h:566
Represents a call to a C++ constructor.
Definition ExprCXX.h:1549
Represents a C++ constructor within a class.
Definition DeclCXX.h:2604
Represents a C++ conversion function within a class.
Definition DeclCXX.h:2937
Represents a C++ base or member initializer.
Definition DeclCXX.h:2369
Represents a C++ deduction guide declaration.
Definition DeclCXX.h:1979
A default argument (C++ [dcl.fct.default]).
Definition ExprCXX.h:1271
A use of a default initializer in a constructor or in aggregate initialization.
Definition ExprCXX.h:1378
Represents a delete expression for memory deallocation and destructor calls, e.g.
Definition ExprCXX.h:2620
Represents a C++ member access expression where the actual member referenced could not be resolved be...
Definition ExprCXX.h:3864
Represents a C++ destructor within a class.
Definition DeclCXX.h:2869
A C++ dynamic_cast expression (C++ [expr.dynamic.cast]).
Definition ExprCXX.h:481
Represents a folding of a pack over an operator.
Definition ExprCXX.h:5026
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
Definition StmtCXX.h:135
Represents an explicit C++ type conversion that uses "functional" notation (C++ [expr....
Definition ExprCXX.h:1833
Represents a call to an inherited base class constructor from an inheriting constructor.
Definition ExprCXX.h:1753
Represents a call to a member function that may be written either with member call syntax (e....
Definition ExprCXX.h:179
Represents a static or instance method of a struct/union/class.
Definition DeclCXX.h:2129
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
Definition ExprCXX.h:2349
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
Definition ExprCXX.h:4303
The null pointer literal (C++11 [lex.nullptr])
Definition ExprCXX.h:768
A call to an overloaded operator written using operator syntax.
Definition ExprCXX.h:84
Represents a list-initialization with parenthesis.
Definition ExprCXX.h:5135
Represents a C++ pseudo-destructor (C++ [expr.pseudo]).
Definition ExprCXX.h:2739
Represents a C++ struct/union/class.
Definition DeclCXX.h:258
A C++ reinterpret_cast expression (C++ [expr.reinterpret.cast]).
Definition ExprCXX.h:526
A rewritten comparison expression that was originally written using operator syntax.
Definition ExprCXX.h:286
An expression "T()" which creates an rvalue of a non-class type T.
Definition ExprCXX.h:2198
A C++ static_cast expression (C++ [expr.static.cast]).
Definition ExprCXX.h:436
Implicit construction of a std::initializer_list<T> object from an array temporary within list-initia...
Definition ExprCXX.h:800
Represents a C++ functional cast expression that builds a temporary object.
Definition ExprCXX.h:1901
Represents the this expression in C++.
Definition ExprCXX.h:1155
A C++ throw-expression (C++ [except.throw]).
Definition ExprCXX.h:1209
CXXTryStmt - A C++ try block, including all handlers.
Definition StmtCXX.h:69
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
Definition ExprCXX.h:848
Describes an explicit type conversion that uses functional notion but could not be resolved because o...
Definition ExprCXX.h:3738
A Microsoft C++ __uuidof expression, which gets the _GUID that corresponds to the supplied type or ex...
Definition ExprCXX.h:1069
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Definition Expr.h:2879
Represents the body of a CapturedStmt, and serves as its DeclContext.
Definition Decl.h:4906
This captures a statement into a function.
Definition Stmt.h:3886
CaseStmt - Represent a case statement.
Definition Stmt.h:1920
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Definition Expr.h:4784
Declaration of a class template.
Represents a 'co_await' expression.
Definition ExprCXX.h:5363
Complex values, per C99 6.2.5p11.
Definition TypeBase.h:3275
CompoundAssignOperator - For compound assignments (e.g.
Definition Expr.h:4236
CompoundLiteralExpr - [C99 6.5.2.5].
Definition Expr.h:3541
CompoundStmt - This represents a group of statements like { stmt stmt }.
Definition Stmt.h:1720
Declaration of a C++20 concept.
A reference to a concept and its template args, as it appears in the code.
Definition ASTConcept.h:126
Represents the specialization of a concept - evaluates to a prvalue of type bool.
ConditionalOperator - The ?
Definition Expr.h:4327
Represents the canonical version of C arrays with a specified constant size.
Definition TypeBase.h:3758
ConstantExpr - An expression that occurs in a constant context and optionally the result of evaluatin...
Definition Expr.h:1084
Represents a concrete matrix type with constant number of rows and columns.
Definition TypeBase.h:4371
Represents a shadow constructor declaration introduced into a class by a C++11 using-declaration that...
Definition DeclCXX.h:3671
ContinueStmt - This represents a continue.
Definition Stmt.h:3119
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
Definition Expr.h:4655
Represents a 'co_return' statement in the C++ Coroutines TS.
Definition StmtCXX.h:473
Represents the body of a coroutine.
Definition StmtCXX.h:320
Represents a 'co_yield' expression.
Definition ExprCXX.h:5444
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
Definition DeclBase.h:1449
DeclContext * getParent()
getParent - Returns the containing DeclContext.
Definition DeclBase.h:2109
A reference to a declared variable, function, enum, etc.
Definition Expr.h:1272
DeclStmt - Adaptor class for mixing declarations with statements and expressions.
Definition Stmt.h:1611
Decl - This represents one declaration (or definition), e.g.
Definition DeclBase.h:86
bool isImplicit() const
isImplicit - Indicates whether the declaration was implicitly generated by the implementation.
Definition DeclBase.h:593
Kind getKind() const
Definition DeclBase.h:442
TemplateDecl * getCXXDeductionGuideTemplate() const
If this name is the name of a C++ deduction guide, return the template associated with that name.
NameKind getNameKind() const
Determine what kind of name this is.
Represents a ValueDecl that came out of a declarator.
Definition Decl.h:779
A decomposition declaration.
Definition DeclCXX.h:4243
Represents a 'co_await' expression while the type of the promise is dependent.
Definition ExprCXX.h:5395
Provides information about a dependent function-template specialization declaration.
A qualified reference to a name whose declaration cannot yet be resolved.
Definition ExprCXX.h:3504
Represents an array type in C++ whose size is a value-dependent expression.
Definition TypeBase.h:4009
Represents an extended vector type where either the type or size is dependent.
Definition TypeBase.h:4099
Represents a vector type where either the type or size is dependent.
Definition TypeBase.h:4225
Represents a C99 designated initializer expression.
Definition Expr.h:5487
DoStmt - This represents a 'do/while' stmt.
Definition Stmt.h:2832
Represents a reference to emded data.
Definition Expr.h:5062
Represents an empty-declaration.
Definition Decl.h:5141
An instance of this object exists for each enum constant that is defined.
Definition Decl.h:3420
Represents an enum.
Definition Decl.h:4004
Represents a standard C++ module export declaration.
Definition Decl.h:5094
Represents an expression – generally a full-expression – that introduces cleanups to be run at the en...
Definition ExprCXX.h:3655
This represents one expression.
Definition Expr.h:112
An expression trait intrinsic.
Definition ExprCXX.h:3063
ExtVectorElementExpr - This represents access to specific elements of a vector, and may occur on the ...
Definition Expr.h:6500
Declaration context for names declared as extern "C" in C++.
Definition Decl.h:246
Represents a member of a struct/union/class.
Definition Decl.h:3157
ForStmt - This represents a 'for (init;cond;inc)' stmt.
Definition Stmt.h:2888
FriendDecl - Represents the declaration of a friend entity, which can be a function,...
Definition DeclFriend.h:54
Declaration of a friend template.
Represents a function declaration or definition.
Definition Decl.h:1999
Represents a K&R-style 'int foo()' function, which has no information available about its arguments.
Definition TypeBase.h:4842
Represents a reference to a function parameter pack, init-capture pack, or binding pack that has been...
Definition ExprCXX.h:4835
Represents a prototype with parameter type info, e.g.
Definition TypeBase.h:5264
Declaration of a template function.
Provides information about a function template specialization, which is a FunctionDecl that has been ...
This represents a GCC inline-assembly statement extension.
Definition Stmt.h:3395
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Definition Expr.h:4859
Represents a C11 generic selection.
Definition Expr.h:6114
AssociationTy< false > Association
Definition Expr.h:6345
GotoStmt - This represents a direct goto.
Definition Stmt.h:2969
HLSLBufferDecl - Represent a cbuffer or tbuffer declaration.
Definition Decl.h:5156
This class represents temporary values used to represent inout and out arguments in HLSL.
Definition Expr.h:7258
IfStmt - This represents an if/then/else.
Definition Stmt.h:2259
ImaginaryLiteral - We support imaginary integer and floating point literals, like "1....
Definition Expr.h:1733
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
Definition Expr.h:3789
Represents an implicitly-generated value initialization of an object of a given type.
Definition Expr.h:5993
Describes a module import declaration, which makes the contents of the named module visible in the cu...
Definition Decl.h:5015
Represents a C array with an unspecified size.
Definition TypeBase.h:3907
Represents a field injected from an anonymous union/struct into the parent scope.
Definition Decl.h:3464
IndirectGotoStmt - This represents an indirect goto.
Definition Stmt.h:3008
Describes an C or C++ initializer list.
Definition Expr.h:5235
Represents the declaration of a label.
Definition Decl.h:523
LabelStmt - Represents a label, which has a substatement.
Definition Stmt.h:2146
Describes the capture of a variable or of this, or of a C++1y init-capture.
A C++ lambda expression, which produces a function object (of unspecified type) that can be invoked l...
Definition ExprCXX.h:1970
Implicit declaration of a temporary that was materialized by a MaterializeTemporaryExpr and lifetime-...
Definition DeclCXX.h:3302
Represents a linkage specification.
Definition DeclCXX.h:3009
This represents a Microsoft inline-assembly statement extension.
Definition Stmt.h:3614
Representation of a Microsoft __if_exists or __if_not_exists statement with a dependent name.
Definition StmtCXX.h:253
A global _GUID constant.
Definition DeclCXX.h:4392
An instance of this class represents the declaration of a property member.
Definition DeclCXX.h:4338
A member reference to an MSPropertyDecl.
Definition ExprCXX.h:936
MS property subscript expression.
Definition ExprCXX.h:1007
Sugar type that represents a type that was qualified by a qualifier written as a macro invocation.
Definition TypeBase.h:6143
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
Definition ExprCXX.h:4914
MatrixSubscriptExpr - Matrix subscript expression for the MatrixType extension.
Definition Expr.h:2801
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
Definition Expr.h:3300
This represents a decl that may have a name.
Definition Decl.h:273
Represents a C++ namespace alias.
Definition DeclCXX.h:3195
Represent a C++ namespace.
Definition Decl.h:591
A C++ nested-name-specifier augmented with source location information.
NestedNameSpecifier getNestedNameSpecifier() const
Retrieve the nested-name-specifier to which this instance refers.
NamespaceAndPrefixLoc castAsNamespaceAndPrefix() const
For a nested-name-specifier that refers to a namespace, retrieve the namespace and its prefix.
TypeLoc castAsTypeLoc() const
For a nested-name-specifier that refers to a type, retrieve the type with source-location information...
Represents a C++ nested name specifier, such as "\::std::vector<int>::".
NamespaceAndPrefix getAsNamespaceAndPrefix() const
@ MicrosoftSuper
Microsoft's '__super' specifier, stored as a CXXRecordDecl* of the class it appeared in.
@ Global
The global specifier '::'. There is no stored value.
@ Namespace
A namespace-like entity, stored as a NamespaceBaseDecl*.
Represents a place-holder for an object not to be initialized by anything.
Definition Expr.h:5813
NonTypeTemplateParmDecl - Declares a non-type template parameter, e.g., "Size" in.
NullStmt - This is the null statement ";": C99 6.8.3p3.
Definition Stmt.h:1683
This represents the 'absent' clause in the 'pragma omp assume' directive.
This represents 'acq_rel' clause in the 'pragma omp atomic|flush' directives.
This represents 'acquire' clause in the 'pragma omp atomic|flush' directives.
This represents clause 'affinity' in the 'pragma omp task'-based directives.
This represents the 'align' clause in the 'pragma omp allocate' directive.
This represents clause 'aligned' in the 'pragma omp ...' directives.
This represents clause 'allocate' in the 'pragma omp ...' directives.
This represents 'pragma omp allocate ...' directive.
Definition DeclOpenMP.h:474
This represents 'allocator' clause in the 'pragma omp ...' directive.
An explicit cast in C or a C-style cast in C++, which uses the syntax ([s1][s2]......
Definition ExprOpenMP.h:24
This represents 'at' clause in the 'pragma omp error' directive.
This represents 'atomic_default_mem_order' clause in the 'pragma omp requires' directive.
This represents 'bind' clause in the 'pragma omp ...' directives.
This represents 'capture' clause in the 'pragma omp atomic' directive.
Pseudo declaration for capturing expressions.
Definition DeclOpenMP.h:383
Class that handles post-update expression for some clauses, like 'lastprivate', 'reduction' etc.
Class that handles pre-initialization statement for some clauses, like 'schedule',...
This is a basic class for representing single OpenMP clause.
This represents 'collapse' clause in the 'pragma omp ...' directive.
This represents 'compare' clause in the 'pragma omp atomic' directive.
This represents the 'contains' clause in the 'pragma omp assume' directive.
This represents clause 'copyin' in the 'pragma omp ...' directives.
This represents clause 'copyprivate' in the 'pragma omp ...' directives.
This represents 'pragma omp declare mapper ...' directive.
Definition DeclOpenMP.h:287
This represents 'pragma omp declare reduction ...' directive.
Definition DeclOpenMP.h:177
This represents 'default' clause in the 'pragma omp ...' directive.
This represents 'defaultmap' clause in the 'pragma omp ...' directive.
This represents implicit clause 'depend' for the 'pragma omp task' directive.
This represents implicit clause 'depobj' for the 'pragma omp depobj' directive.
This represents 'destroy' clause in the 'pragma omp depobj' directive or the 'pragma omp interop' dir...
This represents 'detach' clause in the 'pragma omp task' directive.
This represents 'device' clause in the 'pragma omp ...' directive.
This represents 'dist_schedule' clause in the 'pragma omp ...' directive.
This represents the 'doacross' clause for the 'pragma omp ordered' directive.
This represents 'dynamic_allocators' clause in the 'pragma omp requires' directive.
This represents clause 'exclusive' in the 'pragma omp scan' directive.
This represents 'fail' clause in the 'pragma omp atomic' directive.
This represents 'filter' clause in the 'pragma omp ...' directive.
This represents 'final' clause in the 'pragma omp ...' directive.
This represents clause 'firstprivate' in the 'pragma omp ...' directives.
This represents implicit clause 'flush' for the 'pragma omp flush' directive.
This represents clause 'from' in the 'pragma omp ...' directives.
Representation of the 'full' clause of the 'pragma omp unroll' directive.
This represents 'grainsize' clause in the 'pragma omp ...' directive.
This represents clause 'has_device_ptr' in the 'pragma omp ...' directives.
This represents 'hint' clause in the 'pragma omp ...' directive.
This represents the 'holds' clause in the 'pragma omp assume' directive.
This represents 'if' clause in the 'pragma omp ...' directive.
This represents clause 'in_reduction' in the 'pragma omp task' directives.
This represents clause 'inclusive' in the 'pragma omp scan' directive.
This represents the 'init' clause in 'pragma omp ...' directives.
This represents clause 'is_device_ptr' in the 'pragma omp ...' directives.
OpenMP 5.0 [2.1.6 Iterators] Iterators are identifiers that expand to multiple values in the clause o...
Definition ExprOpenMP.h:151
This represents clause 'lastprivate' in the 'pragma omp ...' directives.
This represents clause 'linear' in the 'pragma omp ...' directives.
This represents clause 'map' in the 'pragma omp ...' directives.
This represents 'mergeable' clause in the 'pragma omp ...' directive.
This represents the 'message' clause in the 'pragma omp error' and the 'pragma omp parallel' directiv...
This represents the 'no_openmp' clause in the 'pragma omp assume' directive.
This represents the 'no_openmp_constructs' clause in the.
This represents the 'no_openmp_routines' clause in the 'pragma omp assume' directive.
This represents the 'no_parallelism' clause in the 'pragma omp assume' directive.
This represents 'nocontext' clause in the 'pragma omp ...' directive.
This represents 'nogroup' clause in the 'pragma omp ...' directive.
This represents clause 'nontemporal' in the 'pragma omp ...' directives.
This represents 'novariants' clause in the 'pragma omp ...' directive.
This represents 'nowait' clause in the 'pragma omp ...' directive.
This represents 'num_tasks' clause in the 'pragma omp ...' directive.
This represents 'num_teams' clause in the 'pragma omp ...' directive.
This represents 'num_threads' clause in the 'pragma omp ...' directive.
This represents 'order' clause in the 'pragma omp ...' directive.
This represents 'ordered' clause in the 'pragma omp ...' directive.
Representation of the 'partial' clause of the 'pragma omp unroll' directive.
This class represents the 'permutation' clause in the 'pragma omp interchange' directive.
This represents 'priority' clause in the 'pragma omp ...' directive.
This represents clause 'private' in the 'pragma omp ...' directives.
This represents 'proc_bind' clause in the 'pragma omp ...' directive.
This represents 'read' clause in the 'pragma omp atomic' directive.
This represents clause 'reduction' in the 'pragma omp ...' directives.
This represents 'relaxed' clause in the 'pragma omp atomic' directives.
This represents 'release' clause in the 'pragma omp atomic|flush' directives.
This represents 'pragma omp requires...' directive.
Definition DeclOpenMP.h:417
This represents 'reverse_offload' clause in the 'pragma omp requires' directive.
This represents 'simd' clause in the 'pragma omp ...' directive.
This represents 'safelen' clause in the 'pragma omp ...' directive.
This represents 'schedule' clause in the 'pragma omp ...' directive.
This represents 'self_maps' clause in the 'pragma omp requires' directive.
This represents 'seq_cst' clause in the 'pragma omp atomic|flush' directives.
This represents the 'severity' clause in the 'pragma omp error' and the 'pragma omp parallel' directi...
This represents clause 'shared' in the 'pragma omp ...' directives.
This represents 'simdlen' clause in the 'pragma omp ...' directive.
This represents the 'sizes' clause in the 'pragma omp tile' directive.
This represents clause 'task_reduction' in the 'pragma omp taskgroup' directives.
This represents 'thread_limit' clause in the 'pragma omp ...' directive.
This represents 'pragma omp threadprivate ...' directive.
Definition DeclOpenMP.h:110
This represents 'threads' clause in the 'pragma omp ...' directive.
This represents clause 'to' in the 'pragma omp ...' directives.
This represents 'unified_address' clause in the 'pragma omp requires' directive.
This represents 'unified_shared_memory' clause in the 'pragma omp requires' directive.
This represents 'untied' clause in the 'pragma omp ...' directive.
This represents 'update' clause in the 'pragma omp atomic' directive.
This represents the 'use' clause in 'pragma omp ...' directives.
This represents clause 'use_device_addr' in the 'pragma omp ...' directives.
This represents clause 'use_device_ptr' in the 'pragma omp ...' directives.
This represents clause 'uses_allocators' in the 'pragma omp target'-based directives.
This represents 'weak' clause in the 'pragma omp atomic' directives.
This represents 'write' clause in the 'pragma omp atomic' directive.
This represents 'ompx_attribute' clause in a directive that might generate an outlined function.
This represents 'ompx_bare' clause in the 'pragma omp target teams ...' directive.
This represents 'ompx_dyn_cgroup_mem' clause in the 'pragma omp target ...' directive.
ObjCArrayLiteral - used for objective-c array containers; as in: @["Hello", NSApp,...
Definition ExprObjC.h:192
Represents Objective-C's @catch statement.
Definition StmtObjC.h:77
Represents a field declaration created by an @defs(...).
Definition DeclObjC.h:2030
Represents Objective-C's @finally statement.
Definition StmtObjC.h:127
Represents Objective-C's @synchronized statement.
Definition StmtObjC.h:303
Represents Objective-C's @throw statement.
Definition StmtObjC.h:358
Represents Objective-C's @try ... @catch ... @finally statement.
Definition StmtObjC.h:167
Represents Objective-C's @autoreleasepool Statement.
Definition StmtObjC.h:394
A runtime availability query.
Definition ExprObjC.h:1703
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
Definition ExprObjC.h:88
ObjCBoxedExpr - used for generalized expression boxing.
Definition ExprObjC.h:128
An Objective-C "bridged" cast expression, which casts between Objective-C pointers and C pointers,...
Definition ExprObjC.h:1643
ObjCCategoryDecl - Represents a category declaration.
Definition DeclObjC.h:2329
ObjCCategoryImplDecl - An object of this class encapsulates a category @implementation declaration.
Definition DeclObjC.h:2545
ObjCCompatibleAliasDecl - Represents alias of a class.
Definition DeclObjC.h:2775
ObjCDictionaryLiteral - AST node to represent objective-c dictionary literals; as in:"name" : NSUserN...
Definition ExprObjC.h:308
ObjCEncodeExpr, used for @encode in Objective-C.
Definition ExprObjC.h:409
Represents Objective-C's collection statement.
Definition StmtObjC.h:23
ObjCImplementationDecl - Represents a class definition - this is where method definitions are specifi...
Definition DeclObjC.h:2597
ObjCIndirectCopyRestoreExpr - Represents the passing of a function argument by indirect copy-restore ...
Definition ExprObjC.h:1582
Represents an ObjC class declaration.
Definition DeclObjC.h:1154
Represents typeof(type), a C23 feature and GCC extension, or `typeof_unqual(type),...
Definition TypeBase.h:7847
ObjCIsaExpr - Represent X->isa and X.isa when X is an ObjC 'id' type.
Definition ExprObjC.h:1498
ObjCIvarDecl - Represents an ObjC instance variable.
Definition DeclObjC.h:1952
ObjCIvarRefExpr - A reference to an ObjC instance variable.
Definition ExprObjC.h:548
An expression that sends a message to the given Objective-C object or class.
Definition ExprObjC.h:940
ObjCMethodDecl - Represents an instance or class method declaration.
Definition DeclObjC.h:140
Represents a pointer to an Objective C object.
Definition TypeBase.h:7903
Represents one property declaration in an Objective-C interface.
Definition DeclObjC.h:731
ObjCPropertyImplDecl - Represents implementation declaration of a property in a class or category imp...
Definition DeclObjC.h:2805
ObjCPropertyRefExpr - A dot-syntax expression to access an ObjC property.
Definition ExprObjC.h:616
Represents an Objective-C protocol declaration.
Definition DeclObjC.h:2084
ObjCProtocolExpr used for protocol expression in Objective-C.
Definition ExprObjC.h:504
ObjCSelectorExpr used for @selector in Objective-C.
Definition ExprObjC.h:454
ObjCStringLiteral, used for Objective-C string literals i.e.
Definition ExprObjC.h:52
ObjCSubscriptRefExpr - used for array and dictionary subscripting.
Definition ExprObjC.h:839
Represents the declaration of an Objective-C type parameter.
Definition DeclObjC.h:578
Stores a list of Objective-C type parameters for a parameterized class or a category/extension thereo...
Definition DeclObjC.h:662
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Definition Expr.h:2529
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
Definition Expr.h:1180
This is a base class for any OpenACC statement-level constructs that have an associated statement.
Definition StmtOpenACC.h:81
This expression type represents an asterisk in an OpenACC Size-Expr, used in the 'tile' and 'gang' cl...
Definition Expr.h:2092
This is the base type for all OpenACC Clauses.
This is the base class for an OpenACC statement-level construct, other construct types are expected t...
Definition StmtOpenACC.h:26
Represents a partial function definition.
Definition Decl.h:4841
Represents a C++11 pack expansion that produces a sequence of expressions.
Definition ExprCXX.h:4357
ParenExpr - This represents a parenthesized expression, e.g.
Definition Expr.h:2184
Sugar for parentheses used when specifying types.
Definition TypeBase.h:3302
Represents a parameter to a function.
Definition Decl.h:1789
PipeType - OpenCL20.
Definition TypeBase.h:8103
Represents a #pragma comment line.
Definition Decl.h:166
Represents a #pragma detect_mismatch line.
Definition Decl.h:200
[C99 6.4.2.2] - A predefined identifier such as func.
Definition Expr.h:2007
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
Definition Expr.h:6692
Expr *const * semantics_iterator
Definition Expr.h:6751
A (possibly-)qualified type.
Definition TypeBase.h:937
Represents a template name as written in source code.
Wrapper of type source information for a type with non-trivial direct qualifiers.
Definition TypeLoc.h:305
UnqualTypeLoc getUnqualifiedLoc() const
Definition TypeLoc.h:309
An rvalue reference type, per C++11 [dcl.ref].
Definition TypeBase.h:3633
Represents a struct/union/class.
Definition Decl.h:4309
Frontend produces RecoveryExprs on semantic errors that prevent creating other well-formed expression...
Definition Expr.h:7364
A class that does preorder or postorder depth-first traversal on the entire Clang AST and visits each...
bool TraverseStmt(Stmt *S, DataRecursionQueue *Queue=nullptr)
Recursively visit a statement or expression, by dispatching to Traverse*() based on the argument's dy...
bool TraverseTemplateArgument(const TemplateArgument &Arg)
Recursively visit a template argument and dispatch to the appropriate method for the argument type.
bool TraverseConceptRequirement(concepts::Requirement *R)
bool dataTraverseStmtPre(Stmt *S)
Invoked before visiting a statement or expression via data recursion.
bool TraverseObjCProtocolLoc(ObjCProtocolLoc ProtocolLoc)
Recursively visit an Objective-C protocol reference with location information.
bool VisitUnqualTypeLoc(UnqualTypeLoc TL)
bool TraverseConceptExprRequirement(concepts::ExprRequirement *R)
bool TraverseNestedNameSpecifier(NestedNameSpecifier NNS)
Recursively visit a C++ nested-name-specifier.
bool TraverseAST(ASTContext &AST)
Recursively visits an entire AST, starting from the TranslationUnitDecl.
bool shouldVisitTemplateInstantiations() const
Return whether this visitor should recurse into template instantiations.
bool TraverseTemplateArgumentLoc(const TemplateArgumentLoc &ArgLoc)
Recursively visit a template argument location and dispatch to the appropriate method for the argumen...
bool canIgnoreChildDeclWhileTraversingDeclContext(const Decl *Child)
bool dataTraverseStmtPost(Stmt *S)
Invoked after visiting a statement or expression via data recursion.
bool TraverseNestedNameSpecifierLoc(NestedNameSpecifierLoc NNS)
Recursively visit a C++ nested-name-specifier with location information.
bool TraverseTemplateName(TemplateName Template)
Recursively visit a template name and dispatch to the appropriate method.
Stmt::child_range getStmtChildren(Stmt *S)
bool shouldVisitImplicitCode() const
Return whether this visitor should recurse into implicit code, e.g., implicit constructors and destru...
bool TraverseConceptReference(ConceptReference *CR)
Recursively visit concept reference with location information.
bool TraverseTemplateArguments(ArrayRef< TemplateArgument > Args)
Recursively visit a set of template arguments.
bool TraverseTypeLoc(TypeLoc TL, bool TraverseQualifier=true)
Recursively visit a type with location, by dispatching to Traverse*TypeLoc() based on the argument ty...
bool WalkUpFromUnqualTypeLoc(UnqualTypeLoc TL)
bool dataTraverseNode(Stmt *S, DataRecursionQueue *Queue)
bool TraverseDecl(Decl *D)
Recursively visit a declaration, by dispatching to Traverse*Decl() based on the argument's dynamic ty...
bool TraverseTypeConstraint(const TypeConstraint *C)
bool WalkUpFromQualifiedTypeLoc(QualifiedTypeLoc TL)
bool TraverseLambdaCapture(LambdaExpr *LE, const LambdaCapture *C, Expr *Init)
Recursively visit a lambda capture.
bool VisitConceptReference(ConceptReference *CR)
bool shouldTraversePostOrder() const
Return whether this visitor should traverse post-order.
SmallVectorImpl< llvm::PointerIntPair< Stmt *, 1, bool > > DataRecursionQueue
A queue used for performing data recursion over statements.
bool shouldVisitLambdaBody() const
Return whether this visitor should recurse into lambda body.
bool TraverseSynOrSemInitListExpr(InitListExpr *S, DataRecursionQueue *Queue=nullptr)
Recursively visit the syntactic or semantic form of an initialization list.
bool TraverseAttr(Attr *At)
Recursively visit an attribute, by dispatching to Traverse*Attr() based on the argument's dynamic typ...
bool TraverseType(QualType T, bool TraverseQualifier=true)
Recursively visit a type, by dispatching to Traverse*Type() based on the argument's getTypeClass() pr...
bool TraverseConceptNestedRequirement(concepts::NestedRequirement *R)
bool VisitQualifiedTypeLoc(QualifiedTypeLoc TL)
bool shouldWalkTypesOfTypeLocs() const
Return whether this visitor should recurse into the types of TypeLocs.
bool TraverseDeclarationNameInfo(DeclarationNameInfo NameInfo)
Recursively visit a name with its location information.
bool TraverseCXXBaseSpecifier(const CXXBaseSpecifier &Base)
Recursively visit a base specifier.
Derived & getDerived()
Return a reference to the derived class.
bool TraverseConceptTypeRequirement(concepts::TypeRequirement *R)
bool TraverseConstructorInitializer(CXXCtorInitializer *Init)
Recursively visit a constructor initializer.
Represents the body of a requires-expression.
Definition DeclCXX.h:2098
C++2a [expr.prim.req]: A requires-expression provides a concise way to express requirements on templa...
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
Definition Stmt.h:3160
Represents a __leave statement.
Definition Stmt.h:3847
SYCLKernelCallStmt represents the transformation that is applied to the body of a function declared w...
Definition StmtSYCL.h:37
Scope - A scope is a transient data structure that is used while parsing the program.
Definition Scope.h:41
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
Definition Expr.h:4579
Represents an expression that computes the length of a parameter pack.
Definition ExprCXX.h:4435
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
Definition Expr.h:4953
Represents a C++11 static_assert declaration.
Definition DeclCXX.h:4130
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
Definition Expr.h:4531
Stmt - This represents one statement.
Definition Stmt.h:85
@ NoStmtClass
Definition Stmt.h:88
child_range children()
Definition Stmt.cpp:295
StmtClass getStmtClass() const
Definition Stmt.h:1472
llvm::iterator_range< child_iterator > child_range
Definition Stmt.h:1561
StringLiteral - This represents a string literal expression, e.g.
Definition Expr.h:1801
Represents a reference to a non-type template parameter that has been substituted with a template arg...
Definition ExprCXX.h:4658
Represents a reference to a non-type template parameter pack that has been substituted with a non-tem...
Definition ExprCXX.h:4748
Abstract type representing delayed type pack expansions.
Definition TypeLoc.h:995
SwitchStmt - This represents a 'switch' stmt.
Definition Stmt.h:2509
Location wrapper for a TemplateArgument.
const TemplateArgument & getArgument() const
TypeSourceInfo * getTypeSourceInfo() const
Expr * getSourceExpression() const
NestedNameSpecifierLoc getTemplateQualifierLoc() const
Represents a template argument.
Expr * getAsExpr() const
Retrieve the template argument as an expression.
QualType getAsType() const
Retrieve the type for a type template argument.
ArrayRef< TemplateArgument > pack_elements() const
Iterator range referencing all of the elements of a template argument pack.
@ Declaration
The template argument is a declaration that was provided for a pointer, reference,...
@ Template
The template argument is a template name that was provided for a template template parameter.
@ StructuralValue
The template argument is a non-type template argument that can't be represented by the special-case D...
@ Pack
The template argument is actually a parameter pack.
@ TemplateExpansion
The template argument is a pack expansion of a template name that was provided for a template templat...
@ NullPtr
The template argument is a null pointer or null pointer to member that was provided for a non-type te...
@ Type
The template argument is a type.
@ Null
Represents an empty template argument, e.g., one that has not been deduced.
@ Integral
The template argument is an integral value stored in an llvm::APSInt that was provided for an integra...
@ Expression
The template argument is an expression, and we've not resolved it to one of the other forms yet,...
ArgKind getKind() const
Return the kind of stored template argument.
TemplateName getAsTemplateOrTemplatePattern() const
Retrieve the template argument as a template name; if the argument is a pack expansion,...
Represents a C++ template name within the type system.
A template parameter object.
Stores a list of template parameters for a TemplateDecl and its derived classes.
NamedDecl ** iterator
Iterates through the template parameters in this list.
TemplateTemplateParmDecl - Declares a template template parameter, e.g., "T" in.
Declaration of a template type parameter.
A declaration that models statements at global scope.
Definition Decl.h:4597
The top declaration context.
Definition Decl.h:104
Represents the declaration of a typedef-name via a C++11 alias-declaration.
Definition Decl.h:3685
Declaration of an alias template.
Models the abbreviated syntax to constrain a template type parameter: template <convertible_to<string...
Definition ASTConcept.h:223
Base wrapper for a particular "section" of type source info.
Definition TypeLoc.h:59
UnqualTypeLoc getUnqualifiedLoc() const
Skips past any qualifiers, if this is qualified.
Definition TypeLoc.h:354
NestedNameSpecifierLoc getPrefix() const
If this type represents a qualified-id, this returns it's nested name specifier.
Definition TypeLoc.cpp:474
TypeLocClass getTypeLocClass() const
Definition TypeLoc.h:116
bool isNull() const
Definition TypeLoc.h:121
T getAsAdjusted() const
Convert to the specified TypeLoc type, returning a null TypeLoc if this TypeLoc is not of the desired...
Definition TypeLoc.h:2715
A container of type source information.
Definition TypeBase.h:8256
TypeLoc getTypeLoc() const
Return the TypeLoc wrapper for the type source info.
Definition TypeLoc.h:272
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
Definition ExprCXX.h:2890
The base class of the type hierarchy.
Definition TypeBase.h:1833
Represents the declaration of a typedef-name via the 'typedef' type specifier.
Definition Decl.h:3664
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
Definition Expr.h:2627
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Definition Expr.h:2246
An artificial decl, representing a global anonymous constant value which is uniquified by value withi...
Definition DeclCXX.h:4449
Wrapper of type source information for a type with no direct qualifiers.
Definition TypeLoc.h:279
A reference to a name which we were able to look up during parsing but could not resolve to a specifi...
Definition ExprCXX.h:3384
Represents a C++ member access expression for which lookup produced a set of overloaded functions.
Definition ExprCXX.h:4120
This node is generated when a using-declaration that was annotated with attribute((using_if_exists)) ...
Definition DeclCXX.h:4112
Represents a dependent using declaration which was marked with typename.
Definition DeclCXX.h:4031
Represents a dependent using declaration which was not marked with typename.
Definition DeclCXX.h:3934
A call to a literal operator (C++11 [over.literal]) written as a user-defined literal (C++11 [lit....
Definition ExprCXX.h:640
Represents a C++ using-declaration.
Definition DeclCXX.h:3585
Represents C++ using-directive.
Definition DeclCXX.h:3090
Represents a C++ using-enum-declaration.
Definition DeclCXX.h:3786
Represents a pack of using declarations that a single using-declarator pack-expanded into.
Definition DeclCXX.h:3867
Represents a shadow declaration implicitly introduced into a scope by a (resolved) using-declaration ...
Definition DeclCXX.h:3393
Represents a call to the builtin function __builtin_va_arg.
Definition Expr.h:4893
Represents a variable declaration or definition.
Definition Decl.h:925
Declaration of a variable template.
Represents a GCC generic vector type.
Definition TypeBase.h:4173
WhileStmt - This represents a 'while' stmt.
Definition Stmt.h:2697
A requires-expression requirement which queries the validity and properties of an expression ('simple...
const ReturnTypeRequirement & getReturnTypeRequirement() const
A requires-expression requirement which is satisfied when a general constraint expression is satisfie...
A static requirement that can be used in a requires-expression to check properties of types and expre...
RequirementKind getKind() const
A requires-expression requirement which queries the existence of a type name or type template special...
TypeSourceInfo * getType() const
LLVM_ATTRIBUTE_ALWAYS_INLINE LLVM_ATTRIBUTE_NODEBUG auto isSameMethod(FirstMethodPtrTy FirstMethodPtr, SecondMethodPtrTy SecondMethodPtr) -> bool
Returns true if and only if FirstMethodPtr and SecondMethodPtr are pointers to the same non-static me...
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Definition Address.h:330
TRY_TO(TraverseNestedNameSpecifier(Qualifier))
DEF_TRAVERSE_TYPELOC(ComplexType, { TRY_TO(TraverseType(TL.getTypePtr() ->getElementType()));}) DEF_TRAVERSE_TYPELOC(PointerType
@ TemplateName
The identifier is a template name. FIXME: Add an annotation for that.
Definition Parser.h:61
OpenACCComputeConstruct(OpenACCDirectiveKind K, SourceLocation Start, SourceLocation DirectiveLoc, SourceLocation End, ArrayRef< const OpenACCClause * > Clauses, Stmt *StructuredBlock)
@ Parameter
The parameter type of a method or function.
Definition TypeBase.h:908
const FunctionProtoType * T
@ Template
We are parsing a template declaration.
Definition Parser.h:81
bool declaresSameEntity(const Decl *D1, const Decl *D2)
Determine whether two declarations declare the same entity.
Definition DeclBase.h:1288
@ TSK_ExplicitInstantiationDefinition
This template specialization was instantiated from a template due to an explicit instantiation defini...
Definition Specifiers.h:206
@ TSK_ExplicitInstantiationDeclaration
This template specialization was instantiated from a template due to an explicit instantiation declar...
Definition Specifiers.h:202
@ TSK_ExplicitSpecialization
This template specialization was declared or defined by an explicit specialization (C++ [temp....
Definition Specifiers.h:198
@ TSK_ImplicitInstantiation
This template specialization was implicitly instantiated from a template.
Definition Specifiers.h:194
@ TSK_Undeclared
This template specialization was formed from a template-id but has not yet been declared,...
Definition Specifiers.h:191
U cast(CodeGen::Address addr)
Definition Address.h:327
@ Class
The "class" keyword introduces the elaborated-type-specifier.
Definition TypeBase.h:5874
DEF_TRAVERSE_TYPE(ComplexType, { TRY_TO(TraverseType(T->getElementType()));}) DEF_TRAVERSE_TYPE(PointerType
Represents an explicit template argument list in C++, e.g., the "<int>" in "sort<int>".
DeclarationNameInfo - A collector data type for bundling together a DeclarationName and the correspon...
DeclarationName getName() const
getName - Returns the embedded declaration name.
TypeSourceInfo * getNamedTypeInfo() const
getNamedTypeInfo - Returns the source type info associated to the name.
Data for list of allocators.