]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - lib/Index/IndexDecl.cpp
Vendor import of clang trunk r306325:
[FreeBSD/FreeBSD.git] / 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   /// \brief 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 (auto I = CXXMD->begin_overridden_methods(),
237            E = CXXMD->end_overridden_methods(); I != E; ++I) {
238         Relations.emplace_back((unsigned)SymbolRole::RelationOverrideOf, *I);
239       }
240     }
241     gatherTemplatePseudoOverrides(D, Relations);
242     if (const auto *Base = D->getPrimaryTemplate())
243       Relations.push_back(
244           SymbolRelation(SymbolRoleSet(SymbolRole::RelationSpecializationOf),
245                          Base->getTemplatedDecl()));
246
247     TRY_DECL(D, IndexCtx.handleDecl(D, Roles, Relations));
248     handleDeclarator(D);
249
250     if (const CXXConstructorDecl *Ctor = dyn_cast<CXXConstructorDecl>(D)) {
251       IndexCtx.handleReference(Ctor->getParent(), Ctor->getLocation(),
252                                Ctor->getParent(), Ctor->getDeclContext());
253
254       // Constructor initializers.
255       for (const auto *Init : Ctor->inits()) {
256         if (Init->isWritten()) {
257           IndexCtx.indexTypeSourceInfo(Init->getTypeSourceInfo(), D);
258           if (const FieldDecl *Member = Init->getAnyMember())
259             IndexCtx.handleReference(Member, Init->getMemberLocation(), D, D,
260                                      (unsigned)SymbolRole::Write);
261           IndexCtx.indexBody(Init->getInit(), D, D);
262         }
263       }
264     } else if (const CXXDestructorDecl *Dtor = dyn_cast<CXXDestructorDecl>(D)) {
265       if (auto TypeNameInfo = Dtor->getNameInfo().getNamedTypeInfo()) {
266         IndexCtx.handleReference(Dtor->getParent(),
267                                  TypeNameInfo->getTypeLoc().getLocStart(),
268                                  Dtor->getParent(), Dtor->getDeclContext());
269       }
270     }
271     // Template specialization arguments.
272     if (const ASTTemplateArgumentListInfo *TemplateArgInfo =
273             D->getTemplateSpecializationArgsAsWritten()) {
274       for (const auto &Arg : TemplateArgInfo->arguments())
275         handleTemplateArgumentLoc(Arg, D, D->getLexicalDeclContext());
276     }
277
278     if (D->isThisDeclarationADefinition()) {
279       const Stmt *Body = D->getBody();
280       if (Body) {
281         IndexCtx.indexBody(Body, D, D);
282       }
283     }
284     return true;
285   }
286
287   bool VisitVarDecl(const VarDecl *D) {
288     SmallVector<SymbolRelation, 4> Relations;
289     gatherTemplatePseudoOverrides(D, Relations);
290     TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
291     handleDeclarator(D);
292     IndexCtx.indexBody(D->getInit(), D);
293     return true;
294   }
295
296   bool VisitDecompositionDecl(const DecompositionDecl *D) {
297     for (const auto *Binding : D->bindings())
298       TRY_DECL(Binding, IndexCtx.handleDecl(Binding));
299     return Base::VisitDecompositionDecl(D);
300   }
301
302   bool VisitFieldDecl(const FieldDecl *D) {
303     SmallVector<SymbolRelation, 4> Relations;
304     gatherTemplatePseudoOverrides(D, Relations);
305     TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
306     handleDeclarator(D);
307     if (D->isBitField())
308       IndexCtx.indexBody(D->getBitWidth(), D);
309     else if (D->hasInClassInitializer())
310       IndexCtx.indexBody(D->getInClassInitializer(), D);
311     return true;
312   }
313
314   bool VisitObjCIvarDecl(const ObjCIvarDecl *D) {
315     if (D->getSynthesize()) {
316       // handled in VisitObjCPropertyImplDecl
317       return true;
318     }
319     TRY_DECL(D, IndexCtx.handleDecl(D));
320     handleDeclarator(D);
321     return true;
322   }
323
324   bool VisitMSPropertyDecl(const MSPropertyDecl *D) {
325     handleDeclarator(D);
326     return true;
327   }
328
329   bool VisitEnumConstantDecl(const EnumConstantDecl *D) {
330     TRY_DECL(D, IndexCtx.handleDecl(D));
331     IndexCtx.indexBody(D->getInitExpr(), D);
332     return true;
333   }
334
335   bool VisitTypedefNameDecl(const TypedefNameDecl *D) {
336     if (!D->isTransparentTag()) {
337       SmallVector<SymbolRelation, 4> Relations;
338       gatherTemplatePseudoOverrides(D, Relations);
339       TRY_DECL(D, IndexCtx.handleDecl(D, SymbolRoleSet(), Relations));
340       IndexCtx.indexTypeSourceInfo(D->getTypeSourceInfo(), D);
341     }
342     return true;
343   }
344
345   bool VisitTagDecl(const TagDecl *D) {
346     // Non-free standing tags are handled in indexTypeSourceInfo.
347     if (D->isFreeStanding()) {
348       if (D->isThisDeclarationADefinition()) {
349         SmallVector<SymbolRelation, 4> Relations;
350         gatherTemplatePseudoOverrides(D, Relations);
351         IndexCtx.indexTagDecl(D, Relations);
352       } else {
353         auto *Parent = dyn_cast<NamedDecl>(D->getDeclContext());
354         SmallVector<SymbolRelation, 1> Relations;
355         gatherTemplatePseudoOverrides(D, Relations);
356         return IndexCtx.handleReference(D, D->getLocation(), Parent,
357                                         D->getLexicalDeclContext(),
358                                         SymbolRoleSet(), Relations);
359       }
360     }
361     return true;
362   }
363
364   bool handleReferencedProtocols(const ObjCProtocolList &ProtList,
365                                  const ObjCContainerDecl *ContD,
366                                  SourceLocation SuperLoc) {
367     ObjCInterfaceDecl::protocol_loc_iterator LI = ProtList.loc_begin();
368     for (ObjCInterfaceDecl::protocol_iterator
369          I = ProtList.begin(), E = ProtList.end(); I != E; ++I, ++LI) {
370       SourceLocation Loc = *LI;
371       ObjCProtocolDecl *PD = *I;
372       SymbolRoleSet roles{};
373       if (Loc == SuperLoc)
374         roles |= (SymbolRoleSet)SymbolRole::Implicit;
375       TRY_TO(IndexCtx.handleReference(PD, Loc, ContD, ContD, roles,
376           SymbolRelation{(unsigned)SymbolRole::RelationBaseOf, ContD}));
377     }
378     return true;
379   }
380
381   bool VisitObjCInterfaceDecl(const ObjCInterfaceDecl *D) {
382     if (D->isThisDeclarationADefinition()) {
383       TRY_DECL(D, IndexCtx.handleDecl(D));
384       SourceLocation SuperLoc = D->getSuperClassLoc();
385       if (auto *SuperD = D->getSuperClass()) {
386         bool hasSuperTypedef = false;
387         if (auto *TInfo = D->getSuperClassTInfo()) {
388           if (auto *TT = TInfo->getType()->getAs<TypedefType>()) {
389             if (auto *TD = TT->getDecl()) {
390               hasSuperTypedef = true;
391               TRY_TO(IndexCtx.handleReference(TD, SuperLoc, D, D,
392                                               SymbolRoleSet()));
393             }
394           }
395         }
396         SymbolRoleSet superRoles{};
397         if (hasSuperTypedef)
398           superRoles |= (SymbolRoleSet)SymbolRole::Implicit;
399         TRY_TO(IndexCtx.handleReference(SuperD, SuperLoc, D, D, superRoles,
400             SymbolRelation{(unsigned)SymbolRole::RelationBaseOf, D}));
401       }
402       TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
403                                        SuperLoc));
404       TRY_TO(IndexCtx.indexDeclContext(D));
405     } else {
406       return IndexCtx.handleReference(D, D->getLocation(), nullptr,
407                                       D->getDeclContext(), SymbolRoleSet());
408     }
409     return true;
410   }
411
412   bool VisitObjCProtocolDecl(const ObjCProtocolDecl *D) {
413     if (D->isThisDeclarationADefinition()) {
414       TRY_DECL(D, IndexCtx.handleDecl(D));
415       TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
416                                        /*superLoc=*/SourceLocation()));
417       TRY_TO(IndexCtx.indexDeclContext(D));
418     } else {
419       return IndexCtx.handleReference(D, D->getLocation(), nullptr,
420                                       D->getDeclContext(), SymbolRoleSet());
421     }
422     return true;
423   }
424
425   bool VisitObjCImplementationDecl(const ObjCImplementationDecl *D) {
426     const ObjCInterfaceDecl *Class = D->getClassInterface();
427     if (!Class)
428       return true;
429
430     if (Class->isImplicitInterfaceDecl())
431       IndexCtx.handleDecl(Class);
432
433     TRY_DECL(D, IndexCtx.handleDecl(D));
434
435     // Visit implicit @synthesize property implementations first as their
436     // location is reported at the name of the @implementation block. This
437     // serves no purpose other than to simplify the FileCheck-based tests.
438     for (const auto *I : D->property_impls()) {
439       if (I->getLocation().isInvalid())
440         IndexCtx.indexDecl(I);
441     }
442     for (const auto *I : D->decls()) {
443       if (!isa<ObjCPropertyImplDecl>(I) ||
444           cast<ObjCPropertyImplDecl>(I)->getLocation().isValid())
445         IndexCtx.indexDecl(I);
446     }
447
448     return true;
449   }
450
451   bool VisitObjCCategoryDecl(const ObjCCategoryDecl *D) {
452     if (!IndexCtx.shouldIndex(D))
453       return true;
454     const ObjCInterfaceDecl *C = D->getClassInterface();
455     if (!C)
456       return true;
457     TRY_TO(IndexCtx.handleReference(C, D->getLocation(), D, D, SymbolRoleSet(),
458                                    SymbolRelation{
459                                      (unsigned)SymbolRole::RelationExtendedBy, D
460                                    }));
461     SourceLocation CategoryLoc = D->getCategoryNameLoc();
462     if (!CategoryLoc.isValid())
463       CategoryLoc = D->getLocation();
464     TRY_TO(IndexCtx.handleDecl(D, CategoryLoc));
465     TRY_TO(handleReferencedProtocols(D->getReferencedProtocols(), D,
466                                      /*superLoc=*/SourceLocation()));
467     TRY_TO(IndexCtx.indexDeclContext(D));
468     return true;
469   }
470
471   bool VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *D) {
472     const ObjCCategoryDecl *Cat = D->getCategoryDecl();
473     if (!Cat)
474       return true;
475     const ObjCInterfaceDecl *C = D->getClassInterface();
476     if (C)
477       TRY_TO(IndexCtx.handleReference(C, D->getLocation(), D, D,
478                                       SymbolRoleSet()));
479     SourceLocation CategoryLoc = D->getCategoryNameLoc();
480     if (!CategoryLoc.isValid())
481       CategoryLoc = D->getLocation();
482     TRY_DECL(D, IndexCtx.handleDecl(D, CategoryLoc));
483     IndexCtx.indexDeclContext(D);
484     return true;
485   }
486
487   bool VisitObjCMethodDecl(const ObjCMethodDecl *D) {
488     // Methods associated with a property, even user-declared ones, are
489     // handled when we handle the property.
490     if (D->isPropertyAccessor())
491       return true;
492
493     handleObjCMethod(D);
494     return true;
495   }
496
497   bool VisitObjCPropertyDecl(const ObjCPropertyDecl *D) {
498     if (ObjCMethodDecl *MD = D->getGetterMethodDecl())
499       if (MD->getLexicalDeclContext() == D->getLexicalDeclContext())
500         handleObjCMethod(MD, D);
501     if (ObjCMethodDecl *MD = D->getSetterMethodDecl())
502       if (MD->getLexicalDeclContext() == D->getLexicalDeclContext())
503         handleObjCMethod(MD, D);
504     TRY_DECL(D, IndexCtx.handleDecl(D));
505     if (IBOutletCollectionAttr *attr = D->getAttr<IBOutletCollectionAttr>())
506       IndexCtx.indexTypeSourceInfo(attr->getInterfaceLoc(), D,
507                                    D->getLexicalDeclContext(), false, true);
508     IndexCtx.indexTypeSourceInfo(D->getTypeSourceInfo(), D);
509     return true;
510   }
511
512   bool VisitObjCPropertyImplDecl(const ObjCPropertyImplDecl *D) {
513     ObjCPropertyDecl *PD = D->getPropertyDecl();
514     auto *Container = cast<ObjCImplDecl>(D->getDeclContext());
515     SourceLocation Loc = D->getLocation();
516     SymbolRoleSet Roles = 0;
517     SmallVector<SymbolRelation, 1> Relations;
518
519     if (ObjCIvarDecl *ID = D->getPropertyIvarDecl())
520       Relations.push_back({(SymbolRoleSet)SymbolRole::RelationAccessorOf, ID});
521     if (Loc.isInvalid()) {
522       Loc = Container->getLocation();
523       Roles |= (SymbolRoleSet)SymbolRole::Implicit;
524     }
525     TRY_DECL(D, IndexCtx.handleDecl(D, Loc, Roles, Relations));
526
527     if (D->getPropertyImplementation() == ObjCPropertyImplDecl::Dynamic)
528       return true;
529
530     assert(D->getPropertyImplementation() == ObjCPropertyImplDecl::Synthesize);
531     SymbolRoleSet AccessorMethodRoles =
532       SymbolRoleSet(SymbolRole::Dynamic) | SymbolRoleSet(SymbolRole::Implicit);
533     if (ObjCMethodDecl *MD = PD->getGetterMethodDecl()) {
534       if (MD->isPropertyAccessor() &&
535           !hasUserDefined(MD, Container))
536         IndexCtx.handleDecl(MD, Loc, AccessorMethodRoles, {}, Container);
537     }
538     if (ObjCMethodDecl *MD = PD->getSetterMethodDecl()) {
539       if (MD->isPropertyAccessor() &&
540           !hasUserDefined(MD, Container))
541         IndexCtx.handleDecl(MD, Loc, AccessorMethodRoles, {}, Container);
542     }
543     if (ObjCIvarDecl *IvarD = D->getPropertyIvarDecl()) {
544       if (IvarD->getSynthesize()) {
545         // For synthesized ivars, use the location of its name in the
546         // corresponding @synthesize. If there isn't one, use the containing
547         // @implementation's location, rather than the property's location,
548         // otherwise the header file containing the @interface will have different
549         // indexing contents based on whether the @implementation was present or
550         // not in the translation unit.
551         SymbolRoleSet IvarRoles = 0;
552         SourceLocation IvarLoc = D->getPropertyIvarDeclLoc();
553         if (D->getLocation().isInvalid()) {
554           IvarLoc = Container->getLocation();
555           IvarRoles = (SymbolRoleSet)SymbolRole::Implicit;
556         } else if (D->getLocation() == IvarLoc) {
557           IvarRoles = (SymbolRoleSet)SymbolRole::Implicit;
558         }
559         TRY_DECL(IvarD, IndexCtx.handleDecl(IvarD, IvarLoc, IvarRoles));
560       } else {
561         IndexCtx.handleReference(IvarD, D->getPropertyIvarDeclLoc(), nullptr,
562                                  D->getDeclContext(), SymbolRoleSet());
563       }
564     }
565     return true;
566   }
567
568   bool VisitNamespaceDecl(const NamespaceDecl *D) {
569     TRY_DECL(D, IndexCtx.handleDecl(D));
570     IndexCtx.indexDeclContext(D);
571     return true;
572   }
573
574   bool VisitNamespaceAliasDecl(const NamespaceAliasDecl *D) {
575     TRY_DECL(D, IndexCtx.handleDecl(D));
576     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), D);
577     IndexCtx.handleReference(D->getAliasedNamespace(), D->getTargetNameLoc(), D,
578                              D->getLexicalDeclContext());
579     return true;
580   }
581
582   bool VisitUsingDecl(const UsingDecl *D) {
583     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
584     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
585
586     IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
587                                          D->getLexicalDeclContext());
588     for (const auto *I : D->shadows())
589       IndexCtx.handleReference(I->getUnderlyingDecl(), D->getLocation(), Parent,
590                                D->getLexicalDeclContext(), SymbolRoleSet());
591     return true;
592   }
593
594   bool VisitUsingDirectiveDecl(const UsingDirectiveDecl *D) {
595     const DeclContext *DC = D->getDeclContext()->getRedeclContext();
596     const NamedDecl *Parent = dyn_cast<NamedDecl>(DC);
597
598     // NNS for the local 'using namespace' directives is visited by the body
599     // visitor.
600     if (!D->getParentFunctionOrMethod())
601       IndexCtx.indexNestedNameSpecifierLoc(D->getQualifierLoc(), Parent,
602                                            D->getLexicalDeclContext());
603
604     return IndexCtx.handleReference(D->getNominatedNamespaceAsWritten(),
605                                     D->getLocation(), Parent,
606                                     D->getLexicalDeclContext(),
607                                     SymbolRoleSet());
608   }
609
610   bool VisitClassTemplateSpecializationDecl(const
611                                            ClassTemplateSpecializationDecl *D) {
612     // FIXME: Notify subsequent callbacks if info comes from implicit
613     // instantiation.
614     llvm::PointerUnion<ClassTemplateDecl *,
615                        ClassTemplatePartialSpecializationDecl *>
616         Template = D->getSpecializedTemplateOrPartial();
617     const Decl *SpecializationOf =
618         Template.is<ClassTemplateDecl *>()
619             ? (Decl *)Template.get<ClassTemplateDecl *>()
620             : Template.get<ClassTemplatePartialSpecializationDecl *>();
621     IndexCtx.indexTagDecl(
622         D, SymbolRelation(SymbolRoleSet(SymbolRole::RelationSpecializationOf),
623                           SpecializationOf));
624     if (TypeSourceInfo *TSI = D->getTypeAsWritten())
625       IndexCtx.indexTypeSourceInfo(TSI, /*Parent=*/nullptr,
626                                    D->getLexicalDeclContext());
627     return true;
628   }
629
630   static bool shouldIndexTemplateParameterDefaultValue(const NamedDecl *D) {
631     if (!D)
632       return false;
633     // We want to index the template parameters only once when indexing the
634     // canonical declaration.
635     if (const auto *FD = dyn_cast<FunctionDecl>(D))
636       return FD->getCanonicalDecl() == FD;
637     else if (const auto *TD = dyn_cast<TagDecl>(D))
638       return TD->getCanonicalDecl() == TD;
639     else if (const auto *VD = dyn_cast<VarDecl>(D))
640       return VD->getCanonicalDecl() == VD;
641     return true;
642   }
643
644   bool VisitTemplateDecl(const TemplateDecl *D) {
645     // FIXME: Template parameters.
646
647     // Index the default values for the template parameters.
648     const NamedDecl *Parent = D->getTemplatedDecl();
649     if (D->getTemplateParameters() &&
650         shouldIndexTemplateParameterDefaultValue(Parent)) {
651       const TemplateParameterList *Params = D->getTemplateParameters();
652       for (const NamedDecl *TP : *Params) {
653         if (const auto *TTP = dyn_cast<TemplateTypeParmDecl>(TP)) {
654           if (TTP->hasDefaultArgument())
655             IndexCtx.indexTypeSourceInfo(TTP->getDefaultArgumentInfo(), Parent);
656         } else if (const auto *NTTP = dyn_cast<NonTypeTemplateParmDecl>(TP)) {
657           if (NTTP->hasDefaultArgument())
658             IndexCtx.indexBody(NTTP->getDefaultArgument(), Parent);
659         } else if (const auto *TTPD = dyn_cast<TemplateTemplateParmDecl>(TP)) {
660           if (TTPD->hasDefaultArgument())
661             handleTemplateArgumentLoc(TTPD->getDefaultArgument(), Parent,
662                                       /*DC=*/nullptr);
663         }
664       }
665     }
666
667     return Visit(D->getTemplatedDecl());
668   }
669
670   bool VisitFriendDecl(const FriendDecl *D) {
671     if (auto ND = D->getFriendDecl()) {
672       // FIXME: Ignore a class template in a dependent context, these are not
673       // linked properly with their redeclarations, ending up with duplicate
674       // USRs.
675       // See comment "Friend templates are visible in fairly strange ways." in
676       // SemaTemplate.cpp which precedes code that prevents the friend template
677       // from becoming visible from the enclosing context.
678       if (isa<ClassTemplateDecl>(ND) && D->getDeclContext()->isDependentContext())
679         return true;
680       return Visit(ND);
681     }
682     if (auto Ty = D->getFriendType()) {
683       IndexCtx.indexTypeSourceInfo(Ty, cast<NamedDecl>(D->getDeclContext()));
684     }
685     return true;
686   }
687
688   bool VisitImportDecl(const ImportDecl *D) {
689     return IndexCtx.importedModule(D);
690   }
691
692   bool VisitStaticAssertDecl(const StaticAssertDecl *D) {
693     IndexCtx.indexBody(D->getAssertExpr(),
694                        dyn_cast<NamedDecl>(D->getDeclContext()),
695                        D->getLexicalDeclContext());
696     return true;
697   }
698 };
699
700 } // anonymous namespace
701
702 bool IndexingContext::indexDecl(const Decl *D) {
703   if (D->isImplicit() && shouldIgnoreIfImplicit(D))
704     return true;
705
706   if (isTemplateImplicitInstantiation(D))
707     return true;
708
709   IndexingDeclVisitor Visitor(*this);
710   bool ShouldContinue = Visitor.Visit(D);
711   if (!ShouldContinue)
712     return false;
713
714   if (!Visitor.Handled && isa<DeclContext>(D))
715     return indexDeclContext(cast<DeclContext>(D));
716
717   return true;
718 }
719
720 bool IndexingContext::indexDeclContext(const DeclContext *DC) {
721   for (const auto *I : DC->decls())
722     if (!indexDecl(I))
723       return false;
724   return true;
725 }
726
727 bool IndexingContext::indexTopLevelDecl(const Decl *D) {
728   if (D->getLocation().isInvalid())
729     return true;
730
731   if (isa<ObjCMethodDecl>(D))
732     return true; // Wait for the objc container.
733
734   return indexDecl(D);
735 }
736
737 bool IndexingContext::indexDeclGroupRef(DeclGroupRef DG) {
738   for (DeclGroupRef::iterator I = DG.begin(), E = DG.end(); I != E; ++I)
739     if (!indexTopLevelDecl(*I))
740       return false;
741   return true;
742 }