1 | |
2 | |
3 | |
4 | |
5 | |
6 | |
7 | |
8 | |
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" |
19 | #include <memory> |
20 | |
21 | using namespace clang; |
22 | using namespace clang::index; |
23 | |
24 | bool IndexDataConsumer::handleDeclOccurence(const Decl *D, SymbolRoleSet Roles, |
25 | ArrayRef<SymbolRelation> Relations, |
26 | SourceLocation Loc, |
27 | ASTNodeInfo ASTNode) { |
28 | return true; |
29 | } |
30 | |
31 | bool IndexDataConsumer::handleMacroOccurence(const IdentifierInfo *Name, |
32 | const MacroInfo *MI, |
33 | SymbolRoleSet Roles, |
34 | SourceLocation Loc) { |
35 | return true; |
36 | } |
37 | |
38 | bool IndexDataConsumer::handleModuleOccurence(const ImportDecl *ImportD, |
39 | const Module *Mod, |
40 | SymbolRoleSet Roles, |
41 | SourceLocation Loc) { |
42 | return true; |
43 | } |
44 | |
45 | namespace { |
46 | |
47 | class IndexASTConsumer : public ASTConsumer { |
48 | std::shared_ptr<Preprocessor> PP; |
49 | std::shared_ptr<IndexingContext> IndexCtx; |
50 | |
51 | public: |
52 | IndexASTConsumer(std::shared_ptr<Preprocessor> PP, |
53 | std::shared_ptr<IndexingContext> IndexCtx) |
54 | : PP(std::move(PP)), IndexCtx(std::move(IndexCtx)) {} |
55 | |
56 | protected: |
57 | void Initialize(ASTContext &Context) override { |
58 | IndexCtx->setASTContext(Context); |
59 | IndexCtx->getDataConsumer().initialize(Context); |
60 | IndexCtx->getDataConsumer().setPreprocessor(PP); |
61 | } |
62 | |
63 | bool HandleTopLevelDecl(DeclGroupRef DG) override { |
64 | return IndexCtx->indexDeclGroupRef(DG); |
65 | } |
66 | |
67 | void HandleInterestingDecl(DeclGroupRef DG) override { |
68 | |
69 | } |
70 | |
71 | void HandleTopLevelDeclInObjCContainer(DeclGroupRef DG) override { |
72 | IndexCtx->indexDeclGroupRef(DG); |
73 | } |
74 | |
75 | void HandleTranslationUnit(ASTContext &Ctx) override { |
76 | } |
77 | }; |
78 | |
79 | class IndexPPCallbacks : public PPCallbacks { |
80 | std::shared_ptr<IndexingContext> IndexCtx; |
81 | |
82 | public: |
83 | IndexPPCallbacks(std::shared_ptr<IndexingContext> IndexCtx) |
84 | : IndexCtx(std::move(IndexCtx)) {} |
85 | |
86 | void MacroExpands(const Token &MacroNameTok, const MacroDefinition &MD, |
87 | SourceRange Range, const MacroArgs *Args) override { |
88 | IndexCtx->handleMacroReference(*MacroNameTok.getIdentifierInfo(), |
89 | Range.getBegin(), *MD.getMacroInfo()); |
90 | } |
91 | |
92 | void MacroDefined(const Token &MacroNameTok, |
93 | const MacroDirective *MD) override { |
94 | IndexCtx->handleMacroDefined(*MacroNameTok.getIdentifierInfo(), |
95 | MacroNameTok.getLocation(), |
96 | *MD->getMacroInfo()); |
97 | } |
98 | |
99 | void MacroUndefined(const Token &MacroNameTok, const MacroDefinition &MD, |
100 | const MacroDirective *Undef) override { |
101 | if (!MD.getMacroInfo()) |
102 | return; |
103 | IndexCtx->handleMacroUndefined(*MacroNameTok.getIdentifierInfo(), |
104 | MacroNameTok.getLocation(), |
105 | *MD.getMacroInfo()); |
106 | } |
107 | }; |
108 | |
109 | class IndexActionBase { |
110 | protected: |
111 | std::shared_ptr<IndexDataConsumer> DataConsumer; |
112 | std::shared_ptr<IndexingContext> IndexCtx; |
113 | |
114 | IndexActionBase(std::shared_ptr<IndexDataConsumer> dataConsumer, |
115 | IndexingOptions Opts) |
116 | : DataConsumer(std::move(dataConsumer)), |
117 | IndexCtx(new IndexingContext(Opts, *DataConsumer)) {} |
118 | |
119 | std::unique_ptr<IndexASTConsumer> |
120 | createIndexASTConsumer(CompilerInstance &CI) { |
121 | return llvm::make_unique<IndexASTConsumer>(CI.getPreprocessorPtr(), |
122 | IndexCtx); |
123 | } |
124 | |
125 | std::unique_ptr<PPCallbacks> createIndexPPCallbacks() { |
126 | return llvm::make_unique<IndexPPCallbacks>(IndexCtx); |
127 | } |
128 | |
129 | void finish() { |
130 | DataConsumer->finish(); |
131 | } |
132 | }; |
133 | |
134 | class IndexAction : public ASTFrontendAction, IndexActionBase { |
135 | public: |
136 | IndexAction(std::shared_ptr<IndexDataConsumer> DataConsumer, |
137 | IndexingOptions Opts) |
138 | : IndexActionBase(std::move(DataConsumer), Opts) {} |
139 | |
140 | protected: |
141 | std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI, |
142 | StringRef InFile) override { |
143 | return createIndexASTConsumer(CI); |
144 | } |
145 | |
146 | bool BeginSourceFileAction(clang::CompilerInstance &CI) override { |
147 | CI.getPreprocessor().addPPCallbacks(createIndexPPCallbacks()); |
148 | return true; |
149 | } |
150 | |
151 | void EndSourceFileAction() override { |
152 | FrontendAction::EndSourceFileAction(); |
153 | finish(); |
154 | } |
155 | }; |
156 | |
157 | class WrappingIndexAction : public WrapperFrontendAction, IndexActionBase { |
158 | bool IndexActionFailed = false; |
159 | |
160 | public: |
161 | WrappingIndexAction(std::unique_ptr<FrontendAction> WrappedAction, |
162 | std::shared_ptr<IndexDataConsumer> DataConsumer, |
163 | IndexingOptions Opts) |
164 | : WrapperFrontendAction(std::move(WrappedAction)), |
165 | IndexActionBase(std::move(DataConsumer), Opts) {} |
166 | |
167 | protected: |
168 | std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI, |
169 | StringRef InFile) override { |
170 | auto OtherConsumer = WrapperFrontendAction::CreateASTConsumer(CI, InFile); |
171 | if (!OtherConsumer) { |
172 | IndexActionFailed = true; |
173 | return nullptr; |
174 | } |
175 | |
176 | std::vector<std::unique_ptr<ASTConsumer>> Consumers; |
177 | Consumers.push_back(std::move(OtherConsumer)); |
178 | Consumers.push_back(createIndexASTConsumer(CI)); |
179 | return llvm::make_unique<MultiplexConsumer>(std::move(Consumers)); |
180 | } |
181 | |
182 | bool BeginSourceFileAction(clang::CompilerInstance &CI) override { |
183 | WrapperFrontendAction::BeginSourceFileAction(CI); |
184 | CI.getPreprocessor().addPPCallbacks(createIndexPPCallbacks()); |
185 | return true; |
186 | } |
187 | |
188 | void EndSourceFileAction() override { |
189 | |
190 | WrapperFrontendAction::EndSourceFileAction(); |
191 | if (!IndexActionFailed) |
192 | finish(); |
193 | } |
194 | }; |
195 | |
196 | } |
197 | |
198 | std::unique_ptr<FrontendAction> |
199 | index::createIndexingAction(std::shared_ptr<IndexDataConsumer> DataConsumer, |
200 | IndexingOptions Opts, |
201 | std::unique_ptr<FrontendAction> WrappedAction) { |
202 | if (WrappedAction) |
203 | return llvm::make_unique<WrappingIndexAction>(std::move(WrappedAction), |
204 | std::move(DataConsumer), |
205 | Opts); |
206 | return llvm::make_unique<IndexAction>(std::move(DataConsumer), Opts); |
207 | } |
208 | |
209 | static bool topLevelDeclVisitor(void *context, const Decl *D) { |
210 | IndexingContext &IndexCtx = *static_cast<IndexingContext*>(context); |
211 | return IndexCtx.indexTopLevelDecl(D); |
212 | } |
213 | |
214 | static void (ASTUnit &Unit, IndexingContext &IndexCtx) { |
215 | Unit.visitLocalTopLevelDecls(&IndexCtx, topLevelDeclVisitor); |
216 | } |
217 | |
218 | static void indexPreprocessorMacros(const Preprocessor &PP, |
219 | IndexDataConsumer &DataConsumer) { |
220 | for (const auto &M : PP.macros()) |
221 | if (MacroDirective *MD = M.second.getLatest()) |
222 | DataConsumer.handleMacroOccurence( |
223 | M.first, MD->getMacroInfo(), |
224 | static_cast<unsigned>(index::SymbolRole::Definition), |
225 | MD->getLocation()); |
226 | } |
227 | |
228 | void index::indexASTUnit(ASTUnit &Unit, IndexDataConsumer &DataConsumer, |
229 | IndexingOptions Opts) { |
230 | IndexingContext IndexCtx(Opts, DataConsumer); |
231 | IndexCtx.setASTContext(Unit.getASTContext()); |
232 | DataConsumer.initialize(Unit.getASTContext()); |
233 | DataConsumer.setPreprocessor(Unit.getPreprocessorPtr()); |
234 | |
235 | if (Opts.IndexMacrosInPreprocessor) |
236 | indexPreprocessorMacros(Unit.getPreprocessor(), DataConsumer); |
237 | indexTranslationUnit(Unit, IndexCtx); |
238 | DataConsumer.finish(); |
239 | } |
240 | |
241 | void index::indexTopLevelDecls(ASTContext &Ctx, Preprocessor &PP, |
242 | ArrayRef<const Decl *> Decls, |
243 | IndexDataConsumer &DataConsumer, |
244 | IndexingOptions Opts) { |
245 | IndexingContext IndexCtx(Opts, DataConsumer); |
246 | IndexCtx.setASTContext(Ctx); |
247 | |
248 | DataConsumer.initialize(Ctx); |
249 | |
250 | if (Opts.IndexMacrosInPreprocessor) |
251 | indexPreprocessorMacros(PP, DataConsumer); |
252 | |
253 | for (const Decl *D : Decls) |
254 | IndexCtx.indexTopLevelDecl(D); |
255 | DataConsumer.finish(); |
256 | } |
257 | |
258 | std::unique_ptr<PPCallbacks> |
259 | index::indexMacrosCallback(IndexDataConsumer &Consumer, IndexingOptions Opts) { |
260 | return llvm::make_unique<IndexPPCallbacks>( |
261 | std::make_shared<IndexingContext>(Opts, Consumer)); |
262 | } |
263 | |
264 | void index::indexModuleFile(serialization::ModuleFile &Mod, ASTReader &Reader, |
265 | IndexDataConsumer &DataConsumer, |
266 | IndexingOptions Opts) { |
267 | ASTContext &Ctx = Reader.getContext(); |
268 | IndexingContext IndexCtx(Opts, DataConsumer); |
269 | IndexCtx.setASTContext(Ctx); |
270 | DataConsumer.initialize(Ctx); |
271 | |
272 | if (Opts.IndexMacrosInPreprocessor) |
273 | indexPreprocessorMacros(Reader.getPreprocessor(), DataConsumer); |
274 | |
275 | for (const Decl *D : Reader.getModuleFileLevelDecls(Mod)) { |
276 | IndexCtx.indexTopLevelDecl(D); |
277 | } |
278 | DataConsumer.finish(); |
279 | } |
280 | |