1 //===- IndexingAction.cpp - Frontend index action -------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "clang/Index/IndexingAction.h"
10 #include "IndexingContext.h"
11 #include "clang/Frontend/CompilerInstance.h"
12 #include "clang/Frontend/FrontendAction.h"
13 #include "clang/Frontend/MultiplexConsumer.h"
14 #include "clang/Index/IndexDataConsumer.h"
15 #include "clang/Lex/PPCallbacks.h"
16 #include "clang/Lex/Preprocessor.h"
17 #include "clang/Serialization/ASTReader.h"
18 #include "llvm/ADT/STLExtras.h"
21 using namespace clang
;
22 using namespace clang::index
;
26 class IndexPPCallbacks final
: public PPCallbacks
{
27 std::shared_ptr
<IndexingContext
> IndexCtx
;
30 IndexPPCallbacks(std::shared_ptr
<IndexingContext
> IndexCtx
)
31 : IndexCtx(std::move(IndexCtx
)) {}
33 void MacroExpands(const Token
&MacroNameTok
, const MacroDefinition
&MD
,
34 SourceRange Range
, const MacroArgs
*Args
) override
{
35 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
36 Range
.getBegin(), *MD
.getMacroInfo());
39 void MacroDefined(const Token
&MacroNameTok
,
40 const MacroDirective
*MD
) override
{
41 IndexCtx
->handleMacroDefined(*MacroNameTok
.getIdentifierInfo(),
42 MacroNameTok
.getLocation(),
46 void MacroUndefined(const Token
&MacroNameTok
, const MacroDefinition
&MD
,
47 const MacroDirective
*Undef
) override
{
48 if (!MD
.getMacroInfo()) // Ignore noop #undef.
50 IndexCtx
->handleMacroUndefined(*MacroNameTok
.getIdentifierInfo(),
51 MacroNameTok
.getLocation(),
55 void Defined(const Token
&MacroNameTok
, const MacroDefinition
&MD
,
56 SourceRange Range
) override
{
57 if (!MD
.getMacroInfo()) // Ignore nonexistent macro.
59 // Note: this is defined(M), not #define M
60 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
61 MacroNameTok
.getLocation(),
64 void Ifdef(SourceLocation Loc
, const Token
&MacroNameTok
,
65 const MacroDefinition
&MD
) override
{
66 if (!MD
.getMacroInfo()) // Ignore non-existent macro.
68 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
69 MacroNameTok
.getLocation(),
72 void Ifndef(SourceLocation Loc
, const Token
&MacroNameTok
,
73 const MacroDefinition
&MD
) override
{
74 if (!MD
.getMacroInfo()) // Ignore nonexistent macro.
76 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
77 MacroNameTok
.getLocation(),
81 using PPCallbacks::Elifdef
;
82 using PPCallbacks::Elifndef
;
83 void Elifdef(SourceLocation Loc
, const Token
&MacroNameTok
,
84 const MacroDefinition
&MD
) override
{
85 if (!MD
.getMacroInfo()) // Ignore non-existent macro.
87 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
88 MacroNameTok
.getLocation(),
91 void Elifndef(SourceLocation Loc
, const Token
&MacroNameTok
,
92 const MacroDefinition
&MD
) override
{
93 if (!MD
.getMacroInfo()) // Ignore non-existent macro.
95 IndexCtx
->handleMacroReference(*MacroNameTok
.getIdentifierInfo(),
96 MacroNameTok
.getLocation(),
101 class IndexASTConsumer final
: public ASTConsumer
{
102 std::shared_ptr
<IndexDataConsumer
> DataConsumer
;
103 std::shared_ptr
<IndexingContext
> IndexCtx
;
104 std::shared_ptr
<Preprocessor
> PP
;
105 std::function
<bool(const Decl
*)> ShouldSkipFunctionBody
;
108 IndexASTConsumer(std::shared_ptr
<IndexDataConsumer
> DataConsumer
,
109 const IndexingOptions
&Opts
,
110 std::shared_ptr
<Preprocessor
> PP
,
111 std::function
<bool(const Decl
*)> ShouldSkipFunctionBody
)
112 : DataConsumer(std::move(DataConsumer
)),
113 IndexCtx(new IndexingContext(Opts
, *this->DataConsumer
)),
115 ShouldSkipFunctionBody(std::move(ShouldSkipFunctionBody
)) {
116 assert(this->DataConsumer
!= nullptr);
117 assert(this->PP
!= nullptr);
121 void Initialize(ASTContext
&Context
) override
{
122 IndexCtx
->setASTContext(Context
);
123 IndexCtx
->getDataConsumer().initialize(Context
);
124 IndexCtx
->getDataConsumer().setPreprocessor(PP
);
125 PP
->addPPCallbacks(std::make_unique
<IndexPPCallbacks
>(IndexCtx
));
128 bool HandleTopLevelDecl(DeclGroupRef DG
) override
{
129 return IndexCtx
->indexDeclGroupRef(DG
);
132 void HandleInterestingDecl(DeclGroupRef DG
) override
{
133 // Ignore deserialized decls.
136 void HandleTopLevelDeclInObjCContainer(DeclGroupRef DG
) override
{
137 IndexCtx
->indexDeclGroupRef(DG
);
140 void HandleTranslationUnit(ASTContext
&Ctx
) override
{
141 DataConsumer
->finish();
144 bool shouldSkipFunctionBody(Decl
*D
) override
{
145 return ShouldSkipFunctionBody(D
);
149 class IndexAction final
: public ASTFrontendAction
{
150 std::shared_ptr
<IndexDataConsumer
> DataConsumer
;
151 IndexingOptions Opts
;
154 IndexAction(std::shared_ptr
<IndexDataConsumer
> DataConsumer
,
155 const IndexingOptions
&Opts
)
156 : DataConsumer(std::move(DataConsumer
)), Opts(Opts
) {
157 assert(this->DataConsumer
!= nullptr);
161 std::unique_ptr
<ASTConsumer
> CreateASTConsumer(CompilerInstance
&CI
,
162 StringRef InFile
) override
{
163 return std::make_unique
<IndexASTConsumer
>(
164 DataConsumer
, Opts
, CI
.getPreprocessorPtr(),
165 /*ShouldSkipFunctionBody=*/[](const Decl
*) { return false; });
169 } // anonymous namespace
171 std::unique_ptr
<ASTConsumer
> index::createIndexingASTConsumer(
172 std::shared_ptr
<IndexDataConsumer
> DataConsumer
,
173 const IndexingOptions
&Opts
, std::shared_ptr
<Preprocessor
> PP
,
174 std::function
<bool(const Decl
*)> ShouldSkipFunctionBody
) {
175 return std::make_unique
<IndexASTConsumer
>(DataConsumer
, Opts
, PP
,
176 ShouldSkipFunctionBody
);
179 std::unique_ptr
<ASTConsumer
> clang::index::createIndexingASTConsumer(
180 std::shared_ptr
<IndexDataConsumer
> DataConsumer
,
181 const IndexingOptions
&Opts
, std::shared_ptr
<Preprocessor
> PP
) {
182 std::function
<bool(const Decl
*)> ShouldSkipFunctionBody
= [](const Decl
*) {
185 if (Opts
.ShouldTraverseDecl
)
186 ShouldSkipFunctionBody
=
187 [ShouldTraverseDecl(Opts
.ShouldTraverseDecl
)](const Decl
*D
) {
188 return !ShouldTraverseDecl(D
);
190 return createIndexingASTConsumer(std::move(DataConsumer
), Opts
, std::move(PP
),
191 std::move(ShouldSkipFunctionBody
));
194 std::unique_ptr
<FrontendAction
>
195 index::createIndexingAction(std::shared_ptr
<IndexDataConsumer
> DataConsumer
,
196 const IndexingOptions
&Opts
) {
197 assert(DataConsumer
!= nullptr);
198 return std::make_unique
<IndexAction
>(std::move(DataConsumer
), Opts
);
201 static bool topLevelDeclVisitor(void *context
, const Decl
*D
) {
202 IndexingContext
&IndexCtx
= *static_cast<IndexingContext
*>(context
);
203 return IndexCtx
.indexTopLevelDecl(D
);
206 static void indexTranslationUnit(ASTUnit
&Unit
, IndexingContext
&IndexCtx
) {
207 Unit
.visitLocalTopLevelDecls(&IndexCtx
, topLevelDeclVisitor
);
210 static void indexPreprocessorMacro(const IdentifierInfo
*II
,
212 MacroDirective::Kind DirectiveKind
,
214 IndexDataConsumer
&DataConsumer
) {
215 // When using modules, it may happen that we find #undef of a macro that
216 // was defined in another module. In such case, MI may be nullptr, since
217 // we only look for macro definitions in the current TU. In that case,
218 // there is nothing to index.
222 // Skip implicit visibility change.
223 if (DirectiveKind
== MacroDirective::MD_Visibility
)
226 auto Role
= DirectiveKind
== MacroDirective::MD_Define
227 ? SymbolRole::Definition
228 : SymbolRole::Undefinition
;
229 DataConsumer
.handleMacroOccurrence(II
, MI
, static_cast<unsigned>(Role
), Loc
);
232 static void indexPreprocessorMacros(Preprocessor
&PP
,
233 IndexDataConsumer
&DataConsumer
) {
234 for (const auto &M
: PP
.macros()) {
235 for (auto *MD
= M
.second
.getLatest(); MD
; MD
= MD
->getPrevious()) {
236 indexPreprocessorMacro(M
.first
, MD
->getMacroInfo(), MD
->getKind(),
237 MD
->getLocation(), DataConsumer
);
242 static void indexPreprocessorModuleMacros(Preprocessor
&PP
,
243 serialization::ModuleFile
&Mod
,
244 IndexDataConsumer
&DataConsumer
) {
245 for (const auto &M
: PP
.macros()) {
246 if (M
.second
.getLatest() == nullptr) {
247 for (auto *MM
: PP
.getLeafModuleMacros(M
.first
)) {
248 auto *OwningMod
= MM
->getOwningModule();
249 if (OwningMod
&& OwningMod
->getASTFile() == Mod
.File
) {
250 if (auto *MI
= MM
->getMacroInfo()) {
251 indexPreprocessorMacro(M
.first
, MI
, MacroDirective::MD_Define
,
252 MI
->getDefinitionLoc(), DataConsumer
);
260 void index::indexASTUnit(ASTUnit
&Unit
, IndexDataConsumer
&DataConsumer
,
261 IndexingOptions Opts
) {
262 IndexingContext
IndexCtx(Opts
, DataConsumer
);
263 IndexCtx
.setASTContext(Unit
.getASTContext());
264 DataConsumer
.initialize(Unit
.getASTContext());
265 DataConsumer
.setPreprocessor(Unit
.getPreprocessorPtr());
267 if (Opts
.IndexMacrosInPreprocessor
)
268 indexPreprocessorMacros(Unit
.getPreprocessor(), DataConsumer
);
269 indexTranslationUnit(Unit
, IndexCtx
);
270 DataConsumer
.finish();
273 void index::indexTopLevelDecls(ASTContext
&Ctx
, Preprocessor
&PP
,
274 ArrayRef
<const Decl
*> Decls
,
275 IndexDataConsumer
&DataConsumer
,
276 IndexingOptions Opts
) {
277 IndexingContext
IndexCtx(Opts
, DataConsumer
);
278 IndexCtx
.setASTContext(Ctx
);
280 DataConsumer
.initialize(Ctx
);
282 if (Opts
.IndexMacrosInPreprocessor
)
283 indexPreprocessorMacros(PP
, DataConsumer
);
285 for (const Decl
*D
: Decls
)
286 IndexCtx
.indexTopLevelDecl(D
);
287 DataConsumer
.finish();
290 std::unique_ptr
<PPCallbacks
>
291 index::indexMacrosCallback(IndexDataConsumer
&Consumer
, IndexingOptions Opts
) {
292 return std::make_unique
<IndexPPCallbacks
>(
293 std::make_shared
<IndexingContext
>(Opts
, Consumer
));
296 void index::indexModuleFile(serialization::ModuleFile
&Mod
, ASTReader
&Reader
,
297 IndexDataConsumer
&DataConsumer
,
298 IndexingOptions Opts
) {
299 ASTContext
&Ctx
= Reader
.getContext();
300 IndexingContext
IndexCtx(Opts
, DataConsumer
);
301 IndexCtx
.setASTContext(Ctx
);
302 DataConsumer
.initialize(Ctx
);
304 if (Opts
.IndexMacrosInPreprocessor
) {
305 indexPreprocessorModuleMacros(Reader
.getPreprocessor(), Mod
, DataConsumer
);
308 for (const Decl
*D
: Reader
.getModuleFileLevelDecls(Mod
)) {
309 IndexCtx
.indexTopLevelDecl(D
);
311 DataConsumer
.finish();