]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/llvm/tools/clang/lib/Index/IndexDecl.cpp
Merge clang trunk r351319, resolve conflicts, and update FREEBSD-Xlist.
[FreeBSD/FreeBSD.git] / contrib / llvm / tools / clang / lib / Index / IndexDecl.cpp
1 //===- IndexDecl.cpp - Indexing declarations ------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9
10 #include "IndexingContext.h"
11 #include "clang/Index/IndexDataConsumer.h"
12 #include "clang/AST/DeclVisitor.h"
13
14 using namespace clang;
15 using namespace index;
16
17 #define TRY_DECL(D,CALL_EXPR)                                                  \
18   do {                                                                         \
19     if (!IndexCtx.shouldIndex(D)) return true;                                 \
20     if (!CALL_EXPR)                                                            \
21       return false;                                                            \
22   } while (0)
23
24 #define TRY_TO(CALL_EXPR)                                                      \
25   do {                                                                         \
26     if (!CALL_EXPR)                                                            \
27       return false;                                                            \
28   } while (0)
29
30 namespace {
31
32 class IndexingDeclVisitor : public ConstDeclVisitor<IndexingDeclVisitor, bool> {
33   IndexingContext &IndexCtx;
34
35 public:
36   explicit IndexingDeclVisitor(IndexingContext &indexCtx)
37     : IndexCtx(indexCtx) { }
38
39   bool Handled = true;
40
41   bool VisitDecl(const Decl *D) {
42     Handled = false;
43     return true;
44   }
45
46   /// Returns true if the given method has been defined explicitly by the
47   /// user.
48   static bool hasUserDefined(const ObjCMethodDecl *D,
49                              const ObjCImplDecl *Container) {
50     const ObjCMethodDecl *MD = Container->getMethod(D->getSelector(),
51                                                     D->isInstanceMethod());
52     return MD && !MD->isImplicit() && MD->isThisDeclarationADefinition();
53   }
54
55   void handleTemplateArgumentLoc(const TemplateArgumentLoc &TALoc,
56                                  const NamedDecl *Parent,
57                                  const DeclContext *DC) {
58     const TemplateArgumentLocInfo &LocInfo = TALoc.getLocInfo();
59     switch (TALoc.getArgument().getKind()) {
60     case TemplateArgument::Expression:
61       IndexCtx.indexBody(LocInfo.getAsExpr(), Parent, DC);
62       break;
63     case TemplateArgument::Type:
64       IndexCtx.indexTypeSourceInfo(LocInfo.getAsTypeSourceInfo(), Parent, DC);
65       break;
66     case TemplateArgument::Template:
67     case TemplateArgument::TemplateExpansion:
68       IndexCtx.indexNestedNameSpecifierLoc(TALoc.getTemplateQualifierLoc(),
69                                            Parent, DC);
70       if (const TemplateDecl *TD = TALoc.getArgument()
71                                        .getAsTemplateOrTemplatePattern()
72                                        .getAsTemplateDecl()) {
73         if (const NamedDecl *TTD = TD->getTemplatedDecl())
74           IndexCtx.handleReference(TTD, TALoc.getTemplateNameLoc(), Parent, DC);
75       }
76       break;
77     default:
78       break;
79     }
80   }
81
82   void handleDeclarator(const DeclaratorDecl *D,
83                         const NamedDecl *Parent = nullptr,
84                         bool isIBType = false) {
85     if (!Parent) Parent = D;
86
87     IndexCtx.indexTypeSourceInfo(D->getTypeSourceInfo(), Parent,
88                                  Parent->getLexicalDeclContext(),
89                                  /*isBase=*/false, isIBType);
90     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent);
91     if (IndexCtx.shouldIndexFunctionLocalSymbols()) {
92       // Only index parameters in definitions, parameters in declarations are
93       // not useful.
94       if (const ParmVarDecl *Parm = dyn_cast<ParmVarDecl>(D)) {
95         auto *DC = Parm->getDeclContext();
96         if (auto *FD = dyn_cast<FunctionDecl>(DC)) {
97           if (FD->isThisDeclarationADefinition())
98             IndexCtx.handleDecl(Parm);
99         } else if (auto *MD = dyn_cast<ObjCMethodDecl>(DC)) {
100           if (MD->isThisDeclarationADefinition())
101             IndexCtx.handleDecl(Parm);
102         } else {
103           IndexCtx.handleDecl(Parm);
104         }
105       } else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
106         if (FD->isThisDeclarationADefinition()) {
107           for (auto PI : FD->parameters()) {
108             IndexCtx.handleDecl(PI);
109           }
110         }
111       }
112     } else {
113       // Index the default parameter value for function definitions.
114       if (const auto *FD = dyn_cast<FunctionDecl>(D)) {
115         if (FD->isThisDeclarationADefinition()) {
116           for (const auto *PV : FD->parameters()) {
117             if (PV->hasDefaultArg() && !PV->hasUninstantiatedDefaultArg() &&
118                 !PV->hasUnparsedDefaultArg())
119               IndexCtx.indexBody(PV->getDefaultArg(), D);
120           }
121         }
122       }
123     }
124   }
125
126   bool handleObjCMethod(const ObjCMethodDecl *D,
127                         const ObjCPropertyDecl *AssociatedProp = nullptr) {
128     SmallVector<SymbolRelation, 4> Relations;
129     SmallVector<const ObjCMethodDecl*, 4> Overriden;
130
131     D->getOverriddenMethods(Overriden);
132     for(auto overridden: Overriden) {
133       Relations.emplace_back((unsigned) SymbolRole::RelationOverrideOf,
134                              overridden);
135     }
136     if (AssociatedProp)
137       Relations.emplace_back((unsigned)SymbolRole::RelationAccessorOf,
138                              AssociatedProp);
139
140     // getLocation() returns beginning token of a method declaration, but for
141     // indexing purposes we want to point to the base name.
142     SourceLocation MethodLoc = D->getSelectorStartLoc();
143     if (MethodLoc.isInvalid())
144       MethodLoc = D->getLocation();
145
146     SourceLocation AttrLoc;
147
148     // check for (getter=/setter=)
149     if (AssociatedProp) {
150       bool isGetter = !D->param_size();
151       AttrLoc = isGetter ?
152         AssociatedProp->getGetterNameLoc():
153         AssociatedProp->getSetterNameLoc();
154     }
155
156     SymbolRoleSet Roles = (SymbolRoleSet)SymbolRole::Dynamic;
157     if (D->isImplicit()) {
158       if (AttrLoc.isValid()) {
159         MethodLoc = AttrLoc;
160       } else {
161         Roles |= (SymbolRoleSet)SymbolRole::Implicit;
162       }
163     } else if (AttrLoc.isValid()) {
164       IndexCtx.handleReference(D, AttrLoc, cast<NamedDecl>(D->getDeclContext()),
165                                D->getDeclContext(), 0);
166     }
167
168     TRY_DECL(D, IndexCtx.handleDecl(D, MethodLoc, Roles, Relations));
169     IndexCtx.indexTypeSourceInfo(D->getReturnTypeSourceInfo(), D);
170     bool hasIBActionAndFirst = D->hasAttr<IBActionAttr>();
171     for (const auto *I : D->parameters()) {
172       handleDeclarator(I, D, /*isIBType=*/hasIBActionAndFirst);
173       hasIBActionAndFirst = false;
174     }
175
176     if (D->isThisDeclarationADefinition()) {
177       const Stmt *Body = D->getBody();
178       if (Body) {
179         IndexCtx.indexBody(Body, D, D);
180       }
181     }
182     return true;
183   }
184
185   /// Gather the declarations which the given declaration \D overrides in a
186   /// pseudo-override manner.
187   ///
188   /// Pseudo-overrides occur when a class template specialization declares
189   /// a declaration that has the same name as a similar declaration in the
190   /// non-specialized template.
191   void
192   gatherTemplatePseudoOverrides(const NamedDecl *D,
193                                 SmallVectorImpl<SymbolRelation> &Relations) {
194     if (!IndexCtx.getLangOpts().CPlusPlus)
195       return;
196     const auto *CTSD =
197         dyn_cast<ClassTemplateSpecializationDecl>(D->getLexicalDeclContext());
198     if (!CTSD)
199       return;
200     llvm::PointerUnion<ClassTemplateDecl *,
201                        ClassTemplatePartialSpecializationDecl *>
202         Template = CTSD->getSpecializedTemplateOrPartial();
203     if (const auto *CTD = Template.dyn_cast<ClassTemplateDecl *>()) {
204       const CXXRecordDecl *Pattern = CTD->getTemplatedDecl();
205       bool TypeOverride = isa<TypeDecl>(D);
206       for (const NamedDecl *ND : Pattern->lookup(D->getDeclName())) {
207         if (const auto *CTD = dyn_cast<ClassTemplateDecl>(ND))
208           ND = CTD->getTemplatedDecl();
209         if (ND->isImplicit())
210           continue;
211         // Types can override other types.
212         if (!TypeOverride) {
213           if (ND->getKind() != D->getKind())
214             continue;
215         } else if (!isa<TypeDecl>(ND))
216           continue;
217         if (const auto *FD = dyn_cast<FunctionDecl>(ND)) {
218           const auto *DFD = cast<FunctionDecl>(D);
219           // Function overrides are approximated using the number of parameters.
220           if (FD->getStorageClass() != DFD->getStorageClass() ||
221               FD->getNumParams() != DFD->getNumParams())
222             continue;
223         }
224         Relations.emplace_back(
225             SymbolRoleSet(SymbolRole::RelationSpecializationOf), ND);
226       }
227     }
228   }
229
230   bool VisitFunctionDecl(const FunctionDecl *D) {
231     SymbolRoleSet Roles{};
232     SmallVector<SymbolRelation, 4> Relations;
233     if (auto *CXXMD = dyn_cast<CXXMethodDecl>(D)) {
234       if (CXXMD->isVirtual())
235         Roles |= (unsigned)SymbolRole::Dynamic;
236       for (const CXXMethodDecl *O : CXXMD->overridden_methods()) {
237         Relations.emplace_back((unsigned)SymbolRole::RelationOverrideOf, O);
238       }
239     }
240     gatherTemplatePseudoOverrides(D, Relations);
241     if (const auto *Base = D->getPrimaryTemplate())
242       Relations.push_back(
243           SymbolRelation(SymbolRoleSet(SymbolRole::RelationSpecializationOf),
244                          Base->getTemplatedDecl()));
245
246     TRY_DECL(D, IndexCtx.handleDecl(D, Roles, Relations));
247     handleDeclarator(D);
248
249     if (const CXXConstructorDecl *Ctor = dyn_cast<CXXConstructorDecl>(D)) {
250       IndexCtx.handleReference(Ctor->getParent(), Ctor->getLocation(),
251                                Ctor->getParent(), Ctor->getDeclContext());
252
253       // Constructor initializers.
254       for (const auto *Init : Ctor->inits()) {
255         if (Init->isWritten()) {
256           IndexCtx.indexTypeSourceInfo(Init->getTypeSourceInfo(), D);
257           if (const FieldDecl *Member = Init->getAnyMember())
258             IndexCtx.handleReference(Member, Init->getMemberLocation(), D, D,
259                                      (unsigned)SymbolRole::Write);
260           IndexCtx.indexBody(Init->getInit(), D, D);
261         }
262       }
263     } else if (const CXXDestructorDecl *Dtor = dyn_cast<CXXDestructorDecl>(D)) {
264       if (auto TypeNameInfo = Dtor->getNameInfo().getNamedTypeInfo()) {
265         IndexCtx.handleReference(Dtor->getParent(),
266                                  TypeNameInfo->getTypeLoc().getBeginLoc(),
267                                  Dtor->getParent(), Dtor->getDeclContext());
268       }
269     } else if (const auto *Guide = dyn_cast<CXXDeductionGuideDecl>(D)) {
270       IndexCtx.handleReference(Guide->getDeducedTemplate()->getTemplatedDecl(),
271                                Guide->getLocation(), Guide,
272                                Guide->getDeclContext());
273     }
274     // Template specialization arguments.
275     if (const ASTTemplateArgumentListInfo *TemplateArgInfo =
276             D->getTemplateSpecializationArgsAsWritten()) {
277       for (const auto &Arg : TemplateArgInfo->arguments())
278         handleTemplateArgumentLoc(Arg, D, D->getLexicalDeclContext());
279     }
280
281     if (D->isThisDeclarationADefinition()) {
282       const Stmt *Body = D->getBody();
283       if (Body) {
284         IndexCtx.indexBody(Body, D, D);
285       }
286     }
287     return true;
288   }
289
290   bool VisitVarDecl(const VarDecl *D) {
291     SmallVector<SymbolRelation, 4> Relations;
292     gatherTemplatePseudoOverrides(D, Relations);
293     TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
294     handleDeclarator(D);
295     IndexCtx.indexBody(D->getInit(), D);
296     return true;
297   }
298
299   bool VisitDecompositionDecl(const DecompositionDecl *D) {
300     for (const auto *Binding : D->bindings())
301       TRY_DECL(Binding, IndexCtx.handleDecl(Binding));
302     return Base::VisitDecompositionDecl(D);
303   }
304
305   bool VisitFieldDecl(const FieldDecl *D) {
306     SmallVector<SymbolRelation, 4> Relations;
307     gatherTemplatePseudoOverrides(D, Relations);
308     TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
309     handleDeclarator(D);
310     if (D->isBitField())
311       IndexCtx.indexBody(D->getBitWidth(), D);
312     else if (D->hasInClassInitializer())
313       IndexCtx.indexBody(D->getInClassInitializer(), D);
314     return true;
315   }
316
317   bool VisitObjCIvarDecl(const ObjCIvarDecl *D) {
318     if (D->getSynthesize()) {
319       // handled in VisitObjCPropertyImplDecl
320       return true;
321     }
322     TRY_DECL(D, IndexCtx.handleDecl(D));
323     handleDeclarator(D);
324     return true;
325   }
326
327   bool VisitMSPropertyDecl(const MSPropertyDecl *D) {
328     handleDeclarator(D);
329     return true;
330   }
331
332   bool VisitEnumConstantDecl(const EnumConstantDecl *D) {
333     TRY_DECL(D, IndexCtx.handleDecl(D));
334     IndexCtx.indexBody(D->getInitExpr(), D);
335     return true;
336   }
337
338   bool VisitTypedefNameDecl(const TypedefNameDecl *D) {
339     if (!D->isTransparentTag()) {
340       SmallVector<SymbolRelation, 4> Relations;
341       gatherTemplatePseudoOverrides(D, Relations);
342       TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
343       IndexCtx.indexTypeSourceInfo(D->getTypeSourceInfo(), D);
344     }
345     return true;
346   }
347
348   bool VisitTagDecl(const TagDecl *D) {
349     // Non-free standing tags are handled in indexTypeSourceInfo.
350     if (D->isFreeStanding()) {
351       if (D->isThisDeclarationADefinition()) {
352         SmallVector<SymbolRelation, 4> Relations;
353         gatherTemplatePseudoOverrides(D, Relations);
354         IndexCtx.indexTagDecl(D, Relations);
355       } else {
356         SmallVector<SymbolRelation, 1> Relations;
357         gatherTemplatePseudoOverrides(D, Relations);
358         return IndexCtx.handleDecl(D, D->getLocation(), SymbolRoleSet(),
359                                    Relations, D->getLexicalDeclContext());
360       }
361     }
362     return true;
363   }
364
365   bool handleReferencedProtocols(const ObjCProtocolList &ProtList,
366                                  const ObjCContainerDecl *ContD,
367                                  SourceLocation SuperLoc) {
368     ObjCInterfaceDecl::protocol_loc_iterator LI = ProtList.loc_begin();
369     for (ObjCInterfaceDecl::protocol_iterator
370          I = ProtList.begin(), E = ProtList.end(); I != E; ++I, ++LI) {
371       SourceLocation Loc = *LI;
372       ObjCProtocolDecl *PD = *I;
373       SymbolRoleSet roles{};
374       if (Loc == SuperLoc)
375         roles |= (SymbolRoleSet)SymbolRole::Implicit;
376       TRY_TO(IndexCtx.handleReference(PD, Loc, ContD, ContD, roles,
377           SymbolRelation{(unsigned)SymbolRole::RelationBaseOf, ContD}));
378     }
379     return true;
380   }
381
382   bool VisitObjCInterfaceDecl(const ObjCInterfaceDecl *D) {
383     if (D->isThisDeclarationADefinition()) {
384       TRY_DECL(D, IndexCtx.handleDecl(D));
385       SourceLocation SuperLoc = D->getSuperClassLoc();
386       if (auto *SuperD = D->getSuperClass()) {
387         bool hasSuperTypedef = false;
388         if (auto *TInfo = D->getSuperClassTInfo()) {
389           if (auto *TT = TInfo->getType()->getAs<TypedefType>()) {
390             if (auto *TD = TT->getDecl()) {
391               hasSuperTypedef = true;
392               TRY_TO(IndexCtx.handleReference(TD, SuperLoc, D, D,
393                                               SymbolRoleSet()));
394             }
395           }
396         }
397         SymbolRoleSet superRoles{};
398         if (hasSuperTypedef)
399           superRoles |= (SymbolRoleSet)SymbolRole::Implicit;
400         TRY_TO(IndexCtx.handleReference(SuperD, SuperLoc, D, D, superRoles,
401             SymbolRelation{(unsigned)SymbolRole::RelationBaseOf, D}));
402       }
403       TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
404                                        SuperLoc));
405       TRY_TO(IndexCtx.indexDeclContext(D));
406     } else {
407       return IndexCtx.handleReference(D, D->getLocation(), nullptr,
408                                       D->getDeclContext(), SymbolRoleSet());
409     }
410     return true;
411   }
412
413   bool VisitObjCProtocolDecl(const ObjCProtocolDecl *D) {
414     if (D->isThisDeclarationADefinition()) {
415       TRY_DECL(D, IndexCtx.handleDecl(D));
416       TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
417                                        /*superLoc=*/SourceLocation()));
418       TRY_TO(IndexCtx.indexDeclContext(D));
419     } else {
420       return IndexCtx.handleReference(D, D->getLocation(), nullptr,
421                                       D->getDeclContext(), SymbolRoleSet());
422     }
423     return true;
424   }
425
426   bool VisitObjCImplementationDecl(const ObjCImplementationDecl *D) {
427     const ObjCInterfaceDecl *Class = D->getClassInterface();
428     if (!Class)
429       return true;
430
431     if (Class->isImplicitInterfaceDecl())
432       IndexCtx.handleDecl(Class);
433
434     TRY_DECL(D, IndexCtx.handleDecl(D));
435
436     // Visit implicit @synthesize property implementations first as their
437     // location is reported at the name of the @implementation block. This
438     // serves no purpose other than to simplify the FileCheck-based tests.
439     for (const auto *I : D->property_impls()) {
440       if (I->getLocation().isInvalid())
441         IndexCtx.indexDecl(I);
442     }
443     for (const auto *I : D->decls()) {
444       if (!isa<ObjCPropertyImplDecl>(I) ||
445           cast<ObjCPropertyImplDecl>(I)->getLocation().isValid())
446         IndexCtx.indexDecl(I);
447     }
448
449     return true;
450   }
451
452   bool VisitObjCCategoryDecl(const ObjCCategoryDecl *D) {
453     if (!IndexCtx.shouldIndex(D))
454       return true;
455     const ObjCInterfaceDecl *C = D->getClassInterface();
456     if (!C)
457       return true;
458     TRY_TO(IndexCtx.handleReference(C, D->getLocation(), D, D, SymbolRoleSet(),
459                                    SymbolRelation{
460                                      (unsigned)SymbolRole::RelationExtendedBy, D
461                                    }));
462     SourceLocation CategoryLoc = D->getCategoryNameLoc();
463     if (!CategoryLoc.isValid())
464       CategoryLoc = D->getLocation();
465     TRY_TO(IndexCtx.handleDecl(D, CategoryLoc));
466     TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
467                                      /*superLoc=*/SourceLocation()));
468     TRY_TO(IndexCtx.indexDeclContext(D));
469     return true;
470   }
471
472   bool VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *D) {
473     const ObjCCategoryDecl *Cat = D->getCategoryDecl();
474     if (!Cat)
475       return true;
476     const ObjCInterfaceDecl *C = D->getClassInterface();
477     if (C)
478       TRY_TO(IndexCtx.handleReference(C, D->getLocation(), D, D,
479                                       SymbolRoleSet()));
480     SourceLocation CategoryLoc = D->getCategoryNameLoc();
481     if (!CategoryLoc.isValid())
482       CategoryLoc = D->getLocation();
483     TRY_DECL(D, IndexCtx.handleDecl(D, CategoryLoc));
484     IndexCtx.indexDeclContext(D);
485     return true;
486   }
487
488   bool VisitObjCMethodDecl(const ObjCMethodDecl *D) {
489     // Methods associated with a property, even user-declared ones, are
490     // handled when we handle the property.
491     if (D->isPropertyAccessor())
492       return true;
493
494     handleObjCMethod(D);
495     return true;
496   }
497
498   bool VisitObjCPropertyDecl(const ObjCPropertyDecl *D) {
499     if (ObjCMethodDecl *MD = D->getGetterMethodDecl())
500       if (MD->getLexicalDeclContext() == D->getLexicalDeclContext())
501         handleObjCMethod(MD, D);
502     if (ObjCMethodDecl *MD = D->getSetterMethodDecl())
503       if (MD->getLexicalDeclContext() == D->getLexicalDeclContext())
504         handleObjCMethod(MD, D);
505     TRY_DECL(D, IndexCtx.handleDecl(D));
506     if (IBOutletCollectionAttr *attr = D->getAttr<IBOutletCollectionAttr>())
507       IndexCtx.indexTypeSourceInfo(attr->getInterfaceLoc(), D,
508                                    D->getLexicalDeclContext(), false, true);
509     IndexCtx.indexTypeSourceInfo(D->getTypeSourceInfo(), D);
510     return true;
511   }
512
513   bool VisitObjCPropertyImplDecl(const ObjCPropertyImplDecl *D) {
514     ObjCPropertyDecl *PD = D->getPropertyDecl();
515     auto *Container = cast<ObjCImplDecl>(D->getDeclContext());
516     SourceLocation Loc = D->getLocation();
517     SymbolRoleSet Roles = 0;
518     SmallVector<SymbolRelation, 1> Relations;
519
520     if (ObjCIvarDecl *ID = D->getPropertyIvarDecl())
521       Relations.push_back({(SymbolRoleSet)SymbolRole::RelationAccessorOf, ID});
522     if (Loc.isInvalid()) {
523       Loc = Container->getLocation();
524       Roles |= (SymbolRoleSet)SymbolRole::Implicit;
525     }
526     TRY_DECL(D, IndexCtx.handleDecl(D, Loc, Roles, Relations));
527
528     if (D->getPropertyImplementation() == ObjCPropertyImplDecl::Dynamic)
529       return true;
530
531     assert(D->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize);
532     SymbolRoleSet AccessorMethodRoles =
533       SymbolRoleSet(SymbolRole::Dynamic) | SymbolRoleSet(SymbolRole::Implicit);
534     if (ObjCMethodDecl *MD = PD->getGetterMethodDecl()) {
535       if (MD->isPropertyAccessor() &&
536           !hasUserDefined(MD, Container))
537         IndexCtx.handleDecl(MD, Loc, AccessorMethodRoles, {}, Container);
538     }
539     if (ObjCMethodDecl *MD = PD->getSetterMethodDecl()) {
540       if (MD->isPropertyAccessor() &&
541           !hasUserDefined(MD, Container))
542         IndexCtx.handleDecl(MD, Loc, AccessorMethodRoles, {}, Container);
543     }
544     if (ObjCIvarDecl *IvarD = D->getPropertyIvarDecl()) {
545       if (IvarD->getSynthesize()) {
546         // For synthesized ivars, use the location of its name in the
547         // corresponding @synthesize. If there isn't one, use the containing
548         // @implementation's location, rather than the property's location,
549         // otherwise the header file containing the @interface will have different
550         // indexing contents based on whether the @implementation was present or
551         // not in the translation unit.
552         SymbolRoleSet IvarRoles = 0;
553         SourceLocation IvarLoc = D->getPropertyIvarDeclLoc();
554         if (D->getLocation().isInvalid()) {
555           IvarLoc = Container->getLocation();
556           IvarRoles = (SymbolRoleSet)SymbolRole::Implicit;
557         } else if (D->getLocation() == IvarLoc) {
558           IvarRoles = (SymbolRoleSet)SymbolRole::Implicit;
559         }
560         TRY_DECL(IvarD, IndexCtx.handleDecl(IvarD, IvarLoc, IvarRoles));
561       } else {
562         IndexCtx.handleReference(IvarD, D->getPropertyIvarDeclLoc(), nullptr,
563                                  D->getDeclContext(), SymbolRoleSet());
564       }
565     }
566     return true;
567   }
568
569   bool VisitNamespaceDecl(const NamespaceDecl *D) {
570     TRY_DECL(D, IndexCtx.handleDecl(D));
571     IndexCtx.indexDeclContext(D);
572     return true;
573   }
574
575   bool VisitNamespaceAliasDecl(const NamespaceAliasDecl *D) {
576     TRY_DECL(D, IndexCtx.handleDecl(D));
577     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), D);
578     IndexCtx.handleReference(D->getAliasedNamespace(), D->getTargetNameLoc(), D,
579                              D->getLexicalDeclContext());
580     return true;
581   }
582
583   bool VisitUsingDecl(const UsingDecl *D) {
584     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
585     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
586
587     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
588                                          D->getLexicalDeclContext());
589     for (const auto *I : D->shadows())
590       IndexCtx.handleReference(I->getUnderlyingDecl(), D->getLocation(), Parent,
591                                D->getLexicalDeclContext(), SymbolRoleSet());
592     return true;
593   }
594
595   bool VisitUsingDirectiveDecl(const UsingDirectiveDecl *D) {
596     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
597     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
598
599     // NNS for the local 'using namespace' directives is visited by the body
600     // visitor.
601     if (!D->getParentFunctionOrMethod())
602       IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
603                                            D->getLexicalDeclContext());
604
605     return IndexCtx.handleReference(D->getNominatedNamespaceAsWritten(),
606                                     D->getLocation(), Parent,
607                                     D->getLexicalDeclContext(),
608                                     SymbolRoleSet());
609   }
610
611   bool VisitUnresolvedUsingValueDecl(const UnresolvedUsingValueDecl *D) {
612     TRY_DECL(D, IndexCtx.handleDecl(D));
613     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
614     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
615     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
616                                          D->getLexicalDeclContext());
617     return true;
618   }
619
620   bool VisitUnresolvedUsingTypenameDecl(const UnresolvedUsingTypenameDecl *D) {
621     TRY_DECL(D, IndexCtx.handleDecl(D));
622     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
623     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
624     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
625                                          D->getLexicalDeclContext());
626     return true;
627   }
628
629   bool VisitClassTemplateSpecializationDecl(const
630                                            ClassTemplateSpecializationDecl *D) {
631     // FIXME: Notify subsequent callbacks if info comes from implicit
632     // instantiation.
633     llvm::PointerUnion<ClassTemplateDecl *,
634                        ClassTemplatePartialSpecializationDecl *>
635         Template = D->getSpecializedTemplateOrPartial();
636     const Decl *SpecializationOf =
637         Template.is<ClassTemplateDecl *>()
638             ? (Decl *)Template.get<ClassTemplateDecl *>()
639             : Template.get<ClassTemplatePartialSpecializationDecl *>();
640     if (!D->isThisDeclarationADefinition())
641       IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), D);
642     IndexCtx.indexTagDecl(
643         D, SymbolRelation(SymbolRoleSet(SymbolRole::RelationSpecializationOf),
644                           SpecializationOf));
645     if (TypeSourceInfo *TSI = D->getTypeAsWritten())
646       IndexCtx.indexTypeSourceInfo(TSI, /*Parent=*/nullptr,
647                                    D->getLexicalDeclContext());
648     return true;
649   }
650
651   static bool shouldIndexTemplateParameterDefaultValue(const NamedDecl *D) {
652     if (!D)
653       return false;
654     // We want to index the template parameters only once when indexing the
655     // canonical declaration.
656     if (const auto *FD = dyn_cast<FunctionDecl>(D))
657       return FD->getCanonicalDecl() == FD;
658     else if (const auto *TD = dyn_cast<TagDecl>(D))
659       return TD->getCanonicalDecl() == TD;
660     else if (const auto *VD = dyn_cast<VarDecl>(D))
661       return VD->getCanonicalDecl() == VD;
662     return true;
663   }
664
665   bool VisitTemplateDecl(const TemplateDecl *D) {
666
667     const NamedDecl *Parent = D->getTemplatedDecl();
668     if (!Parent)
669       return true;
670
671     // Index the default values for the template parameters.
672     if (D->getTemplateParameters() &&
673         shouldIndexTemplateParameterDefaultValue(Parent)) {
674       const TemplateParameterList *Params = D->getTemplateParameters();
675       for (const NamedDecl *TP : *Params) {
676         if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(TP)) {
677           if (TTP->hasDefaultArgument())
678             IndexCtx.indexTypeSourceInfo(TTP->getDefaultArgumentInfo(), Parent);
679         } else if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(TP)) {
680           if (NTTP->hasDefaultArgument())
681             IndexCtx.indexBody(NTTP->getDefaultArgument(), Parent);
682         } else if (const auto *TTPD = dyn_cast<TemplateTemplateParmDecl>(TP)) {
683           if (TTPD->hasDefaultArgument())
684             handleTemplateArgumentLoc(TTPD->getDefaultArgument(), Parent,
685                                       TP->getLexicalDeclContext());
686         }
687       }
688     }
689
690     return Visit(Parent);
691   }
692
693   bool VisitFriendDecl(const FriendDecl *D) {
694     if (auto ND = D->getFriendDecl()) {
695       // FIXME: Ignore a class template in a dependent context, these are not
696       // linked properly with their redeclarations, ending up with duplicate
697       // USRs.
698       // See comment "Friend templates are visible in fairly strange ways." in
699       // SemaTemplate.cpp which precedes code that prevents the friend template
700       // from becoming visible from the enclosing context.
701       if (isa<ClassTemplateDecl>(ND) && D->getDeclContext()->isDependentContext())
702         return true;
703       return Visit(ND);
704     }
705     if (auto Ty = D->getFriendType()) {
706       IndexCtx.indexTypeSourceInfo(Ty, cast<NamedDecl>(D->getDeclContext()));
707     }
708     return true;
709   }
710
711   bool VisitImportDecl(const ImportDecl *D) {
712     return IndexCtx.importedModule(D);
713   }
714
715   bool VisitStaticAssertDecl(const StaticAssertDecl *D) {
716     IndexCtx.indexBody(D->getAssertExpr(),
717                        dyn_cast<NamedDecl>(D->getDeclContext()),
718                        D->getLexicalDeclContext());
719     return true;
720   }
721 };
722
723 } // anonymous namespace
724
725 bool IndexingContext::indexDecl(const Decl *D) {
726   if (D->isImplicit() && shouldIgnoreIfImplicit(D))
727     return true;
728
729   if (isTemplateImplicitInstantiation(D) && !shouldIndexImplicitInstantiation())
730     return true;
731
732   IndexingDeclVisitor Visitor(*this);
733   bool ShouldContinue = Visitor.Visit(D);
734   if (!ShouldContinue)
735     return false;
736
737   if (!Visitor.Handled && isa<DeclContext>(D))
738     return indexDeclContext(cast<DeclContext>(D));
739
740   return true;
741 }
742
743 bool IndexingContext::indexDeclContext(const DeclContext *DC) {
744   for (const auto *I : DC->decls())
745     if (!indexDecl(I))
746       return false;
747   return true;
748 }
749
750 bool IndexingContext::indexTopLevelDecl(const Decl *D) {
751   if (D->getLocation().isInvalid())
752     return true;
753
754   if (isa<ObjCMethodDecl>(D))
755     return true; // Wait for the objc container.
756
757   return indexDecl(D);
758 }
759
760 bool IndexingContext::indexDeclGroupRef(DeclGroupRef DG) {
761   for (DeclGroupRef::iterator I = DG.begin(), E = DG.end(); I != E; ++I)
762     if (!indexTopLevelDecl(*I))
763       return false;
764   return true;
765 }