blob: dcf765409745241be7a5e9635f947debe915b450 [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);
96 ASTLocation VisitVarDecl(VarDecl *D);
97 ASTLocation VisitFunctionDecl(FunctionDecl *D);
98 ASTLocation VisitDecl(Decl *D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +000099};
100
101} // anonymous namespace
102
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000103ASTLocation StmtLocResolver::VisitDeclStmt(DeclStmt *Node) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000104 assert(ContainsLocation(Node) &&
105 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000106
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000107 // Search all declarations of this DeclStmt.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000108 for (DeclStmt::decl_iterator
109 I = Node->decl_begin(), E = Node->decl_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000110 RangePos RP = CheckRange(*I);
111 if (RP == AfterLoc)
112 break;
113 if (RP == ContainsLoc)
114 return DeclLocResolver(Ctx, Loc).Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000115 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000116
117 return ASTLocation(Parent, Node);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000118}
119
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000120ASTLocation StmtLocResolver::VisitStmt(Stmt *Node) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000121 assert(ContainsLocation(Node) &&
122 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000123
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000124 // Search the child statements.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000125 for (Stmt::child_iterator
126 I = Node->child_begin(), E = Node->child_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000127 RangePos RP = CheckRange(*I);
128 if (RP == AfterLoc)
129 break;
130 if (RP == ContainsLoc)
131 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000132 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000133
134 return ASTLocation(Parent, Node);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000135}
136
Argiris Kirtzidis875544f2009-07-14 03:18:17 +0000137ASTLocation DeclLocResolver::VisitDeclContext(DeclContext *DC) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000138 for (DeclContext::decl_iterator
Argiris Kirtzidisab6e38a2009-06-30 02:36:12 +0000139 I = DC->decls_begin(), E = DC->decls_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000140 RangePos RP = CheckRange(*I);
141 if (RP == AfterLoc)
142 break;
143 if (RP == ContainsLoc)
144 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000145 }
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000146
Argiris Kirtzidis875544f2009-07-14 03:18:17 +0000147 return ASTLocation(cast<Decl>(DC));
148}
149
150ASTLocation DeclLocResolver::VisitTranslationUnitDecl(TranslationUnitDecl *TU) {
151 ASTLocation ASTLoc = VisitDeclContext(TU);
152 if (ASTLoc.getDecl() == TU)
153 return ASTLocation();
154 return ASTLoc;
155}
156
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000157ASTLocation DeclLocResolver::VisitFunctionDecl(FunctionDecl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000158 assert(ContainsLocation(D) &&
159 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000160
161 // First, search through the parameters of the function.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000162 for (FunctionDecl::param_iterator
163 I = D->param_begin(), E = D->param_end(); I != E; ++I) {
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000164 RangePos RP = CheckRange(*I);
165 if (RP == AfterLoc)
166 return ASTLocation(D);
167 if (RP == ContainsLoc)
168 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000169 }
170
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000171 // We didn't find the location in the parameters and we didn't get passed it.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000172
Argiris Kirtzidis4770f5e2009-07-05 22:21:17 +0000173 if (!D->isThisDeclarationADefinition())
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000174 return ASTLocation(D);
Argiris Kirtzidis4770f5e2009-07-05 22:21:17 +0000175
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000176 // Second, search through the declarations that are part of the function.
177 // If we find he location there, we won't have to search through its body.
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000178
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000179 for (DeclContext::decl_iterator
180 I = D->decls_begin(), E = D->decls_end(); I != E; ++I) {
181 if (isa<ParmVarDecl>(*I))
182 continue; // We already searched through the parameters.
183
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000184 RangePos RP = CheckRange(*I);
185 if (RP == AfterLoc)
186 break;
187 if (RP == ContainsLoc)
188 return Visit(*I);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000189 }
Argiris Kirtzidis7283e462009-07-05 22:21:46 +0000190
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000191 // We didn't find a declaration that corresponds to the source location.
192
193 // Finally, search through the body of the function.
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000194 Stmt *Body = D->getBody();
195 assert(Body && "Expected definition");
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000196 assert(!isBeforeLocation(Body) &&
197 "This function is supposed to contain the loc");
198 if (isAfterLocation(Body))
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000199 return ASTLocation(D);
200
201 // The body contains the location.
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000202 assert(ContainsLocation(Body));
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000203 return StmtLocResolver(Ctx, Loc, D).Visit(Body);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000204}
205
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000206ASTLocation DeclLocResolver::VisitVarDecl(VarDecl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000207 assert(ContainsLocation(D) &&
208 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000209
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000210 // Check whether the location points to the init expression.
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000211 Expr *Init = D->getInit();
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000212 if (Init && ContainsLocation(Init))
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000213 return StmtLocResolver(Ctx, Loc, D).Visit(Init);
214
215 return ASTLocation(D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000216}
217
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000218ASTLocation DeclLocResolver::VisitDecl(Decl *D) {
Argiris Kirtzidis8be9dda2009-07-10 03:41:26 +0000219 assert(ContainsLocation(D) &&
220 "Should visit only after verifying that loc is in range");
Argiris Kirtzidis62ee8c62009-07-18 00:33:40 +0000221 if (DeclContext *DC = dyn_cast<DeclContext>(D))
222 return VisitDeclContext(DC);
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000223 return ASTLocation(D);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000224}
225
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000226LocResolverBase::RangePos LocResolverBase::CheckRange(SourceRange Range) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000227 if (!Range.isValid())
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000228 return BeforeLoc; // Keep looking.
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000229
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000230 // Update the end source range to cover the full length of the token
231 // positioned at the end of the source range.
232 //
233 // e.g.,
234 // int foo
235 // ^ ^
236 //
237 // will be updated to
238 // int foo
239 // ^ ^
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000240 unsigned TokSize = Lexer::MeasureTokenLength(Range.getEnd(),
241 Ctx.getSourceManager(),
242 Ctx.getLangOptions());
243 Range.setEnd(Range.getEnd().getFileLocWithOffset(TokSize-1));
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000244
245 SourceManager &SourceMgr = Ctx.getSourceManager();
246 if (SourceMgr.isBeforeInTranslationUnit(Range.getEnd(), Loc))
247 return BeforeLoc;
248
249 if (SourceMgr.isBeforeInTranslationUnit(Loc, Range.getBegin()))
250 return AfterLoc;
251
252 return ContainsLoc;
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000253}
254
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000255#ifndef NDEBUG
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000256void LocResolverBase::print(Decl *D) {
257 llvm::raw_ostream &OS = llvm::outs();
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000258 OS << "#### DECL " << D->getDeclKindName() << " ####\n";
Argiris Kirtzidis9a6fb962009-06-30 02:35:04 +0000259 D->print(OS);
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000260 OS << " <";
261 D->getLocStart().print(OS, Ctx.getSourceManager());
262 OS << " > - <";
263 D->getLocEnd().print(OS, Ctx.getSourceManager());
264 OS << ">\n\n";
265 OS.flush();
266}
267
268void LocResolverBase::print(Stmt *Node) {
269 llvm::raw_ostream &OS = llvm::outs();
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000270 OS << "#### STMT " << Node->getStmtClassName() << " ####\n";
Chris Lattner7099c782009-06-30 01:26:17 +0000271 Node->printPretty(OS, Ctx, 0, PrintingPolicy(Ctx.getLangOptions()));
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000272 OS << " <";
273 Node->getLocStart().print(OS, Ctx.getSourceManager());
274 OS << " > - <";
275 Node->getLocEnd().print(OS, Ctx.getSourceManager());
276 OS << ">\n\n";
277 OS.flush();
278}
Argiris Kirtzidise05ecfb2009-07-17 01:20:03 +0000279#endif
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000280
281
282/// \brief Returns the AST node that a source location points to.
283///
Argiris Kirtzidis98eec712009-07-06 21:35:02 +0000284ASTLocation idx::ResolveLocationInAST(ASTContext &Ctx, SourceLocation Loc) {
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000285 if (Loc.isInvalid())
Argiris Kirtzidis2fcbc1e2009-07-06 21:34:20 +0000286 return ASTLocation();
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000287
Argiris Kirtzidis3ed07532009-07-07 00:53:31 +0000288 return DeclLocResolver(Ctx, Loc).Visit(Ctx.getTranslationUnitDecl());
Argiris Kirtzidis40bf2832009-06-25 18:22:41 +0000289}