annotate clang-tools-extra/clangd/FindTarget.cpp @ 236:c4bab56944e8 llvm-original

LLVM 16
author kono
date Wed, 09 Nov 2022 17:45:10 +0900
parents 5f17cb93ff66
children 1f2b6ac9f198
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===--- FindTarget.cpp - What does an AST node refer to? -----------------===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8
anatofuz
parents:
diff changeset
9 #include "FindTarget.h"
anatofuz
parents:
diff changeset
10 #include "AST.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
11 #include "HeuristicResolver.h"
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
12 #include "support/Logger.h"
150
anatofuz
parents:
diff changeset
13 #include "clang/AST/ASTTypeTraits.h"
anatofuz
parents:
diff changeset
14 #include "clang/AST/Decl.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
15 #include "clang/AST/DeclBase.h"
150
anatofuz
parents:
diff changeset
16 #include "clang/AST/DeclCXX.h"
anatofuz
parents:
diff changeset
17 #include "clang/AST/DeclTemplate.h"
anatofuz
parents:
diff changeset
18 #include "clang/AST/DeclVisitor.h"
anatofuz
parents:
diff changeset
19 #include "clang/AST/DeclarationName.h"
anatofuz
parents:
diff changeset
20 #include "clang/AST/Expr.h"
anatofuz
parents:
diff changeset
21 #include "clang/AST/ExprCXX.h"
anatofuz
parents:
diff changeset
22 #include "clang/AST/ExprConcepts.h"
anatofuz
parents:
diff changeset
23 #include "clang/AST/ExprObjC.h"
anatofuz
parents:
diff changeset
24 #include "clang/AST/NestedNameSpecifier.h"
anatofuz
parents:
diff changeset
25 #include "clang/AST/PrettyPrinter.h"
anatofuz
parents:
diff changeset
26 #include "clang/AST/RecursiveASTVisitor.h"
anatofuz
parents:
diff changeset
27 #include "clang/AST/StmtVisitor.h"
anatofuz
parents:
diff changeset
28 #include "clang/AST/TemplateBase.h"
anatofuz
parents:
diff changeset
29 #include "clang/AST/Type.h"
anatofuz
parents:
diff changeset
30 #include "clang/AST/TypeLoc.h"
anatofuz
parents:
diff changeset
31 #include "clang/AST/TypeLocVisitor.h"
anatofuz
parents:
diff changeset
32 #include "clang/AST/TypeVisitor.h"
anatofuz
parents:
diff changeset
33 #include "clang/Basic/LangOptions.h"
anatofuz
parents:
diff changeset
34 #include "clang/Basic/SourceLocation.h"
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
35 #include "clang/Basic/SourceManager.h"
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
36 #include "clang/Basic/Specifiers.h"
150
anatofuz
parents:
diff changeset
37 #include "llvm/ADT/STLExtras.h"
anatofuz
parents:
diff changeset
38 #include "llvm/ADT/SmallVector.h"
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
39 #include "llvm/ADT/StringExtras.h"
150
anatofuz
parents:
diff changeset
40 #include "llvm/Support/Casting.h"
anatofuz
parents:
diff changeset
41 #include "llvm/Support/Compiler.h"
anatofuz
parents:
diff changeset
42 #include "llvm/Support/raw_ostream.h"
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
43 #include <iterator>
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
44 #include <string>
150
anatofuz
parents:
diff changeset
45 #include <utility>
anatofuz
parents:
diff changeset
46 #include <vector>
anatofuz
parents:
diff changeset
47
anatofuz
parents:
diff changeset
48 namespace clang {
anatofuz
parents:
diff changeset
49 namespace clangd {
anatofuz
parents:
diff changeset
50 namespace {
anatofuz
parents:
diff changeset
51
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
52 LLVM_ATTRIBUTE_UNUSED std::string nodeToString(const DynTypedNode &N) {
150
anatofuz
parents:
diff changeset
53 std::string S = std::string(N.getNodeKind().asStringRef());
anatofuz
parents:
diff changeset
54 {
anatofuz
parents:
diff changeset
55 llvm::raw_string_ostream OS(S);
anatofuz
parents:
diff changeset
56 OS << ": ";
anatofuz
parents:
diff changeset
57 N.print(OS, PrintingPolicy(LangOptions()));
anatofuz
parents:
diff changeset
58 }
anatofuz
parents:
diff changeset
59 std::replace(S.begin(), S.end(), '\n', ' ');
anatofuz
parents:
diff changeset
60 return S;
anatofuz
parents:
diff changeset
61 }
anatofuz
parents:
diff changeset
62
anatofuz
parents:
diff changeset
63 const NamedDecl *getTemplatePattern(const NamedDecl *D) {
anatofuz
parents:
diff changeset
64 if (const CXXRecordDecl *CRD = dyn_cast<CXXRecordDecl>(D)) {
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
65 if (const auto *Result = CRD->getTemplateInstantiationPattern())
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
66 return Result;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
67 // getTemplateInstantiationPattern returns null if the Specialization is
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
68 // incomplete (e.g. the type didn't need to be complete), fall back to the
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
69 // primary template.
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
70 if (CRD->getTemplateSpecializationKind() == TSK_Undeclared)
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
71 if (const auto *Spec = dyn_cast<ClassTemplateSpecializationDecl>(CRD))
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
72 return Spec->getSpecializedTemplate()->getTemplatedDecl();
150
anatofuz
parents:
diff changeset
73 } else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
anatofuz
parents:
diff changeset
74 return FD->getTemplateInstantiationPattern();
anatofuz
parents:
diff changeset
75 } else if (auto *VD = dyn_cast<VarDecl>(D)) {
anatofuz
parents:
diff changeset
76 // Hmm: getTIP returns its arg if it's not an instantiation?!
anatofuz
parents:
diff changeset
77 VarDecl *T = VD->getTemplateInstantiationPattern();
anatofuz
parents:
diff changeset
78 return (T == D) ? nullptr : T;
anatofuz
parents:
diff changeset
79 } else if (const auto *ED = dyn_cast<EnumDecl>(D)) {
anatofuz
parents:
diff changeset
80 return ED->getInstantiatedFromMemberEnum();
anatofuz
parents:
diff changeset
81 } else if (isa<FieldDecl>(D) || isa<TypedefNameDecl>(D)) {
anatofuz
parents:
diff changeset
82 if (const auto *Parent = llvm::dyn_cast<NamedDecl>(D->getDeclContext()))
anatofuz
parents:
diff changeset
83 if (const DeclContext *ParentPat =
anatofuz
parents:
diff changeset
84 dyn_cast_or_null<DeclContext>(getTemplatePattern(Parent)))
anatofuz
parents:
diff changeset
85 for (const NamedDecl *BaseND : ParentPat->lookup(D->getDeclName()))
anatofuz
parents:
diff changeset
86 if (!BaseND->isImplicit() && BaseND->getKind() == D->getKind())
anatofuz
parents:
diff changeset
87 return BaseND;
anatofuz
parents:
diff changeset
88 } else if (const auto *ECD = dyn_cast<EnumConstantDecl>(D)) {
anatofuz
parents:
diff changeset
89 if (const auto *ED = dyn_cast<EnumDecl>(ECD->getDeclContext())) {
anatofuz
parents:
diff changeset
90 if (const EnumDecl *Pattern = ED->getInstantiatedFromMemberEnum()) {
anatofuz
parents:
diff changeset
91 for (const NamedDecl *BaseECD : Pattern->lookup(ECD->getDeclName()))
anatofuz
parents:
diff changeset
92 return BaseECD;
anatofuz
parents:
diff changeset
93 }
anatofuz
parents:
diff changeset
94 }
anatofuz
parents:
diff changeset
95 }
anatofuz
parents:
diff changeset
96 return nullptr;
anatofuz
parents:
diff changeset
97 }
anatofuz
parents:
diff changeset
98
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
99 // Returns true if the `TypedefNameDecl` should not be reported.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
100 bool shouldSkipTypedef(const TypedefNameDecl *TD) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
101 // These should be treated as keywords rather than decls - the typedef is an
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
102 // odd implementation detail.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
103 if (TD == TD->getASTContext().getObjCInstanceTypeDecl() ||
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
104 TD == TD->getASTContext().getObjCIdDecl())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
105 return true;
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
106 return false;
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
107 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
108
150
anatofuz
parents:
diff changeset
109 // TargetFinder locates the entities that an AST node refers to.
anatofuz
parents:
diff changeset
110 //
anatofuz
parents:
diff changeset
111 // Typically this is (possibly) one declaration and (possibly) one type, but
anatofuz
parents:
diff changeset
112 // may be more:
anatofuz
parents:
diff changeset
113 // - for ambiguous nodes like OverloadExpr
anatofuz
parents:
diff changeset
114 // - if we want to include e.g. both typedefs and the underlying type
anatofuz
parents:
diff changeset
115 //
anatofuz
parents:
diff changeset
116 // This is organized as a set of mutually recursive helpers for particular node
anatofuz
parents:
diff changeset
117 // types, but for most nodes this is a short walk rather than a deep traversal.
anatofuz
parents:
diff changeset
118 //
anatofuz
parents:
diff changeset
119 // It's tempting to do e.g. typedef resolution as a second normalization step,
anatofuz
parents:
diff changeset
120 // after finding the 'primary' decl etc. But we do this monolithically instead
anatofuz
parents:
diff changeset
121 // because:
anatofuz
parents:
diff changeset
122 // - normalization may require these traversals again (e.g. unwrapping a
anatofuz
parents:
diff changeset
123 // typedef reveals a decltype which must be traversed)
anatofuz
parents:
diff changeset
124 // - it doesn't simplify that much, e.g. the first stage must still be able
anatofuz
parents:
diff changeset
125 // to yield multiple decls to handle OverloadExpr
anatofuz
parents:
diff changeset
126 // - there are cases where it's required for correctness. e.g:
anatofuz
parents:
diff changeset
127 // template<class X> using pvec = vector<x*>; pvec<int> x;
anatofuz
parents:
diff changeset
128 // There's no Decl `pvec<int>`, we must choose `pvec<X>` or `vector<int*>`
anatofuz
parents:
diff changeset
129 // and both are lossy. We must know upfront what the caller ultimately wants.
anatofuz
parents:
diff changeset
130 struct TargetFinder {
anatofuz
parents:
diff changeset
131 using RelSet = DeclRelationSet;
anatofuz
parents:
diff changeset
132 using Rel = DeclRelation;
anatofuz
parents:
diff changeset
133
anatofuz
parents:
diff changeset
134 private:
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
135 const HeuristicResolver *Resolver;
150
anatofuz
parents:
diff changeset
136 llvm::SmallDenseMap<const NamedDecl *,
anatofuz
parents:
diff changeset
137 std::pair<RelSet, /*InsertionOrder*/ size_t>>
anatofuz
parents:
diff changeset
138 Decls;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
139 llvm::SmallDenseMap<const Decl *, RelSet> Seen;
150
anatofuz
parents:
diff changeset
140 RelSet Flags;
anatofuz
parents:
diff changeset
141
anatofuz
parents:
diff changeset
142 template <typename T> void debug(T &Node, RelSet Flags) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
143 dlog("visit [{0}] {1}", Flags, nodeToString(DynTypedNode::create(Node)));
150
anatofuz
parents:
diff changeset
144 }
anatofuz
parents:
diff changeset
145
anatofuz
parents:
diff changeset
146 void report(const NamedDecl *D, RelSet Flags) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
147 dlog("--> [{0}] {1}", Flags, nodeToString(DynTypedNode::create(*D)));
150
anatofuz
parents:
diff changeset
148 auto It = Decls.try_emplace(D, std::make_pair(Flags, Decls.size()));
anatofuz
parents:
diff changeset
149 // If already exists, update the flags.
anatofuz
parents:
diff changeset
150 if (!It.second)
anatofuz
parents:
diff changeset
151 It.first->second.first |= Flags;
anatofuz
parents:
diff changeset
152 }
anatofuz
parents:
diff changeset
153
anatofuz
parents:
diff changeset
154 public:
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
155 TargetFinder(const HeuristicResolver *Resolver) : Resolver(Resolver) {}
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
156
150
anatofuz
parents:
diff changeset
157 llvm::SmallVector<std::pair<const NamedDecl *, RelSet>, 1> takeDecls() const {
anatofuz
parents:
diff changeset
158 using ValTy = std::pair<const NamedDecl *, RelSet>;
anatofuz
parents:
diff changeset
159 llvm::SmallVector<ValTy, 1> Result;
anatofuz
parents:
diff changeset
160 Result.resize(Decls.size());
anatofuz
parents:
diff changeset
161 for (const auto &Elem : Decls)
anatofuz
parents:
diff changeset
162 Result[Elem.second.second] = {Elem.first, Elem.second.first};
anatofuz
parents:
diff changeset
163 return Result;
anatofuz
parents:
diff changeset
164 }
anatofuz
parents:
diff changeset
165
anatofuz
parents:
diff changeset
166 void add(const Decl *Dcl, RelSet Flags) {
anatofuz
parents:
diff changeset
167 const NamedDecl *D = llvm::dyn_cast_or_null<NamedDecl>(Dcl);
anatofuz
parents:
diff changeset
168 if (!D)
anatofuz
parents:
diff changeset
169 return;
anatofuz
parents:
diff changeset
170 debug(*D, Flags);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
171
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
172 // Avoid recursion (which can arise in the presence of heuristic
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
173 // resolution of dependent names) by exiting early if we have
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
174 // already seen this decl with all flags in Flags.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
175 auto Res = Seen.try_emplace(D);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
176 if (!Res.second && Res.first->second.contains(Flags))
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
177 return;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
178 Res.first->second |= Flags;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
179
150
anatofuz
parents:
diff changeset
180 if (const UsingDirectiveDecl *UDD = llvm::dyn_cast<UsingDirectiveDecl>(D))
anatofuz
parents:
diff changeset
181 D = UDD->getNominatedNamespaceAsWritten();
anatofuz
parents:
diff changeset
182
anatofuz
parents:
diff changeset
183 if (const TypedefNameDecl *TND = dyn_cast<TypedefNameDecl>(D)) {
anatofuz
parents:
diff changeset
184 add(TND->getUnderlyingType(), Flags | Rel::Underlying);
anatofuz
parents:
diff changeset
185 Flags |= Rel::Alias; // continue with the alias.
anatofuz
parents:
diff changeset
186 } else if (const UsingDecl *UD = dyn_cast<UsingDecl>(D)) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
187 // no Underlying as this is a non-renaming alias.
150
anatofuz
parents:
diff changeset
188 for (const UsingShadowDecl *S : UD->shadows())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
189 add(S->getUnderlyingDecl(), Flags);
150
anatofuz
parents:
diff changeset
190 Flags |= Rel::Alias; // continue with the alias.
223
5f17cb93ff66 LLVM13 (2021/7/18)
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 221
diff changeset
191 } else if (const UsingEnumDecl *UED = dyn_cast<UsingEnumDecl>(D)) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
192 // UsingEnumDecl is not an alias at all, just a reference.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
193 D = UED->getEnumDecl();
150
anatofuz
parents:
diff changeset
194 } else if (const auto *NAD = dyn_cast<NamespaceAliasDecl>(D)) {
anatofuz
parents:
diff changeset
195 add(NAD->getUnderlyingDecl(), Flags | Rel::Underlying);
anatofuz
parents:
diff changeset
196 Flags |= Rel::Alias; // continue with the alias
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
197 } else if (const UnresolvedUsingValueDecl *UUVD =
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
198 dyn_cast<UnresolvedUsingValueDecl>(D)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
199 if (Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
200 for (const NamedDecl *Target : Resolver->resolveUsingValueDecl(UUVD)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
201 add(Target, Flags); // no Underlying as this is a non-renaming alias
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
202 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
203 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
204 Flags |= Rel::Alias; // continue with the alias
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
205 } else if (isa<UnresolvedUsingTypenameDecl>(D)) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
206 // FIXME: improve common dependent scope using name lookup in primary
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
207 // templates.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
208 Flags |= Rel::Alias;
150
anatofuz
parents:
diff changeset
209 } else if (const UsingShadowDecl *USD = dyn_cast<UsingShadowDecl>(D)) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
210 // Include the introducing UsingDecl, but don't traverse it. This may end
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
211 // up including *all* shadows, which we don't want.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
212 // Don't apply this logic to UsingEnumDecl, which can't easily be
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
213 // conflated with the aliases it introduces.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
214 if (llvm::isa<UsingDecl>(USD->getIntroducer()))
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
215 report(USD->getIntroducer(), Flags | Rel::Alias);
150
anatofuz
parents:
diff changeset
216 // Shadow decls are synthetic and not themselves interesting.
anatofuz
parents:
diff changeset
217 // Record the underlying decl instead, if allowed.
anatofuz
parents:
diff changeset
218 D = USD->getTargetDecl();
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
219 } else if (const auto *DG = dyn_cast<CXXDeductionGuideDecl>(D)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
220 D = DG->getDeducedTemplate();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
221 } else if (const ObjCImplementationDecl *IID =
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
222 dyn_cast<ObjCImplementationDecl>(D)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
223 // Treat ObjC{Interface,Implementation}Decl as if they were a decl/def
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
224 // pair as long as the interface isn't implicit.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
225 if (const auto *CID = IID->getClassInterface())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
226 if (const auto *DD = CID->getDefinition())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
227 if (!DD->isImplicitInterfaceDecl())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
228 D = DD;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
229 } else if (const ObjCCategoryImplDecl *CID =
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
230 dyn_cast<ObjCCategoryImplDecl>(D)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
231 // Treat ObjC{Category,CategoryImpl}Decl as if they were a decl/def pair.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
232 D = CID->getCategoryDecl();
150
anatofuz
parents:
diff changeset
233 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
234 if (!D)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
235 return;
150
anatofuz
parents:
diff changeset
236
anatofuz
parents:
diff changeset
237 if (const Decl *Pat = getTemplatePattern(D)) {
anatofuz
parents:
diff changeset
238 assert(Pat != D);
anatofuz
parents:
diff changeset
239 add(Pat, Flags | Rel::TemplatePattern);
anatofuz
parents:
diff changeset
240 // Now continue with the instantiation.
anatofuz
parents:
diff changeset
241 Flags |= Rel::TemplateInstantiation;
anatofuz
parents:
diff changeset
242 }
anatofuz
parents:
diff changeset
243
anatofuz
parents:
diff changeset
244 report(D, Flags);
anatofuz
parents:
diff changeset
245 }
anatofuz
parents:
diff changeset
246
anatofuz
parents:
diff changeset
247 void add(const Stmt *S, RelSet Flags) {
anatofuz
parents:
diff changeset
248 if (!S)
anatofuz
parents:
diff changeset
249 return;
anatofuz
parents:
diff changeset
250 debug(*S, Flags);
anatofuz
parents:
diff changeset
251 struct Visitor : public ConstStmtVisitor<Visitor> {
anatofuz
parents:
diff changeset
252 TargetFinder &Outer;
anatofuz
parents:
diff changeset
253 RelSet Flags;
anatofuz
parents:
diff changeset
254 Visitor(TargetFinder &Outer, RelSet Flags) : Outer(Outer), Flags(Flags) {}
anatofuz
parents:
diff changeset
255
anatofuz
parents:
diff changeset
256 void VisitCallExpr(const CallExpr *CE) {
anatofuz
parents:
diff changeset
257 Outer.add(CE->getCalleeDecl(), Flags);
anatofuz
parents:
diff changeset
258 }
anatofuz
parents:
diff changeset
259 void VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E) {
anatofuz
parents:
diff changeset
260 Outer.add(E->getNamedConcept(), Flags);
anatofuz
parents:
diff changeset
261 }
anatofuz
parents:
diff changeset
262 void VisitDeclRefExpr(const DeclRefExpr *DRE) {
anatofuz
parents:
diff changeset
263 const Decl *D = DRE->getDecl();
anatofuz
parents:
diff changeset
264 // UsingShadowDecl allows us to record the UsingDecl.
anatofuz
parents:
diff changeset
265 // getFoundDecl() returns the wrong thing in other cases (templates).
anatofuz
parents:
diff changeset
266 if (auto *USD = llvm::dyn_cast<UsingShadowDecl>(DRE->getFoundDecl()))
anatofuz
parents:
diff changeset
267 D = USD;
anatofuz
parents:
diff changeset
268 Outer.add(D, Flags);
anatofuz
parents:
diff changeset
269 }
anatofuz
parents:
diff changeset
270 void VisitMemberExpr(const MemberExpr *ME) {
anatofuz
parents:
diff changeset
271 const Decl *D = ME->getMemberDecl();
anatofuz
parents:
diff changeset
272 if (auto *USD =
anatofuz
parents:
diff changeset
273 llvm::dyn_cast<UsingShadowDecl>(ME->getFoundDecl().getDecl()))
anatofuz
parents:
diff changeset
274 D = USD;
anatofuz
parents:
diff changeset
275 Outer.add(D, Flags);
anatofuz
parents:
diff changeset
276 }
anatofuz
parents:
diff changeset
277 void VisitOverloadExpr(const OverloadExpr *OE) {
anatofuz
parents:
diff changeset
278 for (auto *D : OE->decls())
anatofuz
parents:
diff changeset
279 Outer.add(D, Flags);
anatofuz
parents:
diff changeset
280 }
anatofuz
parents:
diff changeset
281 void VisitSizeOfPackExpr(const SizeOfPackExpr *SE) {
anatofuz
parents:
diff changeset
282 Outer.add(SE->getPack(), Flags);
anatofuz
parents:
diff changeset
283 }
anatofuz
parents:
diff changeset
284 void VisitCXXConstructExpr(const CXXConstructExpr *CCE) {
anatofuz
parents:
diff changeset
285 Outer.add(CCE->getConstructor(), Flags);
anatofuz
parents:
diff changeset
286 }
anatofuz
parents:
diff changeset
287 void VisitDesignatedInitExpr(const DesignatedInitExpr *DIE) {
anatofuz
parents:
diff changeset
288 for (const DesignatedInitExpr::Designator &D :
anatofuz
parents:
diff changeset
289 llvm::reverse(DIE->designators()))
anatofuz
parents:
diff changeset
290 if (D.isFieldDesignator()) {
anatofuz
parents:
diff changeset
291 Outer.add(D.getField(), Flags);
anatofuz
parents:
diff changeset
292 // We don't know which designator was intended, we assume the outer.
anatofuz
parents:
diff changeset
293 break;
anatofuz
parents:
diff changeset
294 }
anatofuz
parents:
diff changeset
295 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
296 void VisitGotoStmt(const GotoStmt *Goto) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
297 if (auto *LabelDecl = Goto->getLabel())
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
298 Outer.add(LabelDecl, Flags);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
299 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
300 void VisitLabelStmt(const LabelStmt *Label) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
301 if (auto *LabelDecl = Label->getDecl())
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
302 Outer.add(LabelDecl, Flags);
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
303 }
150
anatofuz
parents:
diff changeset
304 void
anatofuz
parents:
diff changeset
305 VisitCXXDependentScopeMemberExpr(const CXXDependentScopeMemberExpr *E) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
306 if (Outer.Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
307 for (const NamedDecl *D : Outer.Resolver->resolveMemberExpr(E)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
308 Outer.add(D, Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
309 }
150
anatofuz
parents:
diff changeset
310 }
anatofuz
parents:
diff changeset
311 }
anatofuz
parents:
diff changeset
312 void VisitDependentScopeDeclRefExpr(const DependentScopeDeclRefExpr *E) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
313 if (Outer.Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
314 for (const NamedDecl *D : Outer.Resolver->resolveDeclRefExpr(E)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
315 Outer.add(D, Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
316 }
150
anatofuz
parents:
diff changeset
317 }
anatofuz
parents:
diff changeset
318 }
anatofuz
parents:
diff changeset
319 void VisitObjCIvarRefExpr(const ObjCIvarRefExpr *OIRE) {
anatofuz
parents:
diff changeset
320 Outer.add(OIRE->getDecl(), Flags);
anatofuz
parents:
diff changeset
321 }
anatofuz
parents:
diff changeset
322 void VisitObjCMessageExpr(const ObjCMessageExpr *OME) {
anatofuz
parents:
diff changeset
323 Outer.add(OME->getMethodDecl(), Flags);
anatofuz
parents:
diff changeset
324 }
anatofuz
parents:
diff changeset
325 void VisitObjCPropertyRefExpr(const ObjCPropertyRefExpr *OPRE) {
anatofuz
parents:
diff changeset
326 if (OPRE->isExplicitProperty())
anatofuz
parents:
diff changeset
327 Outer.add(OPRE->getExplicitProperty(), Flags);
anatofuz
parents:
diff changeset
328 else {
anatofuz
parents:
diff changeset
329 if (OPRE->isMessagingGetter())
anatofuz
parents:
diff changeset
330 Outer.add(OPRE->getImplicitPropertyGetter(), Flags);
anatofuz
parents:
diff changeset
331 if (OPRE->isMessagingSetter())
anatofuz
parents:
diff changeset
332 Outer.add(OPRE->getImplicitPropertySetter(), Flags);
anatofuz
parents:
diff changeset
333 }
anatofuz
parents:
diff changeset
334 }
anatofuz
parents:
diff changeset
335 void VisitObjCProtocolExpr(const ObjCProtocolExpr *OPE) {
anatofuz
parents:
diff changeset
336 Outer.add(OPE->getProtocol(), Flags);
anatofuz
parents:
diff changeset
337 }
anatofuz
parents:
diff changeset
338 void VisitOpaqueValueExpr(const OpaqueValueExpr *OVE) {
anatofuz
parents:
diff changeset
339 Outer.add(OVE->getSourceExpr(), Flags);
anatofuz
parents:
diff changeset
340 }
anatofuz
parents:
diff changeset
341 void VisitPseudoObjectExpr(const PseudoObjectExpr *POE) {
anatofuz
parents:
diff changeset
342 Outer.add(POE->getSyntacticForm(), Flags);
anatofuz
parents:
diff changeset
343 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
344 void VisitCXXNewExpr(const CXXNewExpr *CNE) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
345 Outer.add(CNE->getOperatorNew(), Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
346 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
347 void VisitCXXDeleteExpr(const CXXDeleteExpr *CDE) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
348 Outer.add(CDE->getOperatorDelete(), Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
349 }
150
anatofuz
parents:
diff changeset
350 };
anatofuz
parents:
diff changeset
351 Visitor(*this, Flags).Visit(S);
anatofuz
parents:
diff changeset
352 }
anatofuz
parents:
diff changeset
353
anatofuz
parents:
diff changeset
354 void add(QualType T, RelSet Flags) {
anatofuz
parents:
diff changeset
355 if (T.isNull())
anatofuz
parents:
diff changeset
356 return;
anatofuz
parents:
diff changeset
357 debug(T, Flags);
anatofuz
parents:
diff changeset
358 struct Visitor : public TypeVisitor<Visitor> {
anatofuz
parents:
diff changeset
359 TargetFinder &Outer;
anatofuz
parents:
diff changeset
360 RelSet Flags;
anatofuz
parents:
diff changeset
361 Visitor(TargetFinder &Outer, RelSet Flags) : Outer(Outer), Flags(Flags) {}
anatofuz
parents:
diff changeset
362
anatofuz
parents:
diff changeset
363 void VisitTagType(const TagType *TT) {
anatofuz
parents:
diff changeset
364 Outer.add(TT->getAsTagDecl(), Flags);
anatofuz
parents:
diff changeset
365 }
anatofuz
parents:
diff changeset
366
anatofuz
parents:
diff changeset
367 void VisitElaboratedType(const ElaboratedType *ET) {
anatofuz
parents:
diff changeset
368 Outer.add(ET->desugar(), Flags);
anatofuz
parents:
diff changeset
369 }
anatofuz
parents:
diff changeset
370
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
371 void VisitUsingType(const UsingType *ET) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
372 Outer.add(ET->getFoundDecl(), Flags);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
373 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
374
150
anatofuz
parents:
diff changeset
375 void VisitInjectedClassNameType(const InjectedClassNameType *ICNT) {
anatofuz
parents:
diff changeset
376 Outer.add(ICNT->getDecl(), Flags);
anatofuz
parents:
diff changeset
377 }
anatofuz
parents:
diff changeset
378
anatofuz
parents:
diff changeset
379 void VisitDecltypeType(const DecltypeType *DTT) {
anatofuz
parents:
diff changeset
380 Outer.add(DTT->getUnderlyingType(), Flags | Rel::Underlying);
anatofuz
parents:
diff changeset
381 }
anatofuz
parents:
diff changeset
382 void VisitDeducedType(const DeducedType *DT) {
anatofuz
parents:
diff changeset
383 // FIXME: In practice this doesn't work: the AutoType you find inside
anatofuz
parents:
diff changeset
384 // TypeLoc never has a deduced type. https://llvm.org/PR42914
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
385 Outer.add(DT->getDeducedType(), Flags);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
386 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
387 void VisitUnresolvedUsingType(const UnresolvedUsingType *UUT) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
388 Outer.add(UUT->getDecl(), Flags);
150
anatofuz
parents:
diff changeset
389 }
anatofuz
parents:
diff changeset
390 void VisitDeducedTemplateSpecializationType(
anatofuz
parents:
diff changeset
391 const DeducedTemplateSpecializationType *DTST) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
392 if (const auto *USD = DTST->getTemplateName().getAsUsingShadowDecl())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
393 Outer.add(USD, Flags);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
394
150
anatofuz
parents:
diff changeset
395 // FIXME: This is a workaround for https://llvm.org/PR42914,
anatofuz
parents:
diff changeset
396 // which is causing DTST->getDeducedType() to be empty. We
anatofuz
parents:
diff changeset
397 // fall back to the template pattern and miss the instantiation
anatofuz
parents:
diff changeset
398 // even when it's known in principle. Once that bug is fixed,
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
399 // the following code can be removed (the existing handling in
150
anatofuz
parents:
diff changeset
400 // VisitDeducedType() is sufficient).
anatofuz
parents:
diff changeset
401 if (auto *TD = DTST->getTemplateName().getAsTemplateDecl())
anatofuz
parents:
diff changeset
402 Outer.add(TD->getTemplatedDecl(), Flags | Rel::TemplatePattern);
anatofuz
parents:
diff changeset
403 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
404 void VisitDependentNameType(const DependentNameType *DNT) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
405 if (Outer.Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
406 for (const NamedDecl *ND :
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
407 Outer.Resolver->resolveDependentNameType(DNT)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
408 Outer.add(ND, Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
409 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
410 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
411 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
412 void VisitDependentTemplateSpecializationType(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
413 const DependentTemplateSpecializationType *DTST) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
414 if (Outer.Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
415 for (const NamedDecl *ND :
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
416 Outer.Resolver->resolveTemplateSpecializationType(DTST)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
417 Outer.add(ND, Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
418 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
419 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
420 }
150
anatofuz
parents:
diff changeset
421 void VisitTypedefType(const TypedefType *TT) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
422 if (shouldSkipTypedef(TT->getDecl()))
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
423 return;
150
anatofuz
parents:
diff changeset
424 Outer.add(TT->getDecl(), Flags);
anatofuz
parents:
diff changeset
425 }
anatofuz
parents:
diff changeset
426 void
anatofuz
parents:
diff changeset
427 VisitTemplateSpecializationType(const TemplateSpecializationType *TST) {
anatofuz
parents:
diff changeset
428 // Have to handle these case-by-case.
anatofuz
parents:
diff changeset
429
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
430 if (const auto *UTN = TST->getTemplateName().getAsUsingShadowDecl())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
431 Outer.add(UTN, Flags);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
432
150
anatofuz
parents:
diff changeset
433 // templated type aliases: there's no specialized/instantiated using
anatofuz
parents:
diff changeset
434 // decl to point to. So try to find a decl for the underlying type
anatofuz
parents:
diff changeset
435 // (after substitution), and failing that point to the (templated) using
anatofuz
parents:
diff changeset
436 // decl.
anatofuz
parents:
diff changeset
437 if (TST->isTypeAlias()) {
anatofuz
parents:
diff changeset
438 Outer.add(TST->getAliasedType(), Flags | Rel::Underlying);
anatofuz
parents:
diff changeset
439 // Don't *traverse* the alias, which would result in traversing the
anatofuz
parents:
diff changeset
440 // template of the underlying type.
anatofuz
parents:
diff changeset
441 Outer.report(
anatofuz
parents:
diff changeset
442 TST->getTemplateName().getAsTemplateDecl()->getTemplatedDecl(),
anatofuz
parents:
diff changeset
443 Flags | Rel::Alias | Rel::TemplatePattern);
anatofuz
parents:
diff changeset
444 }
anatofuz
parents:
diff changeset
445 // specializations of template template parameters aren't instantiated
anatofuz
parents:
diff changeset
446 // into decls, so they must refer to the parameter itself.
anatofuz
parents:
diff changeset
447 else if (const auto *Parm =
anatofuz
parents:
diff changeset
448 llvm::dyn_cast_or_null<TemplateTemplateParmDecl>(
anatofuz
parents:
diff changeset
449 TST->getTemplateName().getAsTemplateDecl()))
anatofuz
parents:
diff changeset
450 Outer.add(Parm, Flags);
anatofuz
parents:
diff changeset
451 // class template specializations have a (specialized) CXXRecordDecl.
anatofuz
parents:
diff changeset
452 else if (const CXXRecordDecl *RD = TST->getAsCXXRecordDecl())
anatofuz
parents:
diff changeset
453 Outer.add(RD, Flags); // add(Decl) will despecialize if needed.
anatofuz
parents:
diff changeset
454 else {
anatofuz
parents:
diff changeset
455 // fallback: the (un-specialized) declaration from primary template.
anatofuz
parents:
diff changeset
456 if (auto *TD = TST->getTemplateName().getAsTemplateDecl())
anatofuz
parents:
diff changeset
457 Outer.add(TD->getTemplatedDecl(), Flags | Rel::TemplatePattern);
anatofuz
parents:
diff changeset
458 }
anatofuz
parents:
diff changeset
459 }
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
460 void
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
461 VisitSubstTemplateTypeParmType(const SubstTemplateTypeParmType *STTPT) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
462 Outer.add(STTPT->getReplacementType(), Flags);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
463 }
150
anatofuz
parents:
diff changeset
464 void VisitTemplateTypeParmType(const TemplateTypeParmType *TTPT) {
anatofuz
parents:
diff changeset
465 Outer.add(TTPT->getDecl(), Flags);
anatofuz
parents:
diff changeset
466 }
anatofuz
parents:
diff changeset
467 void VisitObjCInterfaceType(const ObjCInterfaceType *OIT) {
anatofuz
parents:
diff changeset
468 Outer.add(OIT->getDecl(), Flags);
anatofuz
parents:
diff changeset
469 }
anatofuz
parents:
diff changeset
470 };
anatofuz
parents:
diff changeset
471 Visitor(*this, Flags).Visit(T.getTypePtr());
anatofuz
parents:
diff changeset
472 }
anatofuz
parents:
diff changeset
473
anatofuz
parents:
diff changeset
474 void add(const NestedNameSpecifier *NNS, RelSet Flags) {
anatofuz
parents:
diff changeset
475 if (!NNS)
anatofuz
parents:
diff changeset
476 return;
anatofuz
parents:
diff changeset
477 debug(*NNS, Flags);
anatofuz
parents:
diff changeset
478 switch (NNS->getKind()) {
anatofuz
parents:
diff changeset
479 case NestedNameSpecifier::Namespace:
anatofuz
parents:
diff changeset
480 add(NNS->getAsNamespace(), Flags);
anatofuz
parents:
diff changeset
481 return;
anatofuz
parents:
diff changeset
482 case NestedNameSpecifier::NamespaceAlias:
anatofuz
parents:
diff changeset
483 add(NNS->getAsNamespaceAlias(), Flags);
anatofuz
parents:
diff changeset
484 return;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
485 case NestedNameSpecifier::Identifier:
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
486 if (Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
487 add(QualType(Resolver->resolveNestedNameSpecifierToType(NNS), 0),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
488 Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
489 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
490 return;
150
anatofuz
parents:
diff changeset
491 case NestedNameSpecifier::TypeSpec:
anatofuz
parents:
diff changeset
492 case NestedNameSpecifier::TypeSpecWithTemplate:
anatofuz
parents:
diff changeset
493 add(QualType(NNS->getAsType(), 0), Flags);
anatofuz
parents:
diff changeset
494 return;
anatofuz
parents:
diff changeset
495 case NestedNameSpecifier::Global:
anatofuz
parents:
diff changeset
496 // This should be TUDecl, but we can't get a pointer to it!
anatofuz
parents:
diff changeset
497 return;
anatofuz
parents:
diff changeset
498 case NestedNameSpecifier::Super:
anatofuz
parents:
diff changeset
499 add(NNS->getAsRecordDecl(), Flags);
anatofuz
parents:
diff changeset
500 return;
anatofuz
parents:
diff changeset
501 }
anatofuz
parents:
diff changeset
502 llvm_unreachable("unhandled NestedNameSpecifier::SpecifierKind");
anatofuz
parents:
diff changeset
503 }
anatofuz
parents:
diff changeset
504
anatofuz
parents:
diff changeset
505 void add(const CXXCtorInitializer *CCI, RelSet Flags) {
anatofuz
parents:
diff changeset
506 if (!CCI)
anatofuz
parents:
diff changeset
507 return;
anatofuz
parents:
diff changeset
508 debug(*CCI, Flags);
anatofuz
parents:
diff changeset
509
anatofuz
parents:
diff changeset
510 if (CCI->isAnyMemberInitializer())
anatofuz
parents:
diff changeset
511 add(CCI->getAnyMember(), Flags);
anatofuz
parents:
diff changeset
512 // Constructor calls contain a TypeLoc node, so we don't handle them here.
anatofuz
parents:
diff changeset
513 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
514
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
515 void add(const TemplateArgument &Arg, RelSet Flags) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
516 // Only used for template template arguments.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
517 // For type and non-type template arguments, SelectionTree
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
518 // will hit a more specific node (e.g. a TypeLoc or a
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
519 // DeclRefExpr).
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
520 if (Arg.getKind() == TemplateArgument::Template ||
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
521 Arg.getKind() == TemplateArgument::TemplateExpansion) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
522 if (TemplateDecl *TD =
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
523 Arg.getAsTemplateOrTemplatePattern().getAsTemplateDecl()) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
524 report(TD, Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
525 }
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
526 if (const auto *USD =
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
527 Arg.getAsTemplateOrTemplatePattern().getAsUsingShadowDecl())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
528 add(USD, Flags);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
529 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
530 }
150
anatofuz
parents:
diff changeset
531 };
anatofuz
parents:
diff changeset
532
anatofuz
parents:
diff changeset
533 } // namespace
anatofuz
parents:
diff changeset
534
anatofuz
parents:
diff changeset
535 llvm::SmallVector<std::pair<const NamedDecl *, DeclRelationSet>, 1>
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
536 allTargetDecls(const DynTypedNode &N, const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
537 dlog("allTargetDecls({0})", nodeToString(N));
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
538 TargetFinder Finder(Resolver);
150
anatofuz
parents:
diff changeset
539 DeclRelationSet Flags;
anatofuz
parents:
diff changeset
540 if (const Decl *D = N.get<Decl>())
anatofuz
parents:
diff changeset
541 Finder.add(D, Flags);
anatofuz
parents:
diff changeset
542 else if (const Stmt *S = N.get<Stmt>())
anatofuz
parents:
diff changeset
543 Finder.add(S, Flags);
anatofuz
parents:
diff changeset
544 else if (const NestedNameSpecifierLoc *NNSL = N.get<NestedNameSpecifierLoc>())
anatofuz
parents:
diff changeset
545 Finder.add(NNSL->getNestedNameSpecifier(), Flags);
anatofuz
parents:
diff changeset
546 else if (const NestedNameSpecifier *NNS = N.get<NestedNameSpecifier>())
anatofuz
parents:
diff changeset
547 Finder.add(NNS, Flags);
anatofuz
parents:
diff changeset
548 else if (const TypeLoc *TL = N.get<TypeLoc>())
anatofuz
parents:
diff changeset
549 Finder.add(TL->getType(), Flags);
anatofuz
parents:
diff changeset
550 else if (const QualType *QT = N.get<QualType>())
anatofuz
parents:
diff changeset
551 Finder.add(*QT, Flags);
anatofuz
parents:
diff changeset
552 else if (const CXXCtorInitializer *CCI = N.get<CXXCtorInitializer>())
anatofuz
parents:
diff changeset
553 Finder.add(CCI, Flags);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
554 else if (const TemplateArgumentLoc *TAL = N.get<TemplateArgumentLoc>())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
555 Finder.add(TAL->getArgument(), Flags);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
556 else if (const CXXBaseSpecifier *CBS = N.get<CXXBaseSpecifier>())
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
557 Finder.add(CBS->getTypeSourceInfo()->getType(), Flags);
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
558 else if (const ObjCProtocolLoc *PL = N.get<ObjCProtocolLoc>())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
559 Finder.add(PL->getProtocol(), Flags);
150
anatofuz
parents:
diff changeset
560 return Finder.takeDecls();
anatofuz
parents:
diff changeset
561 }
anatofuz
parents:
diff changeset
562
anatofuz
parents:
diff changeset
563 llvm::SmallVector<const NamedDecl *, 1>
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
564 targetDecl(const DynTypedNode &N, DeclRelationSet Mask,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
565 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
566 llvm::SmallVector<const NamedDecl *, 1> Result;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
567 for (const auto &Entry : allTargetDecls(N, Resolver)) {
150
anatofuz
parents:
diff changeset
568 if (!(Entry.second & ~Mask))
anatofuz
parents:
diff changeset
569 Result.push_back(Entry.first);
anatofuz
parents:
diff changeset
570 }
anatofuz
parents:
diff changeset
571 return Result;
anatofuz
parents:
diff changeset
572 }
anatofuz
parents:
diff changeset
573
anatofuz
parents:
diff changeset
574 llvm::SmallVector<const NamedDecl *, 1>
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
575 explicitReferenceTargets(DynTypedNode N, DeclRelationSet Mask,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
576 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
577 assert(!(Mask & (DeclRelation::TemplatePattern |
anatofuz
parents:
diff changeset
578 DeclRelation::TemplateInstantiation)) &&
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
579 "explicitReferenceTargets handles templates on its own");
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
580 auto Decls = allTargetDecls(N, Resolver);
150
anatofuz
parents:
diff changeset
581
anatofuz
parents:
diff changeset
582 // We prefer to return template instantiation, but fallback to template
anatofuz
parents:
diff changeset
583 // pattern if instantiation is not available.
anatofuz
parents:
diff changeset
584 Mask |= DeclRelation::TemplatePattern | DeclRelation::TemplateInstantiation;
anatofuz
parents:
diff changeset
585
anatofuz
parents:
diff changeset
586 llvm::SmallVector<const NamedDecl *, 1> TemplatePatterns;
anatofuz
parents:
diff changeset
587 llvm::SmallVector<const NamedDecl *, 1> Targets;
anatofuz
parents:
diff changeset
588 bool SeenTemplateInstantiations = false;
anatofuz
parents:
diff changeset
589 for (auto &D : Decls) {
anatofuz
parents:
diff changeset
590 if (D.second & ~Mask)
anatofuz
parents:
diff changeset
591 continue;
anatofuz
parents:
diff changeset
592 if (D.second & DeclRelation::TemplatePattern) {
anatofuz
parents:
diff changeset
593 TemplatePatterns.push_back(D.first);
anatofuz
parents:
diff changeset
594 continue;
anatofuz
parents:
diff changeset
595 }
anatofuz
parents:
diff changeset
596 if (D.second & DeclRelation::TemplateInstantiation)
anatofuz
parents:
diff changeset
597 SeenTemplateInstantiations = true;
anatofuz
parents:
diff changeset
598 Targets.push_back(D.first);
anatofuz
parents:
diff changeset
599 }
anatofuz
parents:
diff changeset
600 if (!SeenTemplateInstantiations)
anatofuz
parents:
diff changeset
601 Targets.insert(Targets.end(), TemplatePatterns.begin(),
anatofuz
parents:
diff changeset
602 TemplatePatterns.end());
anatofuz
parents:
diff changeset
603 return Targets;
anatofuz
parents:
diff changeset
604 }
anatofuz
parents:
diff changeset
605
anatofuz
parents:
diff changeset
606 namespace {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
607 llvm::SmallVector<ReferenceLoc> refInDecl(const Decl *D,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
608 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
609 struct Visitor : ConstDeclVisitor<Visitor> {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
610 Visitor(const HeuristicResolver *Resolver) : Resolver(Resolver) {}
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
611
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
612 const HeuristicResolver *Resolver;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
613 llvm::SmallVector<ReferenceLoc> Refs;
150
anatofuz
parents:
diff changeset
614
anatofuz
parents:
diff changeset
615 void VisitUsingDirectiveDecl(const UsingDirectiveDecl *D) {
anatofuz
parents:
diff changeset
616 // We want to keep it as non-declaration references, as the
anatofuz
parents:
diff changeset
617 // "using namespace" declaration doesn't have a name.
anatofuz
parents:
diff changeset
618 Refs.push_back(ReferenceLoc{D->getQualifierLoc(),
anatofuz
parents:
diff changeset
619 D->getIdentLocation(),
anatofuz
parents:
diff changeset
620 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
621 {D->getNominatedNamespaceAsWritten()}});
anatofuz
parents:
diff changeset
622 }
anatofuz
parents:
diff changeset
623
anatofuz
parents:
diff changeset
624 void VisitUsingDecl(const UsingDecl *D) {
anatofuz
parents:
diff changeset
625 // "using ns::identifier;" is a non-declaration reference.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
626 Refs.push_back(ReferenceLoc{
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
627 D->getQualifierLoc(), D->getLocation(), /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
628 explicitReferenceTargets(DynTypedNode::create(*D),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
629 DeclRelation::Underlying, Resolver)});
150
anatofuz
parents:
diff changeset
630 }
anatofuz
parents:
diff changeset
631
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
632 void VisitUsingEnumDecl(const UsingEnumDecl *D) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
633 // "using enum ns::E" is a non-declaration reference.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
634 // The reference is covered by the embedded typeloc.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
635 // Don't use the default VisitNamedDecl, which would report a declaration.
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
636 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
637
150
anatofuz
parents:
diff changeset
638 void VisitNamespaceAliasDecl(const NamespaceAliasDecl *D) {
anatofuz
parents:
diff changeset
639 // For namespace alias, "namespace Foo = Target;", we add two references.
anatofuz
parents:
diff changeset
640 // Add a declaration reference for Foo.
anatofuz
parents:
diff changeset
641 VisitNamedDecl(D);
anatofuz
parents:
diff changeset
642 // Add a non-declaration reference for Target.
anatofuz
parents:
diff changeset
643 Refs.push_back(ReferenceLoc{D->getQualifierLoc(),
anatofuz
parents:
diff changeset
644 D->getTargetNameLoc(),
anatofuz
parents:
diff changeset
645 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
646 {D->getAliasedNamespace()}});
anatofuz
parents:
diff changeset
647 }
anatofuz
parents:
diff changeset
648
anatofuz
parents:
diff changeset
649 void VisitNamedDecl(const NamedDecl *ND) {
anatofuz
parents:
diff changeset
650 // We choose to ignore {Class, Function, Var, TypeAlias}TemplateDecls. As
anatofuz
parents:
diff changeset
651 // as their underlying decls, covering the same range, will be visited.
anatofuz
parents:
diff changeset
652 if (llvm::isa<ClassTemplateDecl>(ND) ||
anatofuz
parents:
diff changeset
653 llvm::isa<FunctionTemplateDecl>(ND) ||
anatofuz
parents:
diff changeset
654 llvm::isa<VarTemplateDecl>(ND) ||
anatofuz
parents:
diff changeset
655 llvm::isa<TypeAliasTemplateDecl>(ND))
anatofuz
parents:
diff changeset
656 return;
anatofuz
parents:
diff changeset
657 // FIXME: decide on how to surface destructors when we need them.
anatofuz
parents:
diff changeset
658 if (llvm::isa<CXXDestructorDecl>(ND))
anatofuz
parents:
diff changeset
659 return;
anatofuz
parents:
diff changeset
660 // Filter anonymous decls, name location will point outside the name token
anatofuz
parents:
diff changeset
661 // and the clients are not prepared to handle that.
anatofuz
parents:
diff changeset
662 if (ND->getDeclName().isIdentifier() &&
anatofuz
parents:
diff changeset
663 !ND->getDeclName().getAsIdentifierInfo())
anatofuz
parents:
diff changeset
664 return;
anatofuz
parents:
diff changeset
665 Refs.push_back(ReferenceLoc{getQualifierLoc(*ND),
anatofuz
parents:
diff changeset
666 ND->getLocation(),
anatofuz
parents:
diff changeset
667 /*IsDecl=*/true,
anatofuz
parents:
diff changeset
668 {ND}});
anatofuz
parents:
diff changeset
669 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
670
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
671 void VisitCXXDeductionGuideDecl(const CXXDeductionGuideDecl *DG) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
672 // The class template name in a deduction guide targets the class
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
673 // template.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
674 Refs.push_back(ReferenceLoc{DG->getQualifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
675 DG->getNameInfo().getLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
676 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
677 {DG->getDeducedTemplate()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
678 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
679
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
680 void VisitObjCMethodDecl(const ObjCMethodDecl *OMD) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
681 // The name may have several tokens, we can only report the first.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
682 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
683 OMD->getSelectorStartLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
684 /*IsDecl=*/true,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
685 {OMD}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
686 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
687
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
688 void VisitObjCCategoryDecl(const ObjCCategoryDecl *OCD) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
689 // getLocation is the extended class's location, not the category's.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
690 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
691 OCD->getLocation(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
692 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
693 {OCD->getClassInterface()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
694 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
695 OCD->getCategoryNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
696 /*IsDecl=*/true,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
697 {OCD}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
698 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
699
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
700 void VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *OCID) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
701 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
702 OCID->getLocation(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
703 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
704 {OCID->getClassInterface()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
705 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
706 OCID->getCategoryNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
707 /*IsDecl=*/true,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
708 {OCID->getCategoryDecl()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
709 }
150
anatofuz
parents:
diff changeset
710 };
anatofuz
parents:
diff changeset
711
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
712 Visitor V{Resolver};
150
anatofuz
parents:
diff changeset
713 V.Visit(D);
anatofuz
parents:
diff changeset
714 return V.Refs;
anatofuz
parents:
diff changeset
715 }
anatofuz
parents:
diff changeset
716
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
717 llvm::SmallVector<ReferenceLoc> refInStmt(const Stmt *S,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
718 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
719 struct Visitor : ConstStmtVisitor<Visitor> {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
720 Visitor(const HeuristicResolver *Resolver) : Resolver(Resolver) {}
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
721
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
722 const HeuristicResolver *Resolver;
150
anatofuz
parents:
diff changeset
723 // FIXME: handle more complicated cases: more ObjC, designated initializers.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
724 llvm::SmallVector<ReferenceLoc> Refs;
150
anatofuz
parents:
diff changeset
725
anatofuz
parents:
diff changeset
726 void VisitConceptSpecializationExpr(const ConceptSpecializationExpr *E) {
anatofuz
parents:
diff changeset
727 Refs.push_back(ReferenceLoc{E->getNestedNameSpecifierLoc(),
anatofuz
parents:
diff changeset
728 E->getConceptNameLoc(),
anatofuz
parents:
diff changeset
729 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
730 {E->getNamedConcept()}});
anatofuz
parents:
diff changeset
731 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
732
150
anatofuz
parents:
diff changeset
733 void VisitDeclRefExpr(const DeclRefExpr *E) {
anatofuz
parents:
diff changeset
734 Refs.push_back(ReferenceLoc{E->getQualifierLoc(),
anatofuz
parents:
diff changeset
735 E->getNameInfo().getLoc(),
anatofuz
parents:
diff changeset
736 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
737 {E->getFoundDecl()}});
anatofuz
parents:
diff changeset
738 }
anatofuz
parents:
diff changeset
739
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
740 void VisitDependentScopeDeclRefExpr(const DependentScopeDeclRefExpr *E) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
741 Refs.push_back(ReferenceLoc{
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
742 E->getQualifierLoc(), E->getNameInfo().getLoc(), /*IsDecl=*/false,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
743 explicitReferenceTargets(DynTypedNode::create(*E), {}, Resolver)});
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
744 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
745
150
anatofuz
parents:
diff changeset
746 void VisitMemberExpr(const MemberExpr *E) {
anatofuz
parents:
diff changeset
747 // Skip destructor calls to avoid duplication: TypeLoc within will be
anatofuz
parents:
diff changeset
748 // visited separately.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
749 if (llvm::isa<CXXDestructorDecl>(E->getFoundDecl().getDecl()))
150
anatofuz
parents:
diff changeset
750 return;
anatofuz
parents:
diff changeset
751 Refs.push_back(ReferenceLoc{E->getQualifierLoc(),
anatofuz
parents:
diff changeset
752 E->getMemberNameInfo().getLoc(),
anatofuz
parents:
diff changeset
753 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
754 {E->getFoundDecl()}});
anatofuz
parents:
diff changeset
755 }
anatofuz
parents:
diff changeset
756
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
757 void
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
758 VisitCXXDependentScopeMemberExpr(const CXXDependentScopeMemberExpr *E) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
759 Refs.push_back(ReferenceLoc{
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
760 E->getQualifierLoc(), E->getMemberNameInfo().getLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
761 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
762 explicitReferenceTargets(DynTypedNode::create(*E), {}, Resolver)});
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
763 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
764
150
anatofuz
parents:
diff changeset
765 void VisitOverloadExpr(const OverloadExpr *E) {
anatofuz
parents:
diff changeset
766 Refs.push_back(ReferenceLoc{E->getQualifierLoc(),
anatofuz
parents:
diff changeset
767 E->getNameInfo().getLoc(),
anatofuz
parents:
diff changeset
768 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
769 llvm::SmallVector<const NamedDecl *, 1>(
anatofuz
parents:
diff changeset
770 E->decls().begin(), E->decls().end())});
anatofuz
parents:
diff changeset
771 }
anatofuz
parents:
diff changeset
772
anatofuz
parents:
diff changeset
773 void VisitSizeOfPackExpr(const SizeOfPackExpr *E) {
anatofuz
parents:
diff changeset
774 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
anatofuz
parents:
diff changeset
775 E->getPackLoc(),
anatofuz
parents:
diff changeset
776 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
777 {E->getPack()}});
anatofuz
parents:
diff changeset
778 }
anatofuz
parents:
diff changeset
779
anatofuz
parents:
diff changeset
780 void VisitObjCPropertyRefExpr(const ObjCPropertyRefExpr *E) {
anatofuz
parents:
diff changeset
781 Refs.push_back(ReferenceLoc{
anatofuz
parents:
diff changeset
782 NestedNameSpecifierLoc(), E->getLocation(),
anatofuz
parents:
diff changeset
783 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
784 // Select the getter, setter, or @property depending on the call.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
785 explicitReferenceTargets(DynTypedNode::create(*E), {}, Resolver)});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
786 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
787
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
788 void VisitObjCIvarRefExpr(const ObjCIvarRefExpr *OIRE) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
789 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
790 OIRE->getLocation(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
791 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
792 {OIRE->getDecl()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
793 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
794
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
795 void VisitObjCMessageExpr(const ObjCMessageExpr *E) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
796 // The name may have several tokens, we can only report the first.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
797 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
798 E->getSelectorStartLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
799 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
800 {E->getMethodDecl()}});
150
anatofuz
parents:
diff changeset
801 }
anatofuz
parents:
diff changeset
802
anatofuz
parents:
diff changeset
803 void VisitDesignatedInitExpr(const DesignatedInitExpr *DIE) {
anatofuz
parents:
diff changeset
804 for (const DesignatedInitExpr::Designator &D : DIE->designators()) {
anatofuz
parents:
diff changeset
805 if (!D.isFieldDesignator())
anatofuz
parents:
diff changeset
806 continue;
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
807
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
808 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
809 D.getFieldLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
810 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
811 {D.getField()}});
150
anatofuz
parents:
diff changeset
812 }
anatofuz
parents:
diff changeset
813 }
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
814
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
815 void VisitGotoStmt(const GotoStmt *GS) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
816 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
817 GS->getLabelLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
818 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
819 {GS->getLabel()}});
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
820 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
821
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
822 void VisitLabelStmt(const LabelStmt *LS) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
823 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
824 LS->getIdentLoc(),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
825 /*IsDecl=*/true,
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
826 {LS->getDecl()}});
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
827 }
150
anatofuz
parents:
diff changeset
828 };
anatofuz
parents:
diff changeset
829
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
830 Visitor V{Resolver};
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
831 V.Visit(S);
150
anatofuz
parents:
diff changeset
832 return V.Refs;
anatofuz
parents:
diff changeset
833 }
anatofuz
parents:
diff changeset
834
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
835 llvm::SmallVector<ReferenceLoc>
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
836 refInTypeLoc(TypeLoc L, const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
837 struct Visitor : TypeLocVisitor<Visitor> {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
838 Visitor(const HeuristicResolver *Resolver) : Resolver(Resolver) {}
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
839
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
840 const HeuristicResolver *Resolver;
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
841 llvm::SmallVector<ReferenceLoc> Refs;
150
anatofuz
parents:
diff changeset
842
anatofuz
parents:
diff changeset
843 void VisitElaboratedTypeLoc(ElaboratedTypeLoc L) {
anatofuz
parents:
diff changeset
844 // We only know about qualifier, rest if filled by inner locations.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
845 size_t InitialSize = Refs.size();
150
anatofuz
parents:
diff changeset
846 Visit(L.getNamedTypeLoc().getUnqualifiedLoc());
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
847 size_t NewSize = Refs.size();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
848 // Add qualifier for the newly-added refs.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
849 for (unsigned I = InitialSize; I < NewSize; ++I) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
850 ReferenceLoc *Ref = &Refs[I];
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
851 // Fill in the qualifier.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
852 assert(!Ref->Qualifier.hasQualifier() && "qualifier already set");
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
853 Ref->Qualifier = L.getQualifierLoc();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
854 }
150
anatofuz
parents:
diff changeset
855 }
anatofuz
parents:
diff changeset
856
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
857 void VisitUsingTypeLoc(UsingTypeLoc L) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
858 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
859 L.getLocalSourceRange().getBegin(),
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
860 /*IsDecl=*/false,
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
861 {L.getFoundDecl()}});
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
862 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
863
150
anatofuz
parents:
diff changeset
864 void VisitTagTypeLoc(TagTypeLoc L) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
865 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
866 L.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
867 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
868 {L.getDecl()}});
150
anatofuz
parents:
diff changeset
869 }
anatofuz
parents:
diff changeset
870
anatofuz
parents:
diff changeset
871 void VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc L) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
872 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
873 L.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
874 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
875 {L.getDecl()}});
150
anatofuz
parents:
diff changeset
876 }
anatofuz
parents:
diff changeset
877
anatofuz
parents:
diff changeset
878 void VisitTemplateSpecializationTypeLoc(TemplateSpecializationTypeLoc L) {
anatofuz
parents:
diff changeset
879 // We must ensure template type aliases are included in results if they
anatofuz
parents:
diff changeset
880 // were written in the source code, e.g. in
anatofuz
parents:
diff changeset
881 // template <class T> using valias = vector<T>;
anatofuz
parents:
diff changeset
882 // ^valias<int> x;
anatofuz
parents:
diff changeset
883 // 'explicitReferenceTargets' will return:
anatofuz
parents:
diff changeset
884 // 1. valias with mask 'Alias'.
anatofuz
parents:
diff changeset
885 // 2. 'vector<int>' with mask 'Underlying'.
anatofuz
parents:
diff changeset
886 // we want to return only #1 in this case.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
887 Refs.push_back(ReferenceLoc{
150
anatofuz
parents:
diff changeset
888 NestedNameSpecifierLoc(), L.getTemplateNameLoc(), /*IsDecl=*/false,
anatofuz
parents:
diff changeset
889 explicitReferenceTargets(DynTypedNode::create(L.getType()),
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
890 DeclRelation::Alias, Resolver)});
150
anatofuz
parents:
diff changeset
891 }
anatofuz
parents:
diff changeset
892 void VisitDeducedTemplateSpecializationTypeLoc(
anatofuz
parents:
diff changeset
893 DeducedTemplateSpecializationTypeLoc L) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
894 Refs.push_back(ReferenceLoc{
150
anatofuz
parents:
diff changeset
895 NestedNameSpecifierLoc(), L.getNameLoc(), /*IsDecl=*/false,
anatofuz
parents:
diff changeset
896 explicitReferenceTargets(DynTypedNode::create(L.getType()),
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
897 DeclRelation::Alias, Resolver)});
150
anatofuz
parents:
diff changeset
898 }
anatofuz
parents:
diff changeset
899
anatofuz
parents:
diff changeset
900 void VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
901 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
902 TL.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
903 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
904 {TL.getDecl()}});
150
anatofuz
parents:
diff changeset
905 }
anatofuz
parents:
diff changeset
906
anatofuz
parents:
diff changeset
907 void VisitDependentTemplateSpecializationTypeLoc(
anatofuz
parents:
diff changeset
908 DependentTemplateSpecializationTypeLoc L) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
909 Refs.push_back(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
910 ReferenceLoc{L.getQualifierLoc(), L.getTemplateNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
911 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
912 explicitReferenceTargets(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
913 DynTypedNode::create(L.getType()), {}, Resolver)});
150
anatofuz
parents:
diff changeset
914 }
anatofuz
parents:
diff changeset
915
anatofuz
parents:
diff changeset
916 void VisitDependentNameTypeLoc(DependentNameTypeLoc L) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
917 Refs.push_back(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
918 ReferenceLoc{L.getQualifierLoc(), L.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
919 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
920 explicitReferenceTargets(
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
921 DynTypedNode::create(L.getType()), {}, Resolver)});
150
anatofuz
parents:
diff changeset
922 }
anatofuz
parents:
diff changeset
923
anatofuz
parents:
diff changeset
924 void VisitTypedefTypeLoc(TypedefTypeLoc L) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
925 if (shouldSkipTypedef(L.getTypedefNameDecl()))
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
926 return;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
927 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
928 L.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
929 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
930 {L.getTypedefNameDecl()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
931 }
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
932
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
933 void VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc L) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
934 Refs.push_back(ReferenceLoc{NestedNameSpecifierLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
935 L.getNameLoc(),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
936 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
937 {L.getIFaceDecl()}});
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
938 }
150
anatofuz
parents:
diff changeset
939 };
anatofuz
parents:
diff changeset
940
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
941 Visitor V{Resolver};
150
anatofuz
parents:
diff changeset
942 V.Visit(L.getUnqualifiedLoc());
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
943 return V.Refs;
150
anatofuz
parents:
diff changeset
944 }
anatofuz
parents:
diff changeset
945
anatofuz
parents:
diff changeset
946 class ExplicitReferenceCollector
anatofuz
parents:
diff changeset
947 : public RecursiveASTVisitor<ExplicitReferenceCollector> {
anatofuz
parents:
diff changeset
948 public:
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
949 ExplicitReferenceCollector(llvm::function_ref<void(ReferenceLoc)> Out,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
950 const HeuristicResolver *Resolver)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
951 : Out(Out), Resolver(Resolver) {
150
anatofuz
parents:
diff changeset
952 assert(Out);
anatofuz
parents:
diff changeset
953 }
anatofuz
parents:
diff changeset
954
anatofuz
parents:
diff changeset
955 bool VisitTypeLoc(TypeLoc TTL) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
956 if (TypeLocsToSkip.count(TTL.getBeginLoc()))
150
anatofuz
parents:
diff changeset
957 return true;
anatofuz
parents:
diff changeset
958 visitNode(DynTypedNode::create(TTL));
anatofuz
parents:
diff changeset
959 return true;
anatofuz
parents:
diff changeset
960 }
anatofuz
parents:
diff changeset
961
anatofuz
parents:
diff changeset
962 bool TraverseElaboratedTypeLoc(ElaboratedTypeLoc L) {
anatofuz
parents:
diff changeset
963 // ElaboratedTypeLoc will reports information for its inner type loc.
anatofuz
parents:
diff changeset
964 // Otherwise we loose information about inner types loc's qualifier.
anatofuz
parents:
diff changeset
965 TypeLoc Inner = L.getNamedTypeLoc().getUnqualifiedLoc();
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
966 if (L.getBeginLoc() == Inner.getBeginLoc())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
967 return RecursiveASTVisitor::TraverseTypeLoc(Inner);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
968 else
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
969 TypeLocsToSkip.insert(Inner.getBeginLoc());
150
anatofuz
parents:
diff changeset
970 return RecursiveASTVisitor::TraverseElaboratedTypeLoc(L);
anatofuz
parents:
diff changeset
971 }
anatofuz
parents:
diff changeset
972
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
973 bool VisitStmt(Stmt *S) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
974 visitNode(DynTypedNode::create(*S));
150
anatofuz
parents:
diff changeset
975 return true;
anatofuz
parents:
diff changeset
976 }
anatofuz
parents:
diff changeset
977
anatofuz
parents:
diff changeset
978 bool TraverseOpaqueValueExpr(OpaqueValueExpr *OVE) {
anatofuz
parents:
diff changeset
979 visitNode(DynTypedNode::create(*OVE));
anatofuz
parents:
diff changeset
980 // Not clear why the source expression is skipped by default...
anatofuz
parents:
diff changeset
981 // FIXME: can we just make RecursiveASTVisitor do this?
anatofuz
parents:
diff changeset
982 return RecursiveASTVisitor::TraverseStmt(OVE->getSourceExpr());
anatofuz
parents:
diff changeset
983 }
anatofuz
parents:
diff changeset
984
anatofuz
parents:
diff changeset
985 bool TraversePseudoObjectExpr(PseudoObjectExpr *POE) {
anatofuz
parents:
diff changeset
986 visitNode(DynTypedNode::create(*POE));
anatofuz
parents:
diff changeset
987 // Traverse only the syntactic form to find the *written* references.
anatofuz
parents:
diff changeset
988 // (The semantic form also contains lots of duplication)
anatofuz
parents:
diff changeset
989 return RecursiveASTVisitor::TraverseStmt(POE->getSyntacticForm());
anatofuz
parents:
diff changeset
990 }
anatofuz
parents:
diff changeset
991
anatofuz
parents:
diff changeset
992 // We re-define Traverse*, since there's no corresponding Visit*.
anatofuz
parents:
diff changeset
993 // TemplateArgumentLoc is the only way to get locations for references to
anatofuz
parents:
diff changeset
994 // template template parameters.
anatofuz
parents:
diff changeset
995 bool TraverseTemplateArgumentLoc(TemplateArgumentLoc A) {
anatofuz
parents:
diff changeset
996 switch (A.getArgument().getKind()) {
anatofuz
parents:
diff changeset
997 case TemplateArgument::Template:
anatofuz
parents:
diff changeset
998 case TemplateArgument::TemplateExpansion:
anatofuz
parents:
diff changeset
999 reportReference(ReferenceLoc{A.getTemplateQualifierLoc(),
anatofuz
parents:
diff changeset
1000 A.getTemplateNameLoc(),
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1001 /*IsDecl=*/false,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1002 {A.getArgument()
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1003 .getAsTemplateOrTemplatePattern()
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1004 .getAsTemplateDecl()}},
150
anatofuz
parents:
diff changeset
1005 DynTypedNode::create(A.getArgument()));
anatofuz
parents:
diff changeset
1006 break;
anatofuz
parents:
diff changeset
1007 case TemplateArgument::Declaration:
anatofuz
parents:
diff changeset
1008 break; // FIXME: can this actually happen in TemplateArgumentLoc?
anatofuz
parents:
diff changeset
1009 case TemplateArgument::Integral:
anatofuz
parents:
diff changeset
1010 case TemplateArgument::Null:
anatofuz
parents:
diff changeset
1011 case TemplateArgument::NullPtr:
anatofuz
parents:
diff changeset
1012 break; // no references.
anatofuz
parents:
diff changeset
1013 case TemplateArgument::Pack:
anatofuz
parents:
diff changeset
1014 case TemplateArgument::Type:
anatofuz
parents:
diff changeset
1015 case TemplateArgument::Expression:
anatofuz
parents:
diff changeset
1016 break; // Handled by VisitType and VisitExpression.
anatofuz
parents:
diff changeset
1017 };
anatofuz
parents:
diff changeset
1018 return RecursiveASTVisitor::TraverseTemplateArgumentLoc(A);
anatofuz
parents:
diff changeset
1019 }
anatofuz
parents:
diff changeset
1020
anatofuz
parents:
diff changeset
1021 bool VisitDecl(Decl *D) {
anatofuz
parents:
diff changeset
1022 visitNode(DynTypedNode::create(*D));
anatofuz
parents:
diff changeset
1023 return true;
anatofuz
parents:
diff changeset
1024 }
anatofuz
parents:
diff changeset
1025
anatofuz
parents:
diff changeset
1026 // We have to use Traverse* because there is no corresponding Visit*.
anatofuz
parents:
diff changeset
1027 bool TraverseNestedNameSpecifierLoc(NestedNameSpecifierLoc L) {
anatofuz
parents:
diff changeset
1028 if (!L.getNestedNameSpecifier())
anatofuz
parents:
diff changeset
1029 return true;
anatofuz
parents:
diff changeset
1030 visitNode(DynTypedNode::create(L));
anatofuz
parents:
diff changeset
1031 // Inner type is missing information about its qualifier, skip it.
anatofuz
parents:
diff changeset
1032 if (auto TL = L.getTypeLoc())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1033 TypeLocsToSkip.insert(TL.getBeginLoc());
150
anatofuz
parents:
diff changeset
1034 return RecursiveASTVisitor::TraverseNestedNameSpecifierLoc(L);
anatofuz
parents:
diff changeset
1035 }
anatofuz
parents:
diff changeset
1036
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1037 bool TraverseObjCProtocolLoc(ObjCProtocolLoc ProtocolLoc) {
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1038 visitNode(DynTypedNode::create(ProtocolLoc));
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1039 return true;
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1040 }
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1041
150
anatofuz
parents:
diff changeset
1042 bool TraverseConstructorInitializer(CXXCtorInitializer *Init) {
anatofuz
parents:
diff changeset
1043 visitNode(DynTypedNode::create(*Init));
anatofuz
parents:
diff changeset
1044 return RecursiveASTVisitor::TraverseConstructorInitializer(Init);
anatofuz
parents:
diff changeset
1045 }
anatofuz
parents:
diff changeset
1046
anatofuz
parents:
diff changeset
1047 private:
anatofuz
parents:
diff changeset
1048 /// Obtain information about a reference directly defined in \p N. Does not
anatofuz
parents:
diff changeset
1049 /// recurse into child nodes, e.g. do not expect references for constructor
anatofuz
parents:
diff changeset
1050 /// initializers
anatofuz
parents:
diff changeset
1051 ///
anatofuz
parents:
diff changeset
1052 /// Any of the fields in the returned structure can be empty, but not all of
anatofuz
parents:
diff changeset
1053 /// them, e.g.
anatofuz
parents:
diff changeset
1054 /// - for implicitly generated nodes (e.g. MemberExpr from range-based-for),
anatofuz
parents:
diff changeset
1055 /// source location information may be missing,
anatofuz
parents:
diff changeset
1056 /// - for dependent code, targets may be empty.
anatofuz
parents:
diff changeset
1057 ///
anatofuz
parents:
diff changeset
1058 /// (!) For the purposes of this function declarations are not considered to
anatofuz
parents:
diff changeset
1059 /// be references. However, declarations can have references inside them,
anatofuz
parents:
diff changeset
1060 /// e.g. 'namespace foo = std' references namespace 'std' and this
anatofuz
parents:
diff changeset
1061 /// function will return the corresponding reference.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1062 llvm::SmallVector<ReferenceLoc> explicitReference(DynTypedNode N) {
150
anatofuz
parents:
diff changeset
1063 if (auto *D = N.get<Decl>())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1064 return refInDecl(D, Resolver);
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
1065 if (auto *S = N.get<Stmt>())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1066 return refInStmt(S, Resolver);
150
anatofuz
parents:
diff changeset
1067 if (auto *NNSL = N.get<NestedNameSpecifierLoc>()) {
anatofuz
parents:
diff changeset
1068 // (!) 'DeclRelation::Alias' ensures we do not loose namespace aliases.
anatofuz
parents:
diff changeset
1069 return {ReferenceLoc{
anatofuz
parents:
diff changeset
1070 NNSL->getPrefix(), NNSL->getLocalBeginLoc(), false,
anatofuz
parents:
diff changeset
1071 explicitReferenceTargets(
anatofuz
parents:
diff changeset
1072 DynTypedNode::create(*NNSL->getNestedNameSpecifier()),
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1073 DeclRelation::Alias, Resolver)}};
150
anatofuz
parents:
diff changeset
1074 }
anatofuz
parents:
diff changeset
1075 if (const TypeLoc *TL = N.get<TypeLoc>())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1076 return refInTypeLoc(*TL, Resolver);
150
anatofuz
parents:
diff changeset
1077 if (const CXXCtorInitializer *CCI = N.get<CXXCtorInitializer>()) {
anatofuz
parents:
diff changeset
1078 // Other type initializers (e.g. base initializer) are handled by visiting
anatofuz
parents:
diff changeset
1079 // the typeLoc.
anatofuz
parents:
diff changeset
1080 if (CCI->isAnyMemberInitializer()) {
anatofuz
parents:
diff changeset
1081 return {ReferenceLoc{NestedNameSpecifierLoc(),
anatofuz
parents:
diff changeset
1082 CCI->getMemberLocation(),
anatofuz
parents:
diff changeset
1083 /*IsDecl=*/false,
anatofuz
parents:
diff changeset
1084 {CCI->getAnyMember()}}};
anatofuz
parents:
diff changeset
1085 }
anatofuz
parents:
diff changeset
1086 }
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1087 if (const ObjCProtocolLoc *PL = N.get<ObjCProtocolLoc>())
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1088 return {ReferenceLoc{NestedNameSpecifierLoc(),
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1089 PL->getLocation(),
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1090 /*IsDecl=*/false,
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1091 {PL->getProtocol()}}};
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1092
150
anatofuz
parents:
diff changeset
1093 // We do not have location information for other nodes (QualType, etc)
anatofuz
parents:
diff changeset
1094 return {};
anatofuz
parents:
diff changeset
1095 }
anatofuz
parents:
diff changeset
1096
anatofuz
parents:
diff changeset
1097 void visitNode(DynTypedNode N) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1098 for (auto &R : explicitReference(N))
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1099 reportReference(std::move(R), N);
150
anatofuz
parents:
diff changeset
1100 }
anatofuz
parents:
diff changeset
1101
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1102 void reportReference(ReferenceLoc &&Ref, DynTypedNode N) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1103 // Strip null targets that can arise from invalid code.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1104 // (This avoids having to check for null everywhere we insert)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1105 llvm::erase_value(Ref.Targets, nullptr);
150
anatofuz
parents:
diff changeset
1106 // Our promise is to return only references from the source code. If we lack
anatofuz
parents:
diff changeset
1107 // location information, skip these nodes.
anatofuz
parents:
diff changeset
1108 // Normally this should not happen in practice, unless there are bugs in the
anatofuz
parents:
diff changeset
1109 // traversals or users started the traversal at an implicit node.
anatofuz
parents:
diff changeset
1110 if (Ref.NameLoc.isInvalid()) {
anatofuz
parents:
diff changeset
1111 dlog("invalid location at node {0}", nodeToString(N));
anatofuz
parents:
diff changeset
1112 return;
anatofuz
parents:
diff changeset
1113 }
anatofuz
parents:
diff changeset
1114 Out(Ref);
anatofuz
parents:
diff changeset
1115 }
anatofuz
parents:
diff changeset
1116
anatofuz
parents:
diff changeset
1117 llvm::function_ref<void(ReferenceLoc)> Out;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1118 const HeuristicResolver *Resolver;
150
anatofuz
parents:
diff changeset
1119 /// TypeLocs starting at these locations must be skipped, see
anatofuz
parents:
diff changeset
1120 /// TraverseElaboratedTypeSpecifierLoc for details.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1121 llvm::DenseSet<SourceLocation> TypeLocsToSkip;
150
anatofuz
parents:
diff changeset
1122 };
anatofuz
parents:
diff changeset
1123 } // namespace
anatofuz
parents:
diff changeset
1124
anatofuz
parents:
diff changeset
1125 void findExplicitReferences(const Stmt *S,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1126 llvm::function_ref<void(ReferenceLoc)> Out,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1127 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
1128 assert(S);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1129 ExplicitReferenceCollector(Out, Resolver).TraverseStmt(const_cast<Stmt *>(S));
150
anatofuz
parents:
diff changeset
1130 }
anatofuz
parents:
diff changeset
1131 void findExplicitReferences(const Decl *D,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1132 llvm::function_ref<void(ReferenceLoc)> Out,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1133 const HeuristicResolver *Resolver) {
150
anatofuz
parents:
diff changeset
1134 assert(D);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1135 ExplicitReferenceCollector(Out, Resolver).TraverseDecl(const_cast<Decl *>(D));
150
anatofuz
parents:
diff changeset
1136 }
anatofuz
parents:
diff changeset
1137 void findExplicitReferences(const ASTContext &AST,
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1138 llvm::function_ref<void(ReferenceLoc)> Out,
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1139 const HeuristicResolver *Resolver) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1140 ExplicitReferenceCollector(Out, Resolver)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
1141 .TraverseAST(const_cast<ASTContext &>(AST));
150
anatofuz
parents:
diff changeset
1142 }
anatofuz
parents:
diff changeset
1143
anatofuz
parents:
diff changeset
1144 llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, DeclRelation R) {
anatofuz
parents:
diff changeset
1145 switch (R) {
anatofuz
parents:
diff changeset
1146 #define REL_CASE(X) \
anatofuz
parents:
diff changeset
1147 case DeclRelation::X: \
anatofuz
parents:
diff changeset
1148 return OS << #X;
anatofuz
parents:
diff changeset
1149 REL_CASE(Alias);
anatofuz
parents:
diff changeset
1150 REL_CASE(Underlying);
anatofuz
parents:
diff changeset
1151 REL_CASE(TemplateInstantiation);
anatofuz
parents:
diff changeset
1152 REL_CASE(TemplatePattern);
anatofuz
parents:
diff changeset
1153 #undef REL_CASE
anatofuz
parents:
diff changeset
1154 }
anatofuz
parents:
diff changeset
1155 llvm_unreachable("Unhandled DeclRelation enum");
anatofuz
parents:
diff changeset
1156 }
anatofuz
parents:
diff changeset
1157 llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, DeclRelationSet RS) {
anatofuz
parents:
diff changeset
1158 const char *Sep = "";
anatofuz
parents:
diff changeset
1159 for (unsigned I = 0; I < RS.S.size(); ++I) {
anatofuz
parents:
diff changeset
1160 if (RS.S.test(I)) {
anatofuz
parents:
diff changeset
1161 OS << Sep << static_cast<DeclRelation>(I);
anatofuz
parents:
diff changeset
1162 Sep = "|";
anatofuz
parents:
diff changeset
1163 }
anatofuz
parents:
diff changeset
1164 }
anatofuz
parents:
diff changeset
1165 return OS;
anatofuz
parents:
diff changeset
1166 }
anatofuz
parents:
diff changeset
1167
anatofuz
parents:
diff changeset
1168 llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, ReferenceLoc R) {
anatofuz
parents:
diff changeset
1169 // note we cannot print R.NameLoc without a source manager.
anatofuz
parents:
diff changeset
1170 OS << "targets = {";
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1171 llvm::SmallVector<std::string> Targets;
150
anatofuz
parents:
diff changeset
1172 for (const NamedDecl *T : R.Targets) {
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1173 llvm::raw_string_ostream Target(Targets.emplace_back());
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1174 Target << printQualifiedName(*T) << printTemplateSpecializationArgs(*T);
150
anatofuz
parents:
diff changeset
1175 }
236
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1176 llvm::sort(Targets);
c4bab56944e8 LLVM 16
kono
parents: 223
diff changeset
1177 OS << llvm::join(Targets, ", ");
150
anatofuz
parents:
diff changeset
1178 OS << "}";
anatofuz
parents:
diff changeset
1179 if (R.Qualifier) {
anatofuz
parents:
diff changeset
1180 OS << ", qualifier = '";
anatofuz
parents:
diff changeset
1181 R.Qualifier.getNestedNameSpecifier()->print(OS,
anatofuz
parents:
diff changeset
1182 PrintingPolicy(LangOptions()));
anatofuz
parents:
diff changeset
1183 OS << "'";
anatofuz
parents:
diff changeset
1184 }
anatofuz
parents:
diff changeset
1185 if (R.IsDecl)
anatofuz
parents:
diff changeset
1186 OS << ", decl";
anatofuz
parents:
diff changeset
1187 return OS;
anatofuz
parents:
diff changeset
1188 }
anatofuz
parents:
diff changeset
1189
anatofuz
parents:
diff changeset
1190 } // namespace clangd
anatofuz
parents:
diff changeset
1191 } // namespace clang