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