blob: 616a0797f52b138622f4650d11c693fbf96ae74d [file] [log] [blame]
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001//===- CXIndexDataConsumer.cpp - Index data consumer for libclang----------===//
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00002//
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
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000010#include "CXIndexDataConsumer.h"
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +000011#include "CIndexDiagnostic.h"
Chandler Carruthcc0694c2012-12-04 09:25:21 +000012#include "CXTranslationUnit.h"
David Blaikie0a4e61f2013-09-13 18:32:52 +000013#include "clang/AST/Attr.h"
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +000014#include "clang/AST/DeclCXX.h"
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +000015#include "clang/AST/DeclTemplate.h"
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000016#include "clang/AST/DeclVisitor.h"
Chandler Carruthcc0694c2012-12-04 09:25:21 +000017#include "clang/Frontend/ASTUnit.h"
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +000018
19using namespace clang;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000020using namespace clang::index;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +000021using namespace cxindex;
22using namespace cxcursor;
23
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000024namespace {
25class IndexingDeclVisitor : public ConstDeclVisitor<IndexingDeclVisitor, bool> {
26 CXIndexDataConsumer &DataConsumer;
27 SourceLocation DeclLoc;
28 const DeclContext *LexicalDC;
29
30public:
31 IndexingDeclVisitor(CXIndexDataConsumer &dataConsumer, SourceLocation Loc,
32 const DeclContext *lexicalDC)
33 : DataConsumer(dataConsumer), DeclLoc(Loc), LexicalDC(lexicalDC) { }
34
35 bool VisitFunctionDecl(const FunctionDecl *D) {
36 DataConsumer.handleFunction(D);
37 return true;
38 }
39
40 bool VisitVarDecl(const VarDecl *D) {
41 DataConsumer.handleVar(D);
42 return true;
43 }
44
45 bool VisitFieldDecl(const FieldDecl *D) {
46 DataConsumer.handleField(D);
47 return true;
48 }
49
50 bool VisitMSPropertyDecl(const MSPropertyDecl *D) {
51 return true;
52 }
53
54 bool VisitEnumConstantDecl(const EnumConstantDecl *D) {
55 DataConsumer.handleEnumerator(D);
56 return true;
57 }
58
59 bool VisitTypedefNameDecl(const TypedefNameDecl *D) {
60 DataConsumer.handleTypedefName(D);
61 return true;
62 }
63
64 bool VisitTagDecl(const TagDecl *D) {
65 DataConsumer.handleTagDecl(D);
66 return true;
67 }
68
69 bool VisitObjCInterfaceDecl(const ObjCInterfaceDecl *D) {
70 DataConsumer.handleObjCInterface(D);
71 return true;
72 }
73
74 bool VisitObjCProtocolDecl(const ObjCProtocolDecl *D) {
75 DataConsumer.handleObjCProtocol(D);
76 return true;
77 }
78
79 bool VisitObjCImplementationDecl(const ObjCImplementationDecl *D) {
80 DataConsumer.handleObjCImplementation(D);
81 return true;
82 }
83
84 bool VisitObjCCategoryDecl(const ObjCCategoryDecl *D) {
85 DataConsumer.handleObjCCategory(D);
86 return true;
87 }
88
89 bool VisitObjCCategoryImplDecl(const ObjCCategoryImplDecl *D) {
90 DataConsumer.handleObjCCategoryImpl(D);
91 return true;
92 }
93
94 bool VisitObjCMethodDecl(const ObjCMethodDecl *D) {
Argyrios Kyrtzidis66c49f72016-03-31 20:18:22 +000095 if (isa<ObjCImplDecl>(LexicalDC) && !D->isThisDeclarationADefinition())
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000096 DataConsumer.handleSynthesizedObjCMethod(D, DeclLoc, LexicalDC);
97 else
Argyrios Kyrtzidisaee15fb2017-01-26 02:11:50 +000098 DataConsumer.handleObjCMethod(D, DeclLoc);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +000099 return true;
100 }
101
102 bool VisitObjCPropertyDecl(const ObjCPropertyDecl *D) {
103 DataConsumer.handleObjCProperty(D);
104 return true;
105 }
106
107 bool VisitObjCPropertyImplDecl(const ObjCPropertyImplDecl *D) {
108 DataConsumer.handleSynthesizedObjCProperty(D);
109 return true;
110 }
111
112 bool VisitNamespaceDecl(const NamespaceDecl *D) {
113 DataConsumer.handleNamespace(D);
114 return true;
115 }
116
117 bool VisitUsingDecl(const UsingDecl *D) {
118 return true;
119 }
120
121 bool VisitUsingDirectiveDecl(const UsingDirectiveDecl *D) {
122 return true;
123 }
124
125 bool VisitClassTemplateDecl(const ClassTemplateDecl *D) {
126 DataConsumer.handleClassTemplate(D);
127 return true;
128 }
129
130 bool VisitClassTemplateSpecializationDecl(const
131 ClassTemplateSpecializationDecl *D) {
132 DataConsumer.handleTagDecl(D);
133 return true;
134 }
135
136 bool VisitFunctionTemplateDecl(const FunctionTemplateDecl *D) {
137 DataConsumer.handleFunctionTemplate(D);
138 return true;
139 }
140
141 bool VisitTypeAliasTemplateDecl(const TypeAliasTemplateDecl *D) {
142 DataConsumer.handleTypeAliasTemplate(D);
143 return true;
144 }
145
146 bool VisitImportDecl(const ImportDecl *D) {
147 DataConsumer.importedModule(D);
148 return true;
149 }
150};
Fangrui Song31b97192018-02-12 17:42:09 +0000151
152CXSymbolRole getSymbolRole(SymbolRoleSet Role) {
153 // CXSymbolRole mirrors low 9 bits of clang::index::SymbolRole.
154 return CXSymbolRole(static_cast<uint32_t>(Role) & ((1 << 9) - 1));
155}
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000156}
157
Sam McCallcc026eb2018-04-09 14:12:51 +0000158bool CXIndexDataConsumer::handleDeclOccurence(
159 const Decl *D, SymbolRoleSet Roles, ArrayRef<SymbolRelation> Relations,
160 SourceLocation Loc, ASTNodeInfo ASTNode) {
161 Loc = getASTContext().getSourceManager().getFileLoc(Loc);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000162
163 if (Roles & (unsigned)SymbolRole::Reference) {
164 const NamedDecl *ND = dyn_cast<NamedDecl>(D);
165 if (!ND)
166 return true;
167
168 if (auto *ObjCID = dyn_cast_or_null<ObjCInterfaceDecl>(ASTNode.OrigD)) {
169 if (!ObjCID->isThisDeclarationADefinition() &&
170 ObjCID->getLocation() == Loc) {
171 // The libclang API treats this as ObjCClassRef declaration.
172 IndexingDeclVisitor(*this, Loc, nullptr).Visit(ObjCID);
173 return true;
174 }
175 }
Argyrios Kyrtzidis542f38f2016-03-09 02:12:46 +0000176 if (auto *ObjCPD = dyn_cast_or_null<ObjCProtocolDecl>(ASTNode.OrigD)) {
177 if (!ObjCPD->isThisDeclarationADefinition() &&
178 ObjCPD->getLocation() == Loc) {
179 // The libclang API treats this as ObjCProtocolRef declaration.
180 IndexingDeclVisitor(*this, Loc, nullptr).Visit(ObjCPD);
181 return true;
182 }
183 }
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000184
185 CXIdxEntityRefKind Kind = CXIdxEntityRef_Direct;
186 if (Roles & (unsigned)SymbolRole::Implicit) {
187 Kind = CXIdxEntityRef_Implicit;
188 }
Fangrui Song31b97192018-02-12 17:42:09 +0000189 CXSymbolRole CXRole = getSymbolRole(Roles);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000190
191 CXCursor Cursor;
192 if (ASTNode.OrigE) {
193 Cursor = cxcursor::MakeCXCursor(ASTNode.OrigE,
194 cast<Decl>(ASTNode.ContainerDC),
195 getCXTU());
196 } else {
Argyrios Kyrtzidis66c49f72016-03-31 20:18:22 +0000197 if (ASTNode.OrigD) {
198 if (auto *OrigND = dyn_cast<NamedDecl>(ASTNode.OrigD))
199 Cursor = getRefCursor(OrigND, Loc);
200 else
201 Cursor = MakeCXCursor(ASTNode.OrigD, CXTU);
202 } else {
203 Cursor = getRefCursor(ND, Loc);
204 }
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000205 }
206 handleReference(ND, Loc, Cursor,
207 dyn_cast_or_null<NamedDecl>(ASTNode.Parent),
Fangrui Song31b97192018-02-12 17:42:09 +0000208 ASTNode.ContainerDC, ASTNode.OrigE, Kind, CXRole);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000209
210 } else {
Argyrios Kyrtzidis66c49f72016-03-31 20:18:22 +0000211 const DeclContext *LexicalDC = ASTNode.ContainerDC;
212 if (!LexicalDC) {
213 for (const auto &SymRel : Relations) {
214 if (SymRel.Roles & (unsigned)SymbolRole::RelationChildOf)
215 LexicalDC = dyn_cast<DeclContext>(SymRel.RelatedSymbol);
216 }
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000217 }
Argyrios Kyrtzidis66c49f72016-03-31 20:18:22 +0000218 IndexingDeclVisitor(*this, Loc, LexicalDC).Visit(ASTNode.OrigD);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000219 }
220
221 return !shouldAbort();
222}
223
224bool CXIndexDataConsumer::handleModuleOccurence(const ImportDecl *ImportD,
225 SymbolRoleSet Roles,
Sam McCallcc026eb2018-04-09 14:12:51 +0000226 SourceLocation Loc) {
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000227 IndexingDeclVisitor(*this, SourceLocation(), nullptr).Visit(ImportD);
228 return !shouldAbort();
229}
230
231void CXIndexDataConsumer::finish() {
232 indexDiagnostics();
233}
234
235
236CXIndexDataConsumer::ObjCProtocolListInfo::ObjCProtocolListInfo(
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000237 const ObjCProtocolList &ProtList,
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000238 CXIndexDataConsumer &IdxCtx,
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000239 ScratchAlloc &SA) {
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000240 ObjCInterfaceDecl::protocol_loc_iterator LI = ProtList.loc_begin();
241 for (ObjCInterfaceDecl::protocol_iterator
242 I = ProtList.begin(), E = ProtList.end(); I != E; ++I, ++LI) {
243 SourceLocation Loc = *LI;
244 ObjCProtocolDecl *PD = *I;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000245 ProtEntities.push_back(EntityInfo());
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000246 IdxCtx.getEntityInfo(PD, ProtEntities.back(), SA);
Craig Topper69186e72014-06-08 08:38:04 +0000247 CXIdxObjCProtocolRefInfo ProtInfo = { nullptr,
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000248 MakeCursorObjCProtocolRef(PD, Loc, IdxCtx.CXTU),
249 IdxCtx.getIndexLoc(Loc) };
250 ProtInfos.push_back(ProtInfo);
Argyrios Kyrtzidis41cfce22011-12-15 00:04:56 +0000251
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000252 if (IdxCtx.shouldSuppressRefs())
Argyrios Kyrtzidis41cfce22011-12-15 00:04:56 +0000253 IdxCtx.markEntityOccurrenceInFile(PD, Loc);
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000254 }
255
256 for (unsigned i = 0, e = ProtInfos.size(); i != e; ++i)
257 ProtInfos[i].protocol = &ProtEntities[i];
258
259 for (unsigned i = 0, e = ProtInfos.size(); i != e; ++i)
260 Prots.push_back(&ProtInfos[i]);
261}
262
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000263
264IBOutletCollectionInfo::IBOutletCollectionInfo(
265 const IBOutletCollectionInfo &other)
Argyrios Kyrtzidis908c4dc2012-01-20 01:38:51 +0000266 : AttrInfo(CXIdxAttr_IBOutletCollection, other.cursor, other.loc, other.A) {
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000267
268 IBCollInfo.attrInfo = this;
269 IBCollInfo.classCursor = other.IBCollInfo.classCursor;
270 IBCollInfo.classLoc = other.IBCollInfo.classLoc;
271 if (other.IBCollInfo.objcClass) {
272 ClassInfo = other.ClassInfo;
273 IBCollInfo.objcClass = &ClassInfo;
274 } else
Craig Topper69186e72014-06-08 08:38:04 +0000275 IBCollInfo.objcClass = nullptr;
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000276}
277
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000278AttrListInfo::AttrListInfo(const Decl *D, CXIndexDataConsumer &IdxCtx)
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000279 : SA(IdxCtx), ref_cnt(0) {
280
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000281 if (!D->hasAttrs())
282 return;
283
Aaron Ballmanb97112e2014-03-08 22:19:01 +0000284 for (const auto *A : D->attrs()) {
Dmitri Gribenko9c256e32013-01-14 00:46:27 +0000285 CXCursor C = MakeCXCursor(A, D, IdxCtx.CXTU);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000286 CXIdxLoc Loc = IdxCtx.getIndexLoc(A->getLocation());
287 switch (C.kind) {
288 default:
289 Attrs.push_back(AttrInfo(CXIdxAttr_Unexposed, C, Loc, A));
290 break;
291 case CXCursor_IBActionAttr:
292 Attrs.push_back(AttrInfo(CXIdxAttr_IBAction, C, Loc, A));
293 break;
294 case CXCursor_IBOutletAttr:
295 Attrs.push_back(AttrInfo(CXIdxAttr_IBOutlet, C, Loc, A));
296 break;
297 case CXCursor_IBOutletCollectionAttr:
298 IBCollAttrs.push_back(IBOutletCollectionInfo(C, Loc, A));
299 break;
300 }
301 }
302
303 for (unsigned i = 0, e = IBCollAttrs.size(); i != e; ++i) {
304 IBOutletCollectionInfo &IBInfo = IBCollAttrs[i];
305 CXAttrs.push_back(&IBInfo);
306
307 const IBOutletCollectionAttr *
308 IBAttr = cast<IBOutletCollectionAttr>(IBInfo.A);
Richard Smithb87c4652013-10-31 21:23:20 +0000309 SourceLocation InterfaceLocStart =
310 IBAttr->getInterfaceLoc()->getTypeLoc().getLocStart();
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000311 IBInfo.IBCollInfo.attrInfo = &IBInfo;
Richard Smithb87c4652013-10-31 21:23:20 +0000312 IBInfo.IBCollInfo.classLoc = IdxCtx.getIndexLoc(InterfaceLocStart);
Craig Topper69186e72014-06-08 08:38:04 +0000313 IBInfo.IBCollInfo.objcClass = nullptr;
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000314 IBInfo.IBCollInfo.classCursor = clang_getNullCursor();
315 QualType Ty = IBAttr->getInterface();
Richard Smithb1f9a282013-10-31 01:56:18 +0000316 if (const ObjCObjectType *ObjectTy = Ty->getAs<ObjCObjectType>()) {
317 if (const ObjCInterfaceDecl *InterD = ObjectTy->getInterface()) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000318 IdxCtx.getEntityInfo(InterD, IBInfo.ClassInfo, SA);
319 IBInfo.IBCollInfo.objcClass = &IBInfo.ClassInfo;
Richard Smithb87c4652013-10-31 21:23:20 +0000320 IBInfo.IBCollInfo.classCursor =
321 MakeCursorObjCClassRef(InterD, InterfaceLocStart, IdxCtx.CXTU);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000322 }
323 }
324 }
325
326 for (unsigned i = 0, e = Attrs.size(); i != e; ++i)
327 CXAttrs.push_back(&Attrs[i]);
328}
329
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000330IntrusiveRefCntPtr<AttrListInfo>
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000331AttrListInfo::create(const Decl *D, CXIndexDataConsumer &IdxCtx) {
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000332 ScratchAlloc SA(IdxCtx);
333 AttrListInfo *attrs = SA.allocate<AttrListInfo>();
334 return new (attrs) AttrListInfo(D, IdxCtx);
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000335}
336
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000337CXIndexDataConsumer::CXXBasesListInfo::CXXBasesListInfo(const CXXRecordDecl *D,
338 CXIndexDataConsumer &IdxCtx,
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000339 ScratchAlloc &SA) {
Aaron Ballman574705e2014-03-13 15:41:46 +0000340 for (const auto &Base : D->bases()) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000341 BaseEntities.push_back(EntityInfo());
Craig Topper69186e72014-06-08 08:38:04 +0000342 const NamedDecl *BaseD = nullptr;
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000343 QualType T = Base.getType();
344 SourceLocation Loc = getBaseLoc(Base);
345
346 if (const TypedefType *TDT = T->getAs<TypedefType>()) {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000347 BaseD = TDT->getDecl();
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000348 } else if (const TemplateSpecializationType *
349 TST = T->getAs<TemplateSpecializationType>()) {
350 BaseD = TST->getTemplateName().getAsTemplateDecl();
351 } else if (const RecordType *RT = T->getAs<RecordType>()) {
352 BaseD = RT->getDecl();
353 }
354
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000355 if (BaseD)
356 IdxCtx.getEntityInfo(BaseD, BaseEntities.back(), SA);
Craig Topper69186e72014-06-08 08:38:04 +0000357 CXIdxBaseClassInfo BaseInfo = { nullptr,
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000358 MakeCursorCXXBaseSpecifier(&Base, IdxCtx.CXTU),
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000359 IdxCtx.getIndexLoc(Loc) };
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000360 BaseInfos.push_back(BaseInfo);
361 }
362
363 for (unsigned i = 0, e = BaseInfos.size(); i != e; ++i) {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000364 if (BaseEntities[i].name && BaseEntities[i].USR)
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000365 BaseInfos[i].base = &BaseEntities[i];
366 }
367
368 for (unsigned i = 0, e = BaseInfos.size(); i != e; ++i)
369 CXBases.push_back(&BaseInfos[i]);
370}
371
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000372SourceLocation CXIndexDataConsumer::CXXBasesListInfo::getBaseLoc(
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000373 const CXXBaseSpecifier &Base) const {
374 SourceLocation Loc = Base.getSourceRange().getBegin();
375 TypeLoc TL;
376 if (Base.getTypeSourceInfo())
377 TL = Base.getTypeSourceInfo()->getTypeLoc();
378 if (TL.isNull())
379 return Loc;
380
David Blaikie6adc78e2013-02-18 22:06:02 +0000381 if (QualifiedTypeLoc QL = TL.getAs<QualifiedTypeLoc>())
382 TL = QL.getUnqualifiedLoc();
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000383
David Blaikie6adc78e2013-02-18 22:06:02 +0000384 if (ElaboratedTypeLoc EL = TL.getAs<ElaboratedTypeLoc>())
385 return EL.getNamedTypeLoc().getBeginLoc();
386 if (DependentNameTypeLoc DL = TL.getAs<DependentNameTypeLoc>())
387 return DL.getNameLoc();
388 if (DependentTemplateSpecializationTypeLoc DTL =
389 TL.getAs<DependentTemplateSpecializationTypeLoc>())
390 return DTL.getTemplateNameLoc();
Argyrios Kyrtzidisb3c16ba2011-12-07 20:44:15 +0000391
392 return Loc;
393}
394
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000395const char *ScratchAlloc::toCStr(StringRef Str) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000396 if (Str.empty())
397 return "";
398 if (Str.data()[Str.size()] == '\0')
399 return Str.data();
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000400 return copyCStr(Str);
401}
402
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000403const char *ScratchAlloc::copyCStr(StringRef Str) {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000404 char *buf = IdxCtx.StrScratch.Allocate<char>(Str.size() + 1);
405 std::uninitialized_copy(Str.begin(), Str.end(), buf);
406 buf[Str.size()] = '\0';
407 return buf;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000408}
409
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000410void CXIndexDataConsumer::setASTContext(ASTContext &ctx) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000411 Ctx = &ctx;
Dmitri Gribenkoc22ea1c2013-01-26 18:53:38 +0000412 cxtu::getASTUnit(CXTU)->setASTContext(&ctx);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000413}
414
David Blaikie41565462017-01-05 19:48:07 +0000415void CXIndexDataConsumer::setPreprocessor(std::shared_ptr<Preprocessor> PP) {
416 cxtu::getASTUnit(CXTU)->setPreprocessor(std::move(PP));
Argyrios Kyrtzidisda6e0542012-01-17 18:48:07 +0000417}
418
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000419bool CXIndexDataConsumer::isFunctionLocalDecl(const Decl *D) {
Argyrios Kyrtzidis68e87e12012-09-10 22:58:04 +0000420 assert(D);
421
422 if (!D->getParentFunctionOrMethod())
423 return false;
424
425 if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
Rafael Espindola3ae00052013-05-13 00:12:11 +0000426 switch (ND->getFormalLinkage()) {
Argyrios Kyrtzidis68e87e12012-09-10 22:58:04 +0000427 case NoLinkage:
428 case InternalLinkage:
429 return true;
Richard Smithaf10ea22017-07-08 00:37:59 +0000430 case VisibleNoLinkage:
431 case ModuleInternalLinkage:
Argyrios Kyrtzidis68e87e12012-09-10 22:58:04 +0000432 case UniqueExternalLinkage:
Rafael Espindola3ae00052013-05-13 00:12:11 +0000433 llvm_unreachable("Not a sema linkage");
Richard Smithaf10ea22017-07-08 00:37:59 +0000434 case ModuleLinkage:
Argyrios Kyrtzidis68e87e12012-09-10 22:58:04 +0000435 case ExternalLinkage:
436 return false;
437 }
438 }
439
440 return true;
441}
442
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000443bool CXIndexDataConsumer::shouldAbort() {
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000444 if (!CB.abortQuery)
445 return false;
Craig Topper69186e72014-06-08 08:38:04 +0000446 return CB.abortQuery(ClientData, nullptr);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000447}
448
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000449void CXIndexDataConsumer::enteredMainFile(const FileEntry *File) {
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000450 if (File && CB.enteredMainFile) {
David Greenee6c9fe02013-01-15 22:09:51 +0000451 CXIdxClientFile idxFile =
452 CB.enteredMainFile(ClientData,
Craig Topper69186e72014-06-08 08:38:04 +0000453 static_cast<CXFile>(const_cast<FileEntry *>(File)),
454 nullptr);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000455 FileMap[File] = idxFile;
456 }
457}
458
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000459void CXIndexDataConsumer::ppIncludedFile(SourceLocation hashLoc,
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000460 StringRef filename,
461 const FileEntry *File,
Argyrios Kyrtzidis5e2ec482012-10-18 00:17:05 +0000462 bool isImport, bool isAngled,
463 bool isModuleImport) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000464 if (!CB.ppIncludedFile)
465 return;
466
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000467 ScratchAlloc SA(*this);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000468 CXIdxIncludedFileInfo Info = { getIndexLoc(hashLoc),
469 SA.toCStr(filename),
David Greenee6c9fe02013-01-15 22:09:51 +0000470 static_cast<CXFile>(
471 const_cast<FileEntry *>(File)),
Argyrios Kyrtzidis5e2ec482012-10-18 00:17:05 +0000472 isImport, isAngled, isModuleImport };
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000473 CXIdxClientFile idxFile = CB.ppIncludedFile(ClientData, &Info);
474 FileMap[File] = idxFile;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000475}
476
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000477void CXIndexDataConsumer::importedModule(const ImportDecl *ImportD) {
Argyrios Kyrtzidis472eda02012-10-02 16:10:38 +0000478 if (!CB.importedASTFile)
479 return;
480
Argyrios Kyrtzidis184b1442012-10-03 21:05:44 +0000481 Module *Mod = ImportD->getImportedModule();
482 if (!Mod)
483 return;
Argyrios Kyrtzidis472eda02012-10-02 16:10:38 +0000484
Richard Smith30fc9a92016-09-01 20:15:25 +0000485 // If the imported module is part of the top-level module that we're
486 // indexing, it doesn't correspond to an imported AST file.
487 // FIXME: This assumes that AST files and top-level modules directly
488 // correspond, which is unlikely to remain true forever.
489 if (Module *SrcMod = ImportD->getImportedOwningModule())
490 if (SrcMod->getTopLevelModule() == Mod->getTopLevelModule())
491 return;
492
Argyrios Kyrtzidis472eda02012-10-02 16:10:38 +0000493 CXIdxImportedASTFileInfo Info = {
David Greenee6c9fe02013-01-15 22:09:51 +0000494 static_cast<CXFile>(
495 const_cast<FileEntry *>(Mod->getASTFile())),
Argyrios Kyrtzidisdc78f3e2012-10-05 00:22:40 +0000496 Mod,
Argyrios Kyrtzidis184b1442012-10-03 21:05:44 +0000497 getIndexLoc(ImportD->getLocation()),
Argyrios Kyrtzidisdc78f3e2012-10-05 00:22:40 +0000498 ImportD->isImplicit()
Argyrios Kyrtzidis472eda02012-10-02 16:10:38 +0000499 };
500 CXIdxClientASTFile astFile = CB.importedASTFile(ClientData, &Info);
501 (void)astFile;
502}
503
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000504void CXIndexDataConsumer::importedPCH(const FileEntry *File) {
Argyrios Kyrtzidisf484b132012-10-03 21:05:51 +0000505 if (!CB.importedASTFile)
506 return;
507
508 CXIdxImportedASTFileInfo Info = {
David Greenee6c9fe02013-01-15 22:09:51 +0000509 static_cast<CXFile>(
510 const_cast<FileEntry *>(File)),
Craig Topper69186e72014-06-08 08:38:04 +0000511 /*module=*/nullptr,
Argyrios Kyrtzidisf484b132012-10-03 21:05:51 +0000512 getIndexLoc(SourceLocation()),
Argyrios Kyrtzidisdc78f3e2012-10-05 00:22:40 +0000513 /*isImplicit=*/false
Argyrios Kyrtzidisf484b132012-10-03 21:05:51 +0000514 };
515 CXIdxClientASTFile astFile = CB.importedASTFile(ClientData, &Info);
516 (void)astFile;
517}
518
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000519void CXIndexDataConsumer::startedTranslationUnit() {
Craig Topper69186e72014-06-08 08:38:04 +0000520 CXIdxClientContainer idxCont = nullptr;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000521 if (CB.startedTranslationUnit)
Craig Topper69186e72014-06-08 08:38:04 +0000522 idxCont = CB.startedTranslationUnit(ClientData, nullptr);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000523 addContainerInMap(Ctx->getTranslationUnitDecl(), idxCont);
524}
525
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000526void CXIndexDataConsumer::indexDiagnostics() {
527 if (!hasDiagnosticCallback())
528 return;
529
530 CXDiagnosticSetImpl *DiagSet = cxdiag::lazyCreateDiags(getCXTU());
531 handleDiagnosticSet(DiagSet);
532}
533
534void CXIndexDataConsumer::handleDiagnosticSet(CXDiagnostic CXDiagSet) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000535 if (!CB.diagnostic)
536 return;
537
Craig Topper69186e72014-06-08 08:38:04 +0000538 CB.diagnostic(ClientData, CXDiagSet, nullptr);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000539}
540
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000541bool CXIndexDataConsumer::handleDecl(const NamedDecl *D,
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000542 SourceLocation Loc, CXCursor Cursor,
Argyrios Kyrtzidis34608802012-02-28 17:50:39 +0000543 DeclInfo &DInfo,
Argyrios Kyrtzidisaf75b192016-02-09 19:07:07 +0000544 const DeclContext *LexicalDC,
545 const DeclContext *SemaDC) {
Argyrios Kyrtzidis1cff7952011-11-18 00:26:46 +0000546 if (!CB.indexDeclaration || !D)
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000547 return false;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000548 if (D->isImplicit() && shouldIgnoreIfImplicit(D))
549 return false;
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000550
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000551 ScratchAlloc SA(*this);
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000552 getEntityInfo(D, DInfo.EntInfo, SA);
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000553 if ((!shouldIndexFunctionLocalSymbols() && !DInfo.EntInfo.USR)
Argyrios Kyrtzidis9ef94862012-01-14 02:05:51 +0000554 || Loc.isInvalid())
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000555 return false;
Argyrios Kyrtzidis1cff7952011-11-18 00:26:46 +0000556
Argyrios Kyrtzidis34608802012-02-28 17:50:39 +0000557 if (!LexicalDC)
558 LexicalDC = D->getLexicalDeclContext();
559
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000560 if (shouldSuppressRefs())
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +0000561 markEntityOccurrenceInFile(D, Loc);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000562
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000563 DInfo.entityInfo = &DInfo.EntInfo;
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000564 DInfo.cursor = Cursor;
565 DInfo.loc = getIndexLoc(Loc);
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000566 DInfo.isImplicit = D->isImplicit();
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000567
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000568 DInfo.attributes = DInfo.EntInfo.attributes;
569 DInfo.numAttributes = DInfo.EntInfo.numAttributes;
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000570
Argyrios Kyrtzidisaf75b192016-02-09 19:07:07 +0000571 if (!SemaDC)
572 SemaDC = D->getDeclContext();
573 getContainerInfo(SemaDC, DInfo.SemanticContainer);
Argyrios Kyrtzidis663c8ec2011-12-07 20:44:19 +0000574 DInfo.semanticContainer = &DInfo.SemanticContainer;
Argyrios Kyrtzidise5dc5b32012-02-10 20:10:44 +0000575
Argyrios Kyrtzidisaf75b192016-02-09 19:07:07 +0000576 if (LexicalDC == SemaDC) {
Argyrios Kyrtzidise5dc5b32012-02-10 20:10:44 +0000577 DInfo.lexicalContainer = &DInfo.SemanticContainer;
578 } else if (isTemplateImplicitInstantiation(D)) {
579 // Implicit instantiations have the lexical context of where they were
580 // instantiated first. We choose instead the semantic context because:
581 // 1) at the time that we see the instantiation we have not seen the
582 // function where it occurred yet.
583 // 2) the lexical context of the first instantiation is not useful
584 // information anyway.
585 DInfo.lexicalContainer = &DInfo.SemanticContainer;
586 } else {
Argyrios Kyrtzidis34608802012-02-28 17:50:39 +0000587 getContainerInfo(LexicalDC, DInfo.LexicalContainer);
Argyrios Kyrtzidise5dc5b32012-02-10 20:10:44 +0000588 DInfo.lexicalContainer = &DInfo.LexicalContainer;
589 }
590
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000591 if (DInfo.isContainer) {
592 getContainerInfo(getEntityContainer(D), DInfo.DeclAsContainer);
593 DInfo.declAsContainer = &DInfo.DeclAsContainer;
594 }
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000595
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000596 CB.indexDeclaration(ClientData, &DInfo);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000597 return true;
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000598}
599
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000600bool CXIndexDataConsumer::handleObjCContainer(const ObjCContainerDecl *D,
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000601 SourceLocation Loc, CXCursor Cursor,
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000602 ObjCContainerDeclInfo &ContDInfo) {
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000603 ContDInfo.ObjCContDeclInfo.declInfo = &ContDInfo;
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000604 return handleDecl(D, Loc, Cursor, ContDInfo);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000605}
606
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000607bool CXIndexDataConsumer::handleFunction(const FunctionDecl *D) {
Argyrios Kyrtzidis8b71bc72012-12-06 19:41:16 +0000608 bool isDef = D->isThisDeclarationADefinition();
609 bool isContainer = isDef;
610 bool isSkipped = false;
611 if (D->hasSkippedBody()) {
612 isSkipped = true;
613 isDef = true;
614 isContainer = false;
615 }
616
Rafael Espindola8db352d2013-10-17 15:37:26 +0000617 DeclInfo DInfo(!D->isFirstDecl(), isDef, isContainer);
Argyrios Kyrtzidis8b71bc72012-12-06 19:41:16 +0000618 if (isSkipped)
619 DInfo.flags |= CXIdxDeclFlag_Skipped;
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000620 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000621}
622
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000623bool CXIndexDataConsumer::handleVar(const VarDecl *D) {
Rafael Espindola8db352d2013-10-17 15:37:26 +0000624 DeclInfo DInfo(!D->isFirstDecl(), D->isThisDeclarationADefinition(),
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000625 /*isContainer=*/false);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000626 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000627}
628
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000629bool CXIndexDataConsumer::handleField(const FieldDecl *D) {
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000630 DeclInfo DInfo(/*isRedeclaration=*/false, /*isDefinition=*/true,
631 /*isContainer=*/false);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000632 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000633}
634
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000635bool CXIndexDataConsumer::handleMSProperty(const MSPropertyDecl *D) {
John McCall5e77d762013-04-16 07:28:30 +0000636 DeclInfo DInfo(/*isRedeclaration=*/false, /*isDefinition=*/true,
637 /*isContainer=*/false);
638 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
639}
640
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000641bool CXIndexDataConsumer::handleEnumerator(const EnumConstantDecl *D) {
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000642 DeclInfo DInfo(/*isRedeclaration=*/false, /*isDefinition=*/true,
643 /*isContainer=*/false);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000644 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000645}
646
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000647bool CXIndexDataConsumer::handleTagDecl(const TagDecl *D) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000648 if (const CXXRecordDecl *CXXRD = dyn_cast<CXXRecordDecl>(D))
649 return handleCXXRecordDecl(CXXRD, D);
650
Rafael Espindola8db352d2013-10-17 15:37:26 +0000651 DeclInfo DInfo(!D->isFirstDecl(), D->isThisDeclarationADefinition(),
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000652 D->isThisDeclarationADefinition());
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000653 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000654}
655
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000656bool CXIndexDataConsumer::handleTypedefName(const TypedefNameDecl *D) {
Rafael Espindola8db352d2013-10-17 15:37:26 +0000657 DeclInfo DInfo(!D->isFirstDecl(), /*isDefinition=*/true,
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000658 /*isContainer=*/false);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000659 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000660}
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000661
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000662bool CXIndexDataConsumer::handleObjCInterface(const ObjCInterfaceDecl *D) {
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +0000663 // For @class forward declarations, suppress them the same way as references.
Douglas Gregordeafd0b2011-12-27 22:43:10 +0000664 if (!D->isThisDeclarationADefinition()) {
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000665 if (shouldSuppressRefs() && markEntityOccurrenceInFile(D, D->getLocation()))
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +0000666 return false; // already occurred.
Douglas Gregordeafd0b2011-12-27 22:43:10 +0000667
668 // FIXME: This seems like the wrong definition for redeclaration.
Douglas Gregorec9fd132012-01-14 16:38:05 +0000669 bool isRedeclaration = D->hasDefinition() || D->getPreviousDecl();
Douglas Gregordeafd0b2011-12-27 22:43:10 +0000670 ObjCContainerDeclInfo ContDInfo(/*isForwardRef=*/true, isRedeclaration,
671 /*isImplementation=*/false);
672 return handleObjCContainer(D, D->getLocation(),
673 MakeCursorObjCClassRef(D, D->getLocation(),
674 CXTU),
675 ContDInfo);
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +0000676 }
677
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000678 ScratchAlloc SA(*this);
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000679
680 CXIdxBaseClassInfo BaseClass;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000681 EntityInfo BaseEntity;
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000682 BaseClass.cursor = clang_getNullCursor();
683 if (ObjCInterfaceDecl *SuperD = D->getSuperClass()) {
684 getEntityInfo(SuperD, BaseEntity, SA);
685 SourceLocation SuperLoc = D->getSuperClassLoc();
686 BaseClass.base = &BaseEntity;
687 BaseClass.cursor = MakeCursorObjCSuperClassRef(SuperD, SuperLoc, CXTU);
688 BaseClass.loc = getIndexLoc(SuperLoc);
Argyrios Kyrtzidis41cfce22011-12-15 00:04:56 +0000689
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000690 if (shouldSuppressRefs())
Argyrios Kyrtzidis41cfce22011-12-15 00:04:56 +0000691 markEntityOccurrenceInFile(SuperD, SuperLoc);
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000692 }
693
Douglas Gregorc0ac7d62011-12-15 05:27:12 +0000694 ObjCProtocolList EmptyProtoList;
Douglas Gregore6e48b12012-01-01 19:29:29 +0000695 ObjCProtocolListInfo ProtInfo(D->isThisDeclarationADefinition()
696 ? D->getReferencedProtocols()
697 : EmptyProtoList,
Douglas Gregorc0ac7d62011-12-15 05:27:12 +0000698 *this, SA);
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000699
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000700 ObjCInterfaceDeclInfo InterInfo(D);
701 InterInfo.ObjCProtoListInfo = ProtInfo.getListInfo();
702 InterInfo.ObjCInterDeclInfo.containerInfo = &InterInfo.ObjCContDeclInfo;
Craig Topper69186e72014-06-08 08:38:04 +0000703 InterInfo.ObjCInterDeclInfo.superInfo = D->getSuperClass() ? &BaseClass
704 : nullptr;
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000705 InterInfo.ObjCInterDeclInfo.protocols = &InterInfo.ObjCProtoListInfo;
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000706
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000707 return handleObjCContainer(D, D->getLocation(), getCursor(D), InterInfo);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000708}
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000709
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000710bool CXIndexDataConsumer::handleObjCImplementation(
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000711 const ObjCImplementationDecl *D) {
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000712 ObjCContainerDeclInfo ContDInfo(/*isForwardRef=*/false,
Argyrios Kyrtzidis233f12d2011-11-15 06:20:24 +0000713 /*isRedeclaration=*/true,
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000714 /*isImplementation=*/true);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000715 return handleObjCContainer(D, D->getLocation(), getCursor(D), ContDInfo);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000716}
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000717
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000718bool CXIndexDataConsumer::handleObjCProtocol(const ObjCProtocolDecl *D) {
Douglas Gregorf6102672012-01-01 21:23:57 +0000719 if (!D->isThisDeclarationADefinition()) {
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000720 if (shouldSuppressRefs() && markEntityOccurrenceInFile(D, D->getLocation()))
Douglas Gregorf6102672012-01-01 21:23:57 +0000721 return false; // already occurred.
722
723 // FIXME: This seems like the wrong definition for redeclaration.
Douglas Gregorec9fd132012-01-14 16:38:05 +0000724 bool isRedeclaration = D->hasDefinition() || D->getPreviousDecl();
Douglas Gregorf6102672012-01-01 21:23:57 +0000725 ObjCContainerDeclInfo ContDInfo(/*isForwardRef=*/true,
726 isRedeclaration,
727 /*isImplementation=*/false);
728 return handleObjCContainer(D, D->getLocation(),
729 MakeCursorObjCProtocolRef(D, D->getLocation(),
730 CXTU),
731 ContDInfo);
732 }
733
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000734 ScratchAlloc SA(*this);
Douglas Gregore6e48b12012-01-01 19:29:29 +0000735 ObjCProtocolList EmptyProtoList;
736 ObjCProtocolListInfo ProtListInfo(D->isThisDeclarationADefinition()
737 ? D->getReferencedProtocols()
738 : EmptyProtoList,
739 *this, SA);
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000740
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000741 ObjCProtocolDeclInfo ProtInfo(D);
742 ProtInfo.ObjCProtoRefListInfo = ProtListInfo.getListInfo();
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000743
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000744 return handleObjCContainer(D, D->getLocation(), getCursor(D), ProtInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000745}
746
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000747bool CXIndexDataConsumer::handleObjCCategory(const ObjCCategoryDecl *D) {
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000748 ScratchAlloc SA(*this);
749
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000750 ObjCCategoryDeclInfo CatDInfo(/*isImplementation=*/false);
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000751 EntityInfo ClassEntity;
Argyrios Kyrtzidisd992e142011-11-15 06:20:16 +0000752 const ObjCInterfaceDecl *IFaceD = D->getClassInterface();
753 SourceLocation ClassLoc = D->getLocation();
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000754 SourceLocation CategoryLoc = D->IsClassExtension() ? ClassLoc
755 : D->getCategoryNameLoc();
Argyrios Kyrtzidise4acd232011-11-16 02:34:59 +0000756 getEntityInfo(IFaceD, ClassEntity, SA);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000757
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000758 if (shouldSuppressRefs())
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +0000759 markEntityOccurrenceInFile(IFaceD, ClassLoc);
760
Argyrios Kyrtzidis9b9f7a92011-12-13 18:47:45 +0000761 ObjCProtocolListInfo ProtInfo(D->getReferencedProtocols(), *this, SA);
762
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000763 CatDInfo.ObjCCatDeclInfo.containerInfo = &CatDInfo.ObjCContDeclInfo;
Argyrios Kyrtzidise4acd232011-11-16 02:34:59 +0000764 if (IFaceD) {
765 CatDInfo.ObjCCatDeclInfo.objcClass = &ClassEntity;
766 CatDInfo.ObjCCatDeclInfo.classCursor =
767 MakeCursorObjCClassRef(IFaceD, ClassLoc, CXTU);
768 } else {
Craig Topper69186e72014-06-08 08:38:04 +0000769 CatDInfo.ObjCCatDeclInfo.objcClass = nullptr;
Argyrios Kyrtzidise4acd232011-11-16 02:34:59 +0000770 CatDInfo.ObjCCatDeclInfo.classCursor = clang_getNullCursor();
771 }
Argyrios Kyrtzidisd992e142011-11-15 06:20:16 +0000772 CatDInfo.ObjCCatDeclInfo.classLoc = getIndexLoc(ClassLoc);
Argyrios Kyrtzidis9b9f7a92011-12-13 18:47:45 +0000773 CatDInfo.ObjCProtoListInfo = ProtInfo.getListInfo();
774 CatDInfo.ObjCCatDeclInfo.protocols = &CatDInfo.ObjCProtoListInfo;
775
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000776 return handleObjCContainer(D, CategoryLoc, getCursor(D), CatDInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000777}
778
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000779bool CXIndexDataConsumer::handleObjCCategoryImpl(const ObjCCategoryImplDecl *D) {
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000780 ScratchAlloc SA(*this);
781
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000782 const ObjCCategoryDecl *CatD = D->getCategoryDecl();
Argyrios Kyrtzidis86acd722011-11-14 22:39:19 +0000783 ObjCCategoryDeclInfo CatDInfo(/*isImplementation=*/true);
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000784 EntityInfo ClassEntity;
Argyrios Kyrtzidisdf15c202011-11-16 02:35:05 +0000785 const ObjCInterfaceDecl *IFaceD = CatD->getClassInterface();
786 SourceLocation ClassLoc = D->getLocation();
Argyrios Kyrtzidis4996f5f2011-12-09 00:31:40 +0000787 SourceLocation CategoryLoc = D->getCategoryNameLoc();
Argyrios Kyrtzidisdf15c202011-11-16 02:35:05 +0000788 getEntityInfo(IFaceD, ClassEntity, SA);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000789
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000790 if (shouldSuppressRefs())
Argyrios Kyrtzidis18dc04e2012-01-23 21:28:38 +0000791 markEntityOccurrenceInFile(IFaceD, ClassLoc);
792
Argyrios Kyrtzidis3e429e72011-11-12 02:16:30 +0000793 CatDInfo.ObjCCatDeclInfo.containerInfo = &CatDInfo.ObjCContDeclInfo;
Argyrios Kyrtzidisdf15c202011-11-16 02:35:05 +0000794 if (IFaceD) {
795 CatDInfo.ObjCCatDeclInfo.objcClass = &ClassEntity;
796 CatDInfo.ObjCCatDeclInfo.classCursor =
797 MakeCursorObjCClassRef(IFaceD, ClassLoc, CXTU);
798 } else {
Craig Topper69186e72014-06-08 08:38:04 +0000799 CatDInfo.ObjCCatDeclInfo.objcClass = nullptr;
Argyrios Kyrtzidisdf15c202011-11-16 02:35:05 +0000800 CatDInfo.ObjCCatDeclInfo.classCursor = clang_getNullCursor();
801 }
802 CatDInfo.ObjCCatDeclInfo.classLoc = getIndexLoc(ClassLoc);
Craig Topper69186e72014-06-08 08:38:04 +0000803 CatDInfo.ObjCCatDeclInfo.protocols = nullptr;
Argyrios Kyrtzidis9b9f7a92011-12-13 18:47:45 +0000804
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000805 return handleObjCContainer(D, CategoryLoc, getCursor(D), CatDInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000806}
807
Argyrios Kyrtzidisaee15fb2017-01-26 02:11:50 +0000808bool CXIndexDataConsumer::handleObjCMethod(const ObjCMethodDecl *D,
809 SourceLocation Loc) {
Argyrios Kyrtzidis8b71bc72012-12-06 19:41:16 +0000810 bool isDef = D->isThisDeclarationADefinition();
811 bool isContainer = isDef;
812 bool isSkipped = false;
813 if (D->hasSkippedBody()) {
814 isSkipped = true;
815 isDef = true;
816 isContainer = false;
817 }
818
819 DeclInfo DInfo(!D->isCanonicalDecl(), isDef, isContainer);
820 if (isSkipped)
821 DInfo.flags |= CXIdxDeclFlag_Skipped;
Argyrios Kyrtzidisaee15fb2017-01-26 02:11:50 +0000822 return handleDecl(D, Loc, getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000823}
824
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000825bool CXIndexDataConsumer::handleSynthesizedObjCProperty(
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000826 const ObjCPropertyImplDecl *D) {
827 ObjCPropertyDecl *PD = D->getPropertyDecl();
Argyrios Kyrtzidisa4cffec2016-02-09 19:07:19 +0000828 auto *DC = D->getDeclContext();
829 return handleReference(PD, D->getLocation(), getCursor(D),
830 dyn_cast<NamedDecl>(DC), DC);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000831}
832
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000833bool CXIndexDataConsumer::handleSynthesizedObjCMethod(const ObjCMethodDecl *D,
Argyrios Kyrtzidis34608802012-02-28 17:50:39 +0000834 SourceLocation Loc,
835 const DeclContext *LexicalDC) {
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000836 DeclInfo DInfo(/*isRedeclaration=*/true, /*isDefinition=*/true,
837 /*isContainer=*/false);
Argyrios Kyrtzidis66c49f72016-03-31 20:18:22 +0000838 return handleDecl(D, Loc, getCursor(D), DInfo, LexicalDC, D->getDeclContext());
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000839}
840
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000841bool CXIndexDataConsumer::handleObjCProperty(const ObjCPropertyDecl *D) {
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +0000842 ScratchAlloc SA(*this);
843
Argyrios Kyrtzidis93db2922012-02-28 17:50:33 +0000844 ObjCPropertyDeclInfo DInfo;
845 EntityInfo GetterEntity;
846 EntityInfo SetterEntity;
Argyrios Kyrtzidis93db2922012-02-28 17:50:33 +0000847
848 DInfo.ObjCPropDeclInfo.declInfo = &DInfo;
849
850 if (ObjCMethodDecl *Getter = D->getGetterMethodDecl()) {
851 getEntityInfo(Getter, GetterEntity, SA);
852 DInfo.ObjCPropDeclInfo.getter = &GetterEntity;
853 } else {
Craig Topper69186e72014-06-08 08:38:04 +0000854 DInfo.ObjCPropDeclInfo.getter = nullptr;
Argyrios Kyrtzidis93db2922012-02-28 17:50:33 +0000855 }
856 if (ObjCMethodDecl *Setter = D->getSetterMethodDecl()) {
857 getEntityInfo(Setter, SetterEntity, SA);
858 DInfo.ObjCPropDeclInfo.setter = &SetterEntity;
859 } else {
Craig Topper69186e72014-06-08 08:38:04 +0000860 DInfo.ObjCPropDeclInfo.setter = nullptr;
Argyrios Kyrtzidis93db2922012-02-28 17:50:33 +0000861 }
862
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000863 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000864}
865
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000866bool CXIndexDataConsumer::handleNamespace(const NamespaceDecl *D) {
Argyrios Kyrtzidis2b0b43c2011-12-07 05:52:06 +0000867 DeclInfo DInfo(/*isRedeclaration=*/!D->isOriginalNamespace(),
868 /*isDefinition=*/true,
869 /*isContainer=*/true);
870 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
871}
872
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000873bool CXIndexDataConsumer::handleClassTemplate(const ClassTemplateDecl *D) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000874 return handleCXXRecordDecl(D->getTemplatedDecl(), D);
875}
876
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000877bool CXIndexDataConsumer::handleFunctionTemplate(const FunctionTemplateDecl *D) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000878 DeclInfo DInfo(/*isRedeclaration=*/!D->isCanonicalDecl(),
879 /*isDefinition=*/D->isThisDeclarationADefinition(),
880 /*isContainer=*/D->isThisDeclarationADefinition());
881 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
882}
883
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000884bool CXIndexDataConsumer::handleTypeAliasTemplate(const TypeAliasTemplateDecl *D) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000885 DeclInfo DInfo(/*isRedeclaration=*/!D->isCanonicalDecl(),
886 /*isDefinition=*/true, /*isContainer=*/false);
887 return handleDecl(D, D->getLocation(), getCursor(D), DInfo);
888}
889
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000890bool CXIndexDataConsumer::handleReference(const NamedDecl *D, SourceLocation Loc,
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000891 const NamedDecl *Parent,
892 const DeclContext *DC,
Argyrios Kyrtzidis0c7735e52011-10-18 15:50:50 +0000893 const Expr *E,
Fangrui Song31b97192018-02-12 17:42:09 +0000894 CXIdxEntityRefKind Kind,
895 CXSymbolRole Role) {
Fangrui Songe61045f2018-01-08 18:57:38 +0000896 if (!D || !DC)
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000897 return false;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000898
Dmitri Gribenko9c256e32013-01-14 00:46:27 +0000899 CXCursor Cursor = E ? MakeCXCursor(E, cast<Decl>(DC), CXTU)
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000900 : getRefCursor(D, Loc);
Fangrui Song31b97192018-02-12 17:42:09 +0000901 return handleReference(D, Loc, Cursor, Parent, DC, E, Kind, Role);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000902}
903
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000904bool CXIndexDataConsumer::handleReference(const NamedDecl *D, SourceLocation Loc,
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000905 CXCursor Cursor,
906 const NamedDecl *Parent,
907 const DeclContext *DC,
908 const Expr *E,
Fangrui Song31b97192018-02-12 17:42:09 +0000909 CXIdxEntityRefKind Kind,
910 CXSymbolRole Role) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000911 if (!CB.indexEntityReference)
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000912 return false;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000913
Fangrui Songe61045f2018-01-08 18:57:38 +0000914 if (!D || !DC)
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000915 return false;
916 if (Loc.isInvalid())
917 return false;
Argyrios Kyrtzidis68e87e12012-09-10 22:58:04 +0000918 if (!shouldIndexFunctionLocalSymbols() && isFunctionLocalDecl(D))
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000919 return false;
920 if (isNotFromSourceFile(D->getLocation()))
921 return false;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000922 if (D->isImplicit() && shouldIgnoreIfImplicit(D))
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000923 return false;
Argyrios Kyrtzidis1cff7952011-11-18 00:26:46 +0000924
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +0000925 if (shouldSuppressRefs()) {
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000926 if (markEntityOccurrenceInFile(D, Loc))
927 return false; // already occurred.
Argyrios Kyrtzidise4acd232011-11-16 02:34:59 +0000928 }
929
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000930 ScratchAlloc SA(*this);
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000931 EntityInfo RefEntity, ParentEntity;
932 getEntityInfo(D, RefEntity, SA);
933 if (!RefEntity.USR)
934 return false;
935
936 getEntityInfo(Parent, ParentEntity, SA);
937
938 ContainerInfo Container;
939 getContainerInfo(DC, Container);
940
Argyrios Kyrtzidis663c8ec2011-12-07 20:44:19 +0000941 CXIdxEntityRefInfo Info = { Kind,
942 Cursor,
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000943 getIndexLoc(Loc),
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000944 &RefEntity,
Craig Topper69186e72014-06-08 08:38:04 +0000945 Parent ? &ParentEntity : nullptr,
Fangrui Song31b97192018-02-12 17:42:09 +0000946 &Container,
947 Role };
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000948 CB.indexEntityReference(ClientData, &Info);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +0000949 return true;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000950}
951
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000952bool CXIndexDataConsumer::isNotFromSourceFile(SourceLocation Loc) const {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000953 if (Loc.isInvalid())
954 return true;
955 SourceManager &SM = Ctx->getSourceManager();
956 SourceLocation FileLoc = SM.getFileLoc(Loc);
957 FileID FID = SM.getFileID(FileLoc);
Craig Topper69186e72014-06-08 08:38:04 +0000958 return SM.getFileEntryForID(FID) == nullptr;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000959}
960
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000961void CXIndexDataConsumer::addContainerInMap(const DeclContext *DC,
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +0000962 CXIdxClientContainer container) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000963 if (!DC)
964 return;
965
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +0000966 ContainerMapTy::iterator I = ContainerMap.find(DC);
967 if (I == ContainerMap.end()) {
968 if (container)
969 ContainerMap[DC] = container;
970 return;
971 }
972 // Allow changing the container of a previously seen DeclContext so we
973 // can handle invalid user code, like a function re-definition.
974 if (container)
975 I->second = container;
976 else
977 ContainerMap.erase(I);
978}
979
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000980CXIdxClientEntity CXIndexDataConsumer::getClientEntity(const Decl *D) const {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000981 if (!D)
Craig Topper69186e72014-06-08 08:38:04 +0000982 return nullptr;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000983 EntityMapTy::const_iterator I = EntityMap.find(D);
984 if (I == EntityMap.end())
Craig Topper69186e72014-06-08 08:38:04 +0000985 return nullptr;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000986 return I->second;
987}
988
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000989void CXIndexDataConsumer::setClientEntity(const Decl *D, CXIdxClientEntity client) {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000990 if (!D)
991 return;
992 EntityMap[D] = client;
993}
994
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +0000995bool CXIndexDataConsumer::handleCXXRecordDecl(const CXXRecordDecl *RD,
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +0000996 const NamedDecl *OrigD) {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000997 if (RD->isThisDeclarationADefinition()) {
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +0000998 ScratchAlloc SA(*this);
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +0000999 CXXClassDeclInfo CXXDInfo(/*isRedeclaration=*/!OrigD->isCanonicalDecl(),
1000 /*isDefinition=*/RD->isThisDeclarationADefinition());
1001 CXXBasesListInfo BaseList(RD, *this, SA);
1002 CXXDInfo.CXXClassInfo.declInfo = &CXXDInfo;
1003 CXXDInfo.CXXClassInfo.bases = BaseList.getBases();
1004 CXXDInfo.CXXClassInfo.numBases = BaseList.getNumBases();
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001005
Argyrios Kyrtzidis7e747952012-02-14 22:23:11 +00001006 if (shouldSuppressRefs()) {
Argyrios Kyrtzidis95c0feb2012-02-08 03:04:33 +00001007 // Go through bases and mark them as referenced.
1008 for (unsigned i = 0, e = BaseList.getNumBases(); i != e; ++i) {
1009 const CXIdxBaseClassInfo *baseInfo = BaseList.getBases()[i];
1010 if (baseInfo->base) {
1011 const NamedDecl *BaseD = BaseList.BaseEntities[i].Dcl;
1012 SourceLocation
1013 Loc = SourceLocation::getFromRawEncoding(baseInfo->loc.int_data);
1014 markEntityOccurrenceInFile(BaseD, Loc);
1015 }
1016 }
1017 }
1018
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001019 return handleDecl(OrigD, OrigD->getLocation(), getCursor(OrigD), CXXDInfo);
1020 }
1021
1022 DeclInfo DInfo(/*isRedeclaration=*/!OrigD->isCanonicalDecl(),
1023 /*isDefinition=*/RD->isThisDeclarationADefinition(),
1024 /*isContainer=*/RD->isThisDeclarationADefinition());
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001025 return handleDecl(OrigD, OrigD->getLocation(), getCursor(OrigD), DInfo);
1026}
1027
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001028bool CXIndexDataConsumer::markEntityOccurrenceInFile(const NamedDecl *D,
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001029 SourceLocation Loc) {
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +00001030 if (!D || Loc.isInvalid())
1031 return true;
1032
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001033 SourceManager &SM = Ctx->getSourceManager();
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001034 D = getEntityDecl(D);
1035
Argyrios Kyrtzidisccdf8272011-12-13 18:47:35 +00001036 std::pair<FileID, unsigned> LocInfo = SM.getDecomposedLoc(SM.getFileLoc(Loc));
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001037 FileID FID = LocInfo.first;
1038 if (FID.isInvalid())
1039 return true;
1040
1041 const FileEntry *FE = SM.getFileEntryForID(FID);
1042 if (!FE)
1043 return true;
Alp Toker54ee53a2013-11-30 23:33:14 +00001044 RefFileOccurrence RefOccur(FE, D);
1045 std::pair<llvm::DenseSet<RefFileOccurrence>::iterator, bool>
1046 res = RefFileOccurrences.insert(RefOccur);
Alexander Kornienko1a9f1842015-12-28 15:24:08 +00001047 return !res.second; // already in map
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001048}
1049
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001050const NamedDecl *CXIndexDataConsumer::getEntityDecl(const NamedDecl *D) const {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001051 assert(D);
1052 D = cast<NamedDecl>(D->getCanonicalDecl());
1053
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001054 if (const ObjCImplementationDecl *
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001055 ImplD = dyn_cast<ObjCImplementationDecl>(D)) {
1056 return getEntityDecl(ImplD->getClassInterface());
1057
1058 } else if (const ObjCCategoryImplDecl *
1059 CatImplD = dyn_cast<ObjCCategoryImplDecl>(D)) {
1060 return getEntityDecl(CatImplD->getCategoryDecl());
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001061 } else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
1062 if (FunctionTemplateDecl *TemplD = FD->getDescribedFunctionTemplate())
1063 return getEntityDecl(TemplD);
1064 } else if (const CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D)) {
1065 if (ClassTemplateDecl *TemplD = RD->getDescribedClassTemplate())
1066 return getEntityDecl(TemplD);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001067 }
1068
1069 return D;
1070}
1071
1072const DeclContext *
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001073CXIndexDataConsumer::getEntityContainer(const Decl *D) const {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001074 const DeclContext *DC = dyn_cast<DeclContext>(D);
1075 if (DC)
1076 return DC;
1077
1078 if (const ClassTemplateDecl *ClassTempl = dyn_cast<ClassTemplateDecl>(D)) {
1079 DC = ClassTempl->getTemplatedDecl();
Dmitri Gribenko909eebc2012-12-19 17:29:30 +00001080 } else if (const FunctionTemplateDecl *
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001081 FuncTempl = dyn_cast<FunctionTemplateDecl>(D)) {
1082 DC = FuncTempl->getTemplatedDecl();
1083 }
1084
1085 return DC;
1086}
1087
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001088CXIdxClientContainer
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001089CXIndexDataConsumer::getClientContainerForDC(const DeclContext *DC) const {
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001090 if (!DC)
Craig Topper69186e72014-06-08 08:38:04 +00001091 return nullptr;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001092
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001093 ContainerMapTy::const_iterator I = ContainerMap.find(DC);
Argyrios Kyrtzidisdf15c202011-11-16 02:35:05 +00001094 if (I == ContainerMap.end())
Craig Topper69186e72014-06-08 08:38:04 +00001095 return nullptr;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001096
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001097 return I->second;
1098}
1099
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001100CXIdxClientFile CXIndexDataConsumer::getIndexFile(const FileEntry *File) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001101 if (!File)
Craig Topper69186e72014-06-08 08:38:04 +00001102 return nullptr;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001103
1104 FileMapTy::iterator FI = FileMap.find(File);
1105 if (FI != FileMap.end())
1106 return FI->second;
1107
Craig Topper69186e72014-06-08 08:38:04 +00001108 return nullptr;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001109}
1110
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001111CXIdxLoc CXIndexDataConsumer::getIndexLoc(SourceLocation Loc) const {
Craig Topper69186e72014-06-08 08:38:04 +00001112 CXIdxLoc idxLoc = { {nullptr, nullptr}, 0 };
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001113 if (Loc.isInvalid())
1114 return idxLoc;
1115
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001116 idxLoc.ptr_data[0] = const_cast<CXIndexDataConsumer *>(this);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001117 idxLoc.int_data = Loc.getRawEncoding();
1118 return idxLoc;
1119}
1120
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001121void CXIndexDataConsumer::translateLoc(SourceLocation Loc,
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001122 CXIdxClientFile *indexFile, CXFile *file,
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001123 unsigned *line, unsigned *column,
1124 unsigned *offset) {
1125 if (Loc.isInvalid())
1126 return;
1127
1128 SourceManager &SM = Ctx->getSourceManager();
1129 Loc = SM.getFileLoc(Loc);
1130
1131 std::pair<FileID, unsigned> LocInfo = SM.getDecomposedLoc(Loc);
1132 FileID FID = LocInfo.first;
1133 unsigned FileOffset = LocInfo.second;
1134
1135 if (FID.isInvalid())
1136 return;
1137
1138 const FileEntry *FE = SM.getFileEntryForID(FID);
1139 if (indexFile)
1140 *indexFile = getIndexFile(FE);
1141 if (file)
David Greenee6c9fe02013-01-15 22:09:51 +00001142 *file = const_cast<FileEntry *>(FE);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001143 if (line)
1144 *line = SM.getLineNumber(FID, FileOffset);
1145 if (column)
1146 *column = SM.getColumnNumber(FID, FileOffset);
1147 if (offset)
1148 *offset = FileOffset;
1149}
1150
Ben Langmuir443913f2016-03-25 17:01:59 +00001151static CXIdxEntityKind getEntityKindFromSymbolKind(SymbolKind K, SymbolLanguage L);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001152static CXIdxEntityCXXTemplateKind
Argyrios Kyrtzidisdb469832016-11-11 23:49:55 +00001153getEntityKindFromSymbolProperties(SymbolPropertySet K);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001154static CXIdxEntityLanguage getEntityLangFromSymbolLang(SymbolLanguage L);
1155
1156void CXIndexDataConsumer::getEntityInfo(const NamedDecl *D,
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001157 EntityInfo &EntityInfo,
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +00001158 ScratchAlloc &SA) {
Argyrios Kyrtzidise4acd232011-11-16 02:34:59 +00001159 if (!D)
1160 return;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001161
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001162 D = getEntityDecl(D);
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001163 EntityInfo.cursor = getCursor(D);
1164 EntityInfo.Dcl = D;
1165 EntityInfo.IndexCtx = this;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001166
1167 SymbolInfo SymInfo = getSymbolInfo(D);
Ben Langmuir443913f2016-03-25 17:01:59 +00001168 EntityInfo.kind = getEntityKindFromSymbolKind(SymInfo.Kind, SymInfo.Lang);
Argyrios Kyrtzidisdb469832016-11-11 23:49:55 +00001169 EntityInfo.templateKind = getEntityKindFromSymbolProperties(SymInfo.Properties);
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001170 EntityInfo.lang = getEntityLangFromSymbolLang(SymInfo.Lang);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001171
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +00001172 if (D->hasAttrs()) {
Argyrios Kyrtzidis0dbe9b62012-03-31 01:14:06 +00001173 EntityInfo.AttrList = AttrListInfo::create(D, *this);
1174 EntityInfo.attributes = EntityInfo.AttrList->getAttrs();
1175 EntityInfo.numAttributes = EntityInfo.AttrList->getNumAttrs();
Argyrios Kyrtzidis4d873b72011-12-15 00:05:00 +00001176 }
1177
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001178 if (EntityInfo.kind == CXIdxEntity_Unexposed)
1179 return;
1180
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001181 if (IdentifierInfo *II = D->getIdentifier()) {
1182 EntityInfo.name = SA.toCStr(II->getName());
1183
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001184 } else if (isa<TagDecl>(D) || isa<FieldDecl>(D) || isa<NamespaceDecl>(D)) {
Craig Topper69186e72014-06-08 08:38:04 +00001185 EntityInfo.name = nullptr; // anonymous tag/field/namespace.
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001186
1187 } else {
Dylan Noblesmithf1a13f22012-02-13 12:32:26 +00001188 SmallString<256> StrBuf;
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001189 {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001190 llvm::raw_svector_ostream OS(StrBuf);
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001191 D->printName(OS);
1192 }
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001193 EntityInfo.name = SA.copyCStr(StrBuf.str());
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001194 }
1195
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001196 {
Dylan Noblesmithf1a13f22012-02-13 12:32:26 +00001197 SmallString<512> StrBuf;
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001198 bool Ignore = getDeclCursorUSR(D, StrBuf);
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001199 if (Ignore) {
Craig Topper69186e72014-06-08 08:38:04 +00001200 EntityInfo.USR = nullptr;
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001201 } else {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001202 EntityInfo.USR = SA.copyCStr(StrBuf.str());
Argyrios Kyrtzidis7519c5e2011-11-11 00:23:36 +00001203 }
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001204 }
1205}
1206
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001207void CXIndexDataConsumer::getContainerInfo(const DeclContext *DC,
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001208 ContainerInfo &ContInfo) {
1209 ContInfo.cursor = getCursor(cast<Decl>(DC));
1210 ContInfo.DC = DC;
1211 ContInfo.IndexCtx = this;
1212}
1213
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001214CXCursor CXIndexDataConsumer::getRefCursor(const NamedDecl *D, SourceLocation Loc) {
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001215 if (const TypeDecl *TD = dyn_cast<TypeDecl>(D))
1216 return MakeCursorTypeRef(TD, Loc, CXTU);
1217 if (const ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(D))
1218 return MakeCursorObjCClassRef(ID, Loc, CXTU);
1219 if (const ObjCProtocolDecl *PD = dyn_cast<ObjCProtocolDecl>(D))
1220 return MakeCursorObjCProtocolRef(PD, Loc, CXTU);
Argyrios Kyrtzidiseffdbf52011-11-18 00:26:51 +00001221 if (const TemplateDecl *Template = dyn_cast<TemplateDecl>(D))
1222 return MakeCursorTemplateRef(Template, Loc, CXTU);
1223 if (const NamespaceDecl *Namespace = dyn_cast<NamespaceDecl>(D))
1224 return MakeCursorNamespaceRef(Namespace, Loc, CXTU);
1225 if (const NamespaceAliasDecl *Namespace = dyn_cast<NamespaceAliasDecl>(D))
1226 return MakeCursorNamespaceRef(Namespace, Loc, CXTU);
1227 if (const FieldDecl *Field = dyn_cast<FieldDecl>(D))
1228 return MakeCursorMemberRef(Field, Loc, CXTU);
Douglas Gregor30093832012-02-15 00:54:55 +00001229 if (const VarDecl *Var = dyn_cast<VarDecl>(D))
1230 return MakeCursorVariableRef(Var, Loc, CXTU);
1231
Argyrios Kyrtzidisdc199a32011-10-17 19:48:19 +00001232 return clang_getNullCursor();
1233}
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001234
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001235bool CXIndexDataConsumer::shouldIgnoreIfImplicit(const Decl *D) {
Argyrios Kyrtzidis41fc05c2011-11-23 20:27:26 +00001236 if (isa<ObjCInterfaceDecl>(D))
1237 return false;
1238 if (isa<ObjCCategoryDecl>(D))
1239 return false;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001240 if (isa<ObjCIvarDecl>(D))
1241 return false;
1242 if (isa<ObjCMethodDecl>(D))
1243 return false;
Argyrios Kyrtzidis184b1442012-10-03 21:05:44 +00001244 if (isa<ImportDecl>(D))
1245 return false;
Argyrios Kyrtzidis4c910b12011-11-22 07:24:51 +00001246 return true;
1247}
Argyrios Kyrtzidise5dc5b32012-02-10 20:10:44 +00001248
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001249bool CXIndexDataConsumer::isTemplateImplicitInstantiation(const Decl *D) {
Argyrios Kyrtzidise5dc5b32012-02-10 20:10:44 +00001250 if (const ClassTemplateSpecializationDecl *
1251 SD = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
1252 return SD->getSpecializationKind() == TSK_ImplicitInstantiation;
1253 }
1254 if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
1255 return FD->getTemplateSpecializationKind() == TSK_ImplicitInstantiation;
1256 }
1257 return false;
1258}
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001259
Ben Langmuir443913f2016-03-25 17:01:59 +00001260static CXIdxEntityKind getEntityKindFromSymbolKind(SymbolKind K, SymbolLanguage Lang) {
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001261 switch (K) {
1262 case SymbolKind::Unknown:
1263 case SymbolKind::Module:
1264 case SymbolKind::Macro:
Ben Langmuir443913f2016-03-25 17:01:59 +00001265 case SymbolKind::ClassProperty:
Ben Langmuirfd6e39c2017-08-16 23:12:21 +00001266 case SymbolKind::Using:
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001267 return CXIdxEntity_Unexposed;
1268
1269 case SymbolKind::Enum: return CXIdxEntity_Enum;
1270 case SymbolKind::Struct: return CXIdxEntity_Struct;
1271 case SymbolKind::Union: return CXIdxEntity_Union;
Ben Langmuir443913f2016-03-25 17:01:59 +00001272 case SymbolKind::TypeAlias:
1273 if (Lang == SymbolLanguage::CXX)
1274 return CXIdxEntity_CXXTypeAlias;
1275 return CXIdxEntity_Typedef;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001276 case SymbolKind::Function: return CXIdxEntity_Function;
1277 case SymbolKind::Variable: return CXIdxEntity_Variable;
Ben Langmuir443913f2016-03-25 17:01:59 +00001278 case SymbolKind::Field:
1279 if (Lang == SymbolLanguage::ObjC)
1280 return CXIdxEntity_ObjCIvar;
1281 return CXIdxEntity_Field;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001282 case SymbolKind::EnumConstant: return CXIdxEntity_EnumConstant;
Ben Langmuir443913f2016-03-25 17:01:59 +00001283 case SymbolKind::Class:
1284 if (Lang == SymbolLanguage::ObjC)
1285 return CXIdxEntity_ObjCClass;
1286 return CXIdxEntity_CXXClass;
1287 case SymbolKind::Protocol:
1288 if (Lang == SymbolLanguage::ObjC)
1289 return CXIdxEntity_ObjCProtocol;
1290 return CXIdxEntity_CXXInterface;
1291 case SymbolKind::Extension: return CXIdxEntity_ObjCCategory;
1292 case SymbolKind::InstanceMethod:
1293 if (Lang == SymbolLanguage::ObjC)
1294 return CXIdxEntity_ObjCInstanceMethod;
1295 return CXIdxEntity_CXXInstanceMethod;
1296 case SymbolKind::ClassMethod: return CXIdxEntity_ObjCClassMethod;
1297 case SymbolKind::StaticMethod: return CXIdxEntity_CXXStaticMethod;
1298 case SymbolKind::InstanceProperty: return CXIdxEntity_ObjCProperty;
1299 case SymbolKind::StaticProperty: return CXIdxEntity_CXXStaticVariable;
1300 case SymbolKind::Namespace: return CXIdxEntity_CXXNamespace;
1301 case SymbolKind::NamespaceAlias: return CXIdxEntity_CXXNamespaceAlias;
1302 case SymbolKind::Constructor: return CXIdxEntity_CXXConstructor;
1303 case SymbolKind::Destructor: return CXIdxEntity_CXXDestructor;
1304 case SymbolKind::ConversionFunction: return CXIdxEntity_CXXConversionFunction;
Argyrios Kyrtzidis6d1a15b22017-02-26 05:37:56 +00001305 case SymbolKind::Parameter: return CXIdxEntity_Variable;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001306 }
Saleem Abdulrasool9b0ac332016-02-15 00:36:52 +00001307 llvm_unreachable("invalid symbol kind");
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001308}
1309
1310static CXIdxEntityCXXTemplateKind
Argyrios Kyrtzidisdb469832016-11-11 23:49:55 +00001311getEntityKindFromSymbolProperties(SymbolPropertySet K) {
Sam McCalle83ae112017-12-23 19:31:24 +00001312 if (K & (SymbolPropertySet)SymbolProperty::TemplatePartialSpecialization)
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001313 return CXIdxEntity_TemplatePartialSpecialization;
Sam McCalle83ae112017-12-23 19:31:24 +00001314 if (K & (SymbolPropertySet)SymbolProperty::TemplateSpecialization)
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001315 return CXIdxEntity_TemplateSpecialization;
Sam McCalle83ae112017-12-23 19:31:24 +00001316 if (K & (SymbolPropertySet)SymbolProperty::Generic)
Argyrios Kyrtzidisf2142cb2016-04-22 07:21:04 +00001317 return CXIdxEntity_Template;
1318 return CXIdxEntity_NonTemplate;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001319}
1320
1321static CXIdxEntityLanguage getEntityLangFromSymbolLang(SymbolLanguage L) {
1322 switch (L) {
1323 case SymbolLanguage::C: return CXIdxEntityLang_C;
1324 case SymbolLanguage::ObjC: return CXIdxEntityLang_ObjC;
1325 case SymbolLanguage::CXX: return CXIdxEntityLang_CXX;
Argyrios Kyrtzidisb4b85f22017-04-24 14:52:00 +00001326 case SymbolLanguage::Swift: return CXIdxEntityLang_Swift;
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001327 }
Saleem Abdulrasool9b0ac332016-02-15 00:36:52 +00001328 llvm_unreachable("invalid symbol language");
Argyrios Kyrtzidisf4fb85b2016-02-12 23:10:59 +00001329}