| 1 | //===- UsedDeclVisitor.h - ODR-used declarations 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 | //  This file defines UsedDeclVisitor, a CRTP class which visits all the | 
|---|
| 9 | //  declarations that are ODR-used by an expression or statement. | 
|---|
| 10 | // | 
|---|
| 11 | //===----------------------------------------------------------------------===// | 
|---|
| 12 |  | 
|---|
| 13 | #ifndef LLVM_CLANG_LIB_SEMA_USEDDECLVISITOR_H | 
|---|
| 14 | #define LLVM_CLANG_LIB_SEMA_USEDDECLVISITOR_H | 
|---|
| 15 |  | 
|---|
| 16 | #include "clang/AST/EvaluatedExprVisitor.h" | 
|---|
| 17 | #include "clang/Sema/SemaInternal.h" | 
|---|
| 18 |  | 
|---|
| 19 | namespace clang { | 
|---|
| 20 | template <class Derived> | 
|---|
| 21 | class UsedDeclVisitor : public EvaluatedExprVisitor<Derived> { | 
|---|
| 22 | protected: | 
|---|
| 23 | Sema &S; | 
|---|
| 24 |  | 
|---|
| 25 | public: | 
|---|
| 26 | typedef EvaluatedExprVisitor<Derived> Inherited; | 
|---|
| 27 |  | 
|---|
| 28 | UsedDeclVisitor(Sema &S) : Inherited(S.Context), S(S) {} | 
|---|
| 29 |  | 
|---|
| 30 | Derived &asImpl() { return *static_cast<Derived *>(this); } | 
|---|
| 31 |  | 
|---|
| 32 | void VisitDeclRefExpr(DeclRefExpr *E) { | 
|---|
| 33 | auto *D = E->getDecl(); | 
|---|
| 34 | if (isa<FunctionDecl>(Val: D) || isa<VarDecl>(Val: D)) { | 
|---|
| 35 | asImpl().visitUsedDecl(E->getLocation(), D); | 
|---|
| 36 | } | 
|---|
| 37 | } | 
|---|
| 38 |  | 
|---|
| 39 | void VisitMemberExpr(MemberExpr *E) { | 
|---|
| 40 | auto *D = E->getMemberDecl(); | 
|---|
| 41 | if (isa<FunctionDecl>(Val: D) || isa<VarDecl>(Val: D)) { | 
|---|
| 42 | asImpl().visitUsedDecl(E->getMemberLoc(), D); | 
|---|
| 43 | } | 
|---|
| 44 | asImpl().Visit(E->getBase()); | 
|---|
| 45 | } | 
|---|
| 46 |  | 
|---|
| 47 | void VisitCapturedStmt(CapturedStmt *Node) { | 
|---|
| 48 | asImpl().visitUsedDecl(Node->getBeginLoc(), Node->getCapturedDecl()); | 
|---|
| 49 | Inherited::VisitCapturedStmt(Node); | 
|---|
| 50 | } | 
|---|
| 51 |  | 
|---|
| 52 | void VisitCXXBindTemporaryExpr(CXXBindTemporaryExpr *E) { | 
|---|
| 53 | asImpl().visitUsedDecl( | 
|---|
| 54 | E->getBeginLoc(), | 
|---|
| 55 | const_cast<CXXDestructorDecl *>(E->getTemporary()->getDestructor())); | 
|---|
| 56 | asImpl().Visit(E->getSubExpr()); | 
|---|
| 57 | } | 
|---|
| 58 |  | 
|---|
| 59 | void VisitCXXNewExpr(CXXNewExpr *E) { | 
|---|
| 60 | if (E->getOperatorNew()) | 
|---|
| 61 | asImpl().visitUsedDecl(E->getBeginLoc(), E->getOperatorNew()); | 
|---|
| 62 | if (E->getOperatorDelete()) | 
|---|
| 63 | asImpl().visitUsedDecl(E->getBeginLoc(), E->getOperatorDelete()); | 
|---|
| 64 | Inherited::VisitCXXNewExpr(E); | 
|---|
| 65 | } | 
|---|
| 66 |  | 
|---|
| 67 | void VisitCXXDeleteExpr(CXXDeleteExpr *E) { | 
|---|
| 68 | if (E->getOperatorDelete()) | 
|---|
| 69 | asImpl().visitUsedDecl(E->getBeginLoc(), E->getOperatorDelete()); | 
|---|
| 70 | QualType DestroyedOrNull = E->getDestroyedType(); | 
|---|
| 71 | if (!DestroyedOrNull.isNull()) { | 
|---|
| 72 | QualType Destroyed = S.Context.getBaseElementType(QT: DestroyedOrNull); | 
|---|
| 73 | if (const RecordType *DestroyedRec = Destroyed->getAs<RecordType>()) { | 
|---|
| 74 | CXXRecordDecl *Record = cast<CXXRecordDecl>(Val: DestroyedRec->getDecl()); | 
|---|
| 75 | if (Record->getDefinition()) | 
|---|
| 76 | asImpl().visitUsedDecl(E->getBeginLoc(), S.LookupDestructor(Class: Record)); | 
|---|
| 77 | } | 
|---|
| 78 | } | 
|---|
| 79 |  | 
|---|
| 80 | Inherited::VisitCXXDeleteExpr(E); | 
|---|
| 81 | } | 
|---|
| 82 |  | 
|---|
| 83 | void VisitCXXConstructExpr(CXXConstructExpr *E) { | 
|---|
| 84 | asImpl().visitUsedDecl(E->getBeginLoc(), E->getConstructor()); | 
|---|
| 85 | CXXConstructorDecl *D = E->getConstructor(); | 
|---|
| 86 | for (const CXXCtorInitializer *Init : D->inits()) { | 
|---|
| 87 | if (Init->isInClassMemberInitializer()) | 
|---|
| 88 | asImpl().Visit(Init->getInit()); | 
|---|
| 89 | } | 
|---|
| 90 | Inherited::VisitCXXConstructExpr(E); | 
|---|
| 91 | } | 
|---|
| 92 |  | 
|---|
| 93 | void VisitCXXDefaultArgExpr(CXXDefaultArgExpr *E) { | 
|---|
| 94 | asImpl().Visit(E->getExpr()); | 
|---|
| 95 | Inherited::VisitCXXDefaultArgExpr(E); | 
|---|
| 96 | } | 
|---|
| 97 |  | 
|---|
| 98 | void VisitCXXDefaultInitExpr(CXXDefaultInitExpr *E) { | 
|---|
| 99 | asImpl().Visit(E->getExpr()); | 
|---|
| 100 | Inherited::VisitCXXDefaultInitExpr(E); | 
|---|
| 101 | } | 
|---|
| 102 |  | 
|---|
| 103 | void VisitInitListExpr(InitListExpr *ILE) { | 
|---|
| 104 | if (ILE->hasArrayFiller()) | 
|---|
| 105 | asImpl().Visit(ILE->getArrayFiller()); | 
|---|
| 106 | Inherited::VisitInitListExpr(ILE); | 
|---|
| 107 | } | 
|---|
| 108 |  | 
|---|
| 109 | void visitUsedDecl(SourceLocation Loc, Decl *D) { | 
|---|
| 110 | if (auto *CD = dyn_cast<CapturedDecl>(Val: D)) { | 
|---|
| 111 | if (auto *S = CD->getBody()) { | 
|---|
| 112 | asImpl().Visit(S); | 
|---|
| 113 | } | 
|---|
| 114 | } else if (auto *CD = dyn_cast<BlockDecl>(Val: D)) { | 
|---|
| 115 | if (auto *S = CD->getBody()) { | 
|---|
| 116 | asImpl().Visit(S); | 
|---|
| 117 | } | 
|---|
| 118 | } | 
|---|
| 119 | } | 
|---|
| 120 | }; | 
|---|
| 121 | } // end namespace clang | 
|---|
| 122 |  | 
|---|
| 123 | #endif // LLVM_CLANG_LIB_SEMA_USEDDECLVISITOR_H | 
|---|
| 124 |  | 
|---|