blob: 91fbbfb268432cd8faeaca3b130150460955b422 [file] [log] [blame]
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001//===- IndexingContext.cpp - Indexing context data ------------------------===//
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/ASTContext.h"
13#include "clang/AST/DeclTemplate.h"
14#include "clang/AST/DeclObjC.h"
15#include "clang/Basic/SourceManager.h"
16
17using namespace clang;
18using namespace index;
19
20bool IndexingContext::shouldIndexFunctionLocalSymbols() const {
21 return IndexOpts.IndexFunctionLocals;
22}
23
24bool IndexingContext::handleDecl(const Decl *D,
25 SymbolRoleSet Roles,
26 ArrayRef<SymbolRelation> Relations) {
27 return handleDeclOccurrence(D, D->getLocation(), /*IsRef=*/false,
28 cast<Decl>(D->getDeclContext()), Roles, Relations,
29 nullptr, nullptr, D->getDeclContext());
30}
31
32bool IndexingContext::handleDecl(const Decl *D, SourceLocation Loc,
33 SymbolRoleSet Roles,
34 ArrayRef<SymbolRelation> Relations,
35 const DeclContext *DC) {
36 if (!DC)
37 DC = D->getDeclContext();
38 return handleDeclOccurrence(D, Loc, /*IsRef=*/false, cast<Decl>(DC),
39 Roles, Relations,
40 nullptr, nullptr, DC);
41}
42
43bool IndexingContext::handleReference(const NamedDecl *D, SourceLocation Loc,
44 const NamedDecl *Parent,
45 const DeclContext *DC,
46 SymbolRoleSet Roles,
47 ArrayRef<SymbolRelation> Relations,
48 const Expr *RefE,
49 const Decl *RefD) {
50 if (!shouldIndexFunctionLocalSymbols() && isFunctionLocalDecl(D))
51 return true;
52
53 if (isa<NonTypeTemplateParmDecl>(D) || isa<TemplateTypeParmDecl>(D))
54 return true;
55
56 return handleDeclOccurrence(D, Loc, /*IsRef=*/true, Parent, Roles, Relations,
57 RefE, RefD, DC);
58}
59
60bool IndexingContext::importedModule(const ImportDecl *ImportD) {
61 SourceLocation Loc = ImportD->getLocation();
62 SourceManager &SM = Ctx->getSourceManager();
63 Loc = SM.getFileLoc(Loc);
64 if (Loc.isInvalid())
65 return true;
66
67 FileID FID;
68 unsigned Offset;
69 std::tie(FID, Offset) = SM.getDecomposedLoc(Loc);
70 if (FID.isInvalid())
71 return true;
72
73 bool Invalid = false;
74 const SrcMgr::SLocEntry &SEntry = SM.getSLocEntry(FID, &Invalid);
75 if (Invalid || !SEntry.isFile())
76 return true;
77
78 if (SEntry.getFile().getFileCharacteristic() != SrcMgr::C_User) {
79 switch (IndexOpts.SystemSymbolFilter) {
80 case IndexingOptions::SystemSymbolFilterKind::None:
81 case IndexingOptions::SystemSymbolFilterKind::DeclarationsOnly:
82 return true;
83 case IndexingOptions::SystemSymbolFilterKind::All:
84 break;
85 }
86 }
87
88 SymbolRoleSet Roles{};
89 if (ImportD->isImplicit())
90 Roles |= (unsigned)SymbolRole::Implicit;
91
92 return DataConsumer.handleModuleOccurence(ImportD, Roles, FID, Offset);
93}
94
95bool IndexingContext::isFunctionLocalDecl(const Decl *D) {
96 assert(D);
97
98 if (isa<TemplateTemplateParmDecl>(D))
99 return true;
100
101 if (!D->getParentFunctionOrMethod())
102 return false;
103
104 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
105 switch (ND->getFormalLinkage()) {
106 case NoLinkage:
107 case VisibleNoLinkage:
108 case InternalLinkage:
109 return true;
110 case UniqueExternalLinkage:
111 llvm_unreachable("Not a sema linkage");
112 case ExternalLinkage:
113 return false;
114 }
115 }
116
117 return true;
118}
119
120bool IndexingContext::isTemplateImplicitInstantiation(const Decl *D) {
121 TemplateSpecializationKind TKind = TSK_Undeclared;
122 if (const ClassTemplateSpecializationDecl *
123 SD = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
124 TKind = SD->getSpecializationKind();
125 }
126 if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
127 TKind = FD->getTemplateSpecializationKind();
128 }
129 switch (TKind) {
130 case TSK_Undeclared:
131 case TSK_ExplicitSpecialization:
132 return false;
133 case TSK_ImplicitInstantiation:
134 case TSK_ExplicitInstantiationDeclaration:
135 case TSK_ExplicitInstantiationDefinition:
136 return true;
137 }
138}
139
140bool IndexingContext::shouldIgnoreIfImplicit(const Decl *D) {
141 if (isa<ObjCInterfaceDecl>(D))
142 return false;
143 if (isa<ObjCCategoryDecl>(D))
144 return false;
145 if (isa<ObjCIvarDecl>(D))
146 return false;
147 if (isa<ObjCMethodDecl>(D))
148 return false;
149 if (isa<ImportDecl>(D))
150 return false;
151 return true;
152}
153
154static const Decl *adjustTemplateImplicitInstantiation(const Decl *D) {
155 if (const ClassTemplateSpecializationDecl *
156 SD = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
157 return SD->getTemplateInstantiationPattern();
158 }
159 if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
160 return FD->getTemplateInstantiationPattern();
161 }
162 return nullptr;
163}
164
165static bool isDeclADefinition(const Decl *D, ASTContext &Ctx) {
166 if (auto VD = dyn_cast<VarDecl>(D))
167 return VD->isThisDeclarationADefinition(Ctx);
168
169 if (auto FD = dyn_cast<FunctionDecl>(D))
170 return FD->isThisDeclarationADefinition();
171
172 if (auto TD = dyn_cast<TagDecl>(D))
173 return TD->isThisDeclarationADefinition();
174
175 if (auto MD = dyn_cast<ObjCMethodDecl>(D))
176 return MD->isThisDeclarationADefinition();
177
178 if (isa<TypedefNameDecl>(D) ||
179 isa<EnumConstantDecl>(D) ||
180 isa<FieldDecl>(D) ||
181 isa<MSPropertyDecl>(D) ||
182 isa<ObjCImplDecl>(D) ||
183 isa<ObjCPropertyImplDecl>(D))
184 return true;
185
186 return false;
187}
188
189static const Decl *adjustParent(const Decl *Parent) {
190 if (!Parent)
191 return nullptr;
192 for (;; Parent = cast<Decl>(Parent->getDeclContext())) {
193 if (isa<TranslationUnitDecl>(Parent))
194 return nullptr;
195 if (isa<LinkageSpecDecl>(Parent) || isa<BlockDecl>(Parent))
196 continue;
197 if (auto NS = dyn_cast<NamespaceDecl>(Parent)) {
198 if (NS->isAnonymousNamespace())
199 continue;
200 } else if (auto EnumD = dyn_cast<EnumDecl>(Parent)) {
201 // Move enumerators under anonymous enum to the enclosing parent.
202 if (EnumD->getDeclName().isEmpty())
203 continue;
204 } else if (auto RD = dyn_cast<RecordDecl>(Parent)) {
205 if (RD->isAnonymousStructOrUnion())
206 continue;
207 } else if (auto FD = dyn_cast<FieldDecl>(Parent)) {
208 if (FD->getDeclName().isEmpty())
209 continue;
210 }
211 return Parent;
212 }
213}
214
215static const Decl *getCanonicalDecl(const Decl *D) {
216 D = D->getCanonicalDecl();
217 if (auto TD = dyn_cast<TemplateDecl>(D)) {
218 D = TD->getTemplatedDecl();
219 assert(D->isCanonicalDecl());
220 }
221
222 return D;
223}
224
225bool IndexingContext::handleDeclOccurrence(const Decl *D, SourceLocation Loc,
226 bool IsRef, const Decl *Parent,
227 SymbolRoleSet Roles,
228 ArrayRef<SymbolRelation> Relations,
229 const Expr *OrigE,
230 const Decl *OrigD,
231 const DeclContext *ContainerDC) {
232 if (D->isImplicit() && !isa<ObjCMethodDecl>(D))
233 return true;
234 if (!isa<NamedDecl>(D) ||
235 (cast<NamedDecl>(D)->getDeclName().isEmpty() &&
236 !isa<TagDecl>(D) && !isa<ObjCCategoryDecl>(D)))
237 return true;
238
239 SourceManager &SM = Ctx->getSourceManager();
240 Loc = SM.getFileLoc(Loc);
241 if (Loc.isInvalid())
242 return true;
243
244 FileID FID;
245 unsigned Offset;
246 std::tie(FID, Offset) = SM.getDecomposedLoc(Loc);
247 if (FID.isInvalid())
248 return true;
249
250 bool Invalid = false;
251 const SrcMgr::SLocEntry &SEntry = SM.getSLocEntry(FID, &Invalid);
252 if (Invalid || !SEntry.isFile())
253 return true;
254
255 if (SEntry.getFile().getFileCharacteristic() != SrcMgr::C_User) {
256 switch (IndexOpts.SystemSymbolFilter) {
257 case IndexingOptions::SystemSymbolFilterKind::None:
258 return true;
259 case IndexingOptions::SystemSymbolFilterKind::DeclarationsOnly:
260 if (IsRef)
261 return true;
262 break;
263 case IndexingOptions::SystemSymbolFilterKind::All:
264 break;
265 }
266 }
267
268 if (isTemplateImplicitInstantiation(D)) {
269 if (!IsRef)
270 return true;
271 D = adjustTemplateImplicitInstantiation(D);
272 if (!D)
273 return true;
274 assert(!isTemplateImplicitInstantiation(D));
275 }
276
277 if (!OrigD)
278 OrigD = D;
279
280 if (IsRef)
281 Roles |= (unsigned)SymbolRole::Reference;
282 else if (isDeclADefinition(D, *Ctx))
283 Roles |= (unsigned)SymbolRole::Definition;
284 else
285 Roles |= (unsigned)SymbolRole::Declaration;
286
287 D = getCanonicalDecl(D);
288 if (D->isImplicit() && !isa<ObjCMethodDecl>(D)) {
289 // operator new declarations will link to the implicit one as canonical.
290 return true;
291 }
292 Parent = adjustParent(Parent);
293 if (Parent)
294 Parent = getCanonicalDecl(Parent);
295 assert(!Parent || !Parent->isImplicit() ||
296 isa<ObjCInterfaceDecl>(Parent) || isa<ObjCMethodDecl>(Parent));
297
298 SmallVector<SymbolRelation, 6> FinalRelations;
299 FinalRelations.reserve(Relations.size()+1);
300
301 auto addRelation = [&](SymbolRelation Rel) {
302 auto It = std::find_if(FinalRelations.begin(), FinalRelations.end(),
303 [&](SymbolRelation Elem)->bool {
304 return Elem.RelatedSymbol == Rel.RelatedSymbol;
305 });
306 if (It != FinalRelations.end()) {
307 It->Roles |= Rel.Roles;
308 } else {
309 FinalRelations.push_back(Rel);
310 }
311 Roles |= Rel.Roles;
312 };
313
314 if (!IsRef && Parent && !cast<DeclContext>(Parent)->isFunctionOrMethod()) {
315 addRelation(SymbolRelation{(unsigned)SymbolRole::RelationChildOf, Parent});
316 }
317 for (auto &Rel : Relations) {
318 addRelation(SymbolRelation(Rel.Roles,
319 Rel.RelatedSymbol->getCanonicalDecl()));
320 }
321
322 IndexDataConsumer::ASTNodeInfo Node{ OrigE, OrigD, Parent, ContainerDC };
323 return DataConsumer.handleDeclOccurence(D, Roles, FinalRelations, FID, Offset,
324 Node);
325}