blob: 4bff08e2ea7fdc826b39ba374505622094d21aa8 [file] [log] [blame]
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +00001//===--- ResolveLocation.cpp - Source location resolver ---------*- C++ -*-===//
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// This defines the ResolveLocationInAST function, which resolves a
Argiris Kirtzidis98eec712009-07-06 21:35:02 +000011// source location into a ASTLocation.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000012//
13//===----------------------------------------------------------------------===//
14
Argiris Kirtzidis98eec712009-07-06 21:35:02 +000015#include "clang/Index/Utils.h"
Argiris Kirtzidis75abd9c2009-07-06 21:34:47 +000016#include "clang/Index/ASTLocation.h"
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000017#include "clang/AST/DeclVisitor.h"
18#include "clang/AST/StmtVisitor.h"
19#include "clang/Lex/Lexer.h"
20#include "clang/Basic/SourceManager.h"
21#include "llvm/Support/Compiler.h"
22using namespace clang;
Argiris Kirtzidis75abd9c2009-07-06 21:34:47 +000023using namespace idx;
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000024
25namespace {
26
27/// \brief Base for the LocResolver classes. Mostly does source range checking.
28class VISIBILITY_HIDDEN LocResolverBase {
29protected:
30 ASTContext &Ctx;
31 SourceLocation Loc;
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000032
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000033 enum RangePos {
34 BeforeLoc,
35 ContainsLoc,
36 AfterLoc
37 };
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000038
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000039 RangePos CheckRange(SourceRange Range);
40 RangePos CheckRange(Decl *D) { return CheckRange(D->getSourceRange()); }
41 RangePos CheckRange(Stmt *Node) { return CheckRange(Node->getSourceRange()); }
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000042
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +000043 template <typename T>
44 bool isBeforeLocation(T *Node) {
45 return CheckRange(Node) == BeforeLoc;
46 }
47
48 template <typename T>
49 bool ContainsLocation(T *Node) {
50 return CheckRange(Node) == ContainsLoc;
51 }
52
53 template <typename T>
54 bool isAfterLocation(T *Node) {
55 return CheckRange(Node) == AfterLoc;
56 }
57
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000058public:
59 LocResolverBase(ASTContext &ctx, SourceLocation loc)
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000060 : Ctx(ctx), Loc(loc) {}
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000061
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +000062#ifndef NDEBUG
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000063 /// \brief Debugging output.
64 void print(Decl *D);
65 /// \brief Debugging output.
66 void print(Stmt *Node);
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +000067#endif
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000068};
69
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000070/// \brief Searches a statement for the ASTLocation that corresponds to a source
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000071/// location.
72class VISIBILITY_HIDDEN StmtLocResolver : public LocResolverBase,
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000073 public StmtVisitor<StmtLocResolver,
74 ASTLocation > {
75 Decl *Parent;
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000076
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000077public:
78 StmtLocResolver(ASTContext &ctx, SourceLocation loc, Decl *parent)
79 : LocResolverBase(ctx, loc), Parent(parent) {}
80
81 ASTLocation VisitDeclStmt(DeclStmt *Node);
82 ASTLocation VisitStmt(Stmt *Node);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000083};
84
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000085/// \brief Searches a declaration for the ASTLocation that corresponds to a
86/// source location.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000087class VISIBILITY_HIDDEN DeclLocResolver : public LocResolverBase,
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000088 public DeclVisitor<DeclLocResolver,
89 ASTLocation > {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000090public:
91 DeclLocResolver(ASTContext &ctx, SourceLocation loc)
92 : LocResolverBase(ctx, loc) {}
93
Argiris Kirtzidis875544f2009-07-14 03:18:17 +000094 ASTLocation VisitDeclContext(DeclContext *DC);
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000095 ASTLocation VisitTranslationUnitDecl(TranslationUnitDecl *TU);
Argiris Kirtzidis875544f2009-07-14 03:18:17 +000096 ASTLocation VisitRecordDecl(RecordDecl *D);
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +000097 ASTLocation VisitVarDecl(VarDecl *D);
98 ASTLocation VisitFunctionDecl(FunctionDecl *D);
99 ASTLocation VisitDecl(Decl *D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000100};
101
102} // anonymous namespace
103
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000104ASTLocation StmtLocResolver::VisitDeclStmt(DeclStmt *Node) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000105 assert(ContainsLocation(Node) &&
106 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000107
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000108 // Search all declarations of this DeclStmt.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000109 for (DeclStmt::decl_iterator
110 I = Node->decl_begin(), E = Node->decl_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000111 RangePos RP = CheckRange(*I);
112 if (RP == AfterLoc)
113 break;
114 if (RP == ContainsLoc)
115 return DeclLocResolver(Ctx, Loc).Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000116 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000117
118 return ASTLocation(Parent, Node);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000119}
120
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000121ASTLocation StmtLocResolver::VisitStmt(Stmt *Node) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000122 assert(ContainsLocation(Node) &&
123 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000124
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000125 // Search the child statements.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000126 for (Stmt::child_iterator
127 I = Node->child_begin(), E = Node->child_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000128 RangePos RP = CheckRange(*I);
129 if (RP == AfterLoc)
130 break;
131 if (RP == ContainsLoc)
132 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000133 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000134
135 return ASTLocation(Parent, Node);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000136}
137
Argiris Kirtzidis875544f2009-07-14 03:18:17 +0000138ASTLocation DeclLocResolver::VisitDeclContext(DeclContext *DC) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000139 for (DeclContext::decl_iterator
Argiris Kirtzidisab6e38a2009-06-30 02:36:12 +0000140 I = DC->decls_begin(), E = DC->decls_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000141 RangePos RP = CheckRange(*I);
142 if (RP == AfterLoc)
143 break;
144 if (RP == ContainsLoc)
145 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000146 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000147
Argiris Kirtzidis875544f2009-07-14 03:18:17 +0000148 return ASTLocation(cast<Decl>(DC));
149}
150
151ASTLocation DeclLocResolver::VisitTranslationUnitDecl(TranslationUnitDecl *TU) {
152 ASTLocation ASTLoc = VisitDeclContext(TU);
153 if (ASTLoc.getDecl() == TU)
154 return ASTLocation();
155 return ASTLoc;
156}
157
158ASTLocation DeclLocResolver::VisitRecordDecl(RecordDecl *D) {
159 assert(ContainsLocation(D) &&
160 "Should visit only after verifying that loc is in range");
161 return VisitDeclContext(D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000162}
163
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000164ASTLocation DeclLocResolver::VisitFunctionDecl(FunctionDecl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000165 assert(ContainsLocation(D) &&
166 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000167
168 // First, search through the parameters of the function.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000169 for (FunctionDecl::param_iterator
170 I = D->param_begin(), E = D->param_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000171 RangePos RP = CheckRange(*I);
172 if (RP == AfterLoc)
173 return ASTLocation(D);
174 if (RP == ContainsLoc)
175 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000176 }
177
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000178 // We didn't find the location in the parameters and we didn't get passed it.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000179
Argiris Kirtzidis4770f5e2009-07-05 22:21:17 +0000180 if (!D->isThisDeclarationADefinition())
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000181 return ASTLocation(D);
Argiris Kirtzidis4770f5e2009-07-05 22:21:17 +0000182
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000183 // Second, search through the declarations that are part of the function.
184 // If we find he location there, we won't have to search through its body.
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000185
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000186 for (DeclContext::decl_iterator
187 I = D->decls_begin(), E = D->decls_end(); I != E; ++I) {
188 if (isa<ParmVarDecl>(*I))
189 continue; // We already searched through the parameters.
190
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000191 RangePos RP = CheckRange(*I);
192 if (RP == AfterLoc)
193 break;
194 if (RP == ContainsLoc)
195 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000196 }
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000197
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000198 // We didn't find a declaration that corresponds to the source location.
199
200 // Finally, search through the body of the function.
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000201 Stmt *Body = D->getBody();
202 assert(Body && "Expected definition");
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000203 assert(!isBeforeLocation(Body) &&
204 "This function is supposed to contain the loc");
205 if (isAfterLocation(Body))
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000206 return ASTLocation(D);
207
208 // The body contains the location.
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000209 assert(ContainsLocation(Body));
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000210 return StmtLocResolver(Ctx, Loc, D).Visit(Body);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000211}
212
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000213ASTLocation DeclLocResolver::VisitVarDecl(VarDecl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000214 assert(ContainsLocation(D) &&
215 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000216
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000217 // Check whether the location points to the init expression.
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000218 Expr *Init = D->getInit();
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000219 if (Init && ContainsLocation(Init))
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000220 return StmtLocResolver(Ctx, Loc, D).Visit(Init);
221
222 return ASTLocation(D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000223}
224
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000225ASTLocation DeclLocResolver::VisitDecl(Decl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000226 assert(ContainsLocation(D) &&
227 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000228 return ASTLocation(D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000229}
230
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000231LocResolverBase::RangePos LocResolverBase::CheckRange(SourceRange Range) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000232 if (!Range.isValid())
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000233 return BeforeLoc; // Keep looking.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000234
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000235 // Update the end source range to cover the full length of the token
236 // positioned at the end of the source range.
237 //
238 // e.g.,
239 // int foo
240 // ^ ^
241 //
242 // will be updated to
243 // int foo
244 // ^ ^
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000245 unsigned TokSize = Lexer::MeasureTokenLength(Range.getEnd(),
246 Ctx.getSourceManager(),
247 Ctx.getLangOptions());
248 Range.setEnd(Range.getEnd().getFileLocWithOffset(TokSize-1));
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000249
250 SourceManager &SourceMgr = Ctx.getSourceManager();
251 if (SourceMgr.isBeforeInTranslationUnit(Range.getEnd(), Loc))
252 return BeforeLoc;
253
254 if (SourceMgr.isBeforeInTranslationUnit(Loc, Range.getBegin()))
255 return AfterLoc;
256
257 return ContainsLoc;
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000258}
259
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000260#ifndef NDEBUG
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000261void LocResolverBase::print(Decl *D) {
262 llvm::raw_ostream &OS = llvm::outs();
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000263 OS << "#### DECL " << D->getDeclKindName() << " ####\n";
Argiris Kirtzidis9a6fb962009-06-30 02:35:04 +0000264 D->print(OS);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000265 OS << " <";
266 D->getLocStart().print(OS, Ctx.getSourceManager());
267 OS << " > - <";
268 D->getLocEnd().print(OS, Ctx.getSourceManager());
269 OS << ">\n\n";
270 OS.flush();
271}
272
273void LocResolverBase::print(Stmt *Node) {
274 llvm::raw_ostream &OS = llvm::outs();
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000275 OS << "#### STMT " << Node->getStmtClassName() << " ####\n";
Chris Lattner7099c782009-06-30 01:26:17 +0000276 Node->printPretty(OS, Ctx, 0, PrintingPolicy(Ctx.getLangOptions()));
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000277 OS << " <";
278 Node->getLocStart().print(OS, Ctx.getSourceManager());
279 OS << " > - <";
280 Node->getLocEnd().print(OS, Ctx.getSourceManager());
281 OS << ">\n\n";
282 OS.flush();
283}
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000284#endif
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000285
286
287/// \brief Returns the AST node that a source location points to.
288///
Argiris Kirtzidis98eec712009-07-06 21:35:02 +0000289ASTLocation idx::ResolveLocationInAST(ASTContext &Ctx, SourceLocation Loc) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000290 if (Loc.isInvalid())
Argiris Kirtzidis2fcbc1e2009-07-06 21:34:20 +0000291 return ASTLocation();
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000292
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000293 return DeclLocResolver(Ctx, Loc).Visit(Ctx.getTranslationUnitDecl());
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000294}