CIndex.cpp revision b49e728a4d1a84b72f3aebf60ff494684f9cb004
1d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//===- CIndex.cpp - Clang-C Source Indexing Library -----------------------===//
2d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//
3d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//                     The LLVM Compiler Infrastructure
4d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//
5d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek// This file is distributed under the University of Illinois Open Source
6d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek// License. See LICENSE.TXT for details.
70d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar//
8d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//===----------------------------------------------------------------------===//
9d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//
10ab1889321f6f8f200f2b318ac26883ac18e49d03Ted Kremenek// This file implements the main API hooks in the Clang-C Source Indexing
11ab1889321f6f8f200f2b318ac26883ac18e49d03Ted Kremenek// library.
12d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//
13d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek//===----------------------------------------------------------------------===//
14d2fa56687f8bd5ac6ebf9d9468d0efd714986a54Ted Kremenek
15ab1889321f6f8f200f2b318ac26883ac18e49d03Ted Kremenek#include "CIndexer.h"
1616c440a377b7ec8b722a2e2c7c864f75c95bd305Ted Kremenek#include "CXCursor.h"
170a90d32523bfe5fa63e11b648686c9699f786d15Ted Kremenek#include "CXTranslationUnit.h"
18ed122735639d83c10f18c28c7fd117bfcd0f62cbTed Kremenek#include "CXString.h"
1995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek#include "CXType.h"
20a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek#include "CXSourceLocation.h"
215352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor#include "CIndexDiagnostic.h"
22e397bf1bd90cfceb0166606ebcd2580b7671a828Argyrios Kyrtzidis#include "CursorVisitor.h"
23ab1889321f6f8f200f2b318ac26883ac18e49d03Ted Kremenek
2404bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek#include "clang/Basic/Version.h"
25936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
26fb5704295c6137685a7b90b92cd6b958028740c8Steve Naroff#include "clang/AST/StmtVisitor.h"
27b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Basic/Diagnostic.h"
28b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Frontend/ASTUnit.h"
29b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Frontend/CompilerInstance.h"
30936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor#include "clang/Frontend/FrontendDiagnostic.h"
31d8210650ed948de65a08a8daf16d291b747717c4Ted Kremenek#include "clang/Lex/Lexer.h"
32dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor#include "clang/Lex/HeaderSearch.h"
33b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Lex/PreprocessingRecord.h"
3433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor#include "clang/Lex/Preprocessor.h"
35a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor#include "llvm/ADT/STLExtras.h"
36d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek#include "llvm/ADT/Optional.h"
37f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor#include "llvm/ADT/StringSwitch.h"
38d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek#include "clang/Analysis/Support/SaveAndRestore.h"
39c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar#include "llvm/Support/CrashRecoveryContext.h"
4048615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar#include "llvm/Support/PrettyStackTrace.h"
4102465750c8c3fa96b1e7e596b02297e24361dc4fDouglas Gregor#include "llvm/Support/MemoryBuffer.h"
42358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor#include "llvm/Support/raw_ostream.h"
437a07fcb8f10fe45ea65a0a41798eb1c40777bde4Douglas Gregor#include "llvm/Support/Timer.h"
4403013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Mutex.h"
4503013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Program.h"
4603013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Signals.h"
4703013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Threading.h"
4837f1ea0eb08a00fa90edbecb427cfbb50ca0f4d0Ted Kremenek#include "llvm/Support/Compiler.h"
49fc0622155fa61349698a8fd0053773c37d9f7ac4Ted Kremenek
5050398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroffusing namespace clang;
5116c440a377b7ec8b722a2e2c7c864f75c95bd305Ted Kremenekusing namespace clang::cxcursor;
52ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenekusing namespace clang::cxstring;
539049cf6cb468c856888e88251dab659955fa767eArgyrios Kyrtzidisusing namespace clang::cxtu;
5450398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
559049cf6cb468c856888e88251dab659955fa767eArgyrios KyrtzidisCXTranslationUnit cxtu::MakeCXTranslationUnit(ASTUnit *TU) {
56a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  if (!TU)
57a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return 0;
58a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit D = new CXTranslationUnitImpl();
59a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  D->TUData = TU;
60a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  D->StringPool = createCXStringPool();
61153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  D->Diagnostics = 0;
62a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return D;
63a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek}
64a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek
654e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidiscxtu::CXTUOwner::~CXTUOwner() {
664e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis  if (TU)
674e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis    clang_disposeTranslationUnit(TU);
684e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis}
694e7064fa7e344e8f87a5b8457e96dfdd252c4a9eArgyrios Kyrtzidis
70f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek/// \brief Compare two source ranges to determine their relative position in
7133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// the translation unit.
72f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekstatic RangeComparisonResult RangeCompare(SourceManager &SM,
73f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                          SourceRange R1,
7433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor                                          SourceRange R2) {
7533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  assert(R1.isValid() && "First range is invalid?");
7633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  assert(R2.isValid() && "Second range is invalid?");
77a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R1.getEnd() != R2.getBegin() &&
78d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar      SM.isBeforeInTranslationUnit(R1.getEnd(), R2.getBegin()))
7933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    return RangeBefore;
80a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R2.getEnd() != R1.getBegin() &&
81d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar      SM.isBeforeInTranslationUnit(R2.getEnd(), R1.getBegin()))
8233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    return RangeAfter;
8333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  return RangeOverlap;
8433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
8533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
86fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek/// \brief Determine if a source location falls within, before, or after a
87fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek///   a given source range.
88fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekstatic RangeComparisonResult LocationCompare(SourceManager &SM,
89fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                             SourceLocation L, SourceRange R) {
90fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  assert(R.isValid() && "First range is invalid?");
91fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  assert(L.isValid() && "Second range is invalid?");
92a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (L == R.getBegin() || L == R.getEnd())
93fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeOverlap;
94fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (SM.isBeforeInTranslationUnit(L, R.getBegin()))
95fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeBefore;
96fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (SM.isBeforeInTranslationUnit(R.getEnd(), L))
97fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeAfter;
98fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  return RangeOverlap;
99fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek}
100fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
10176dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// \brief Translate a Clang source range into a CIndex source range.
10276dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar///
10376dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// Clang internally represents ranges where the end location points to the
10476dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// start of the token at the end. However, for external clients it is more
10576dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// useful to have a CXSourceRange be a proper half-open interval. This routine
10676dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// does the appropriate translation.
107f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed KremenekCXSourceRange cxloc::translateSourceRange(const SourceManager &SM,
10876dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                                          const LangOptions &LangOpts,
1090a76aae8c03cb7dd7bdbe683485560afaf695959Chris Lattner                                          const CharSourceRange &R) {
11076dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  // We want the last character in this location, so we will adjust the
1116a5a23f8e7fb65e028c8092bc1d1a1d9dfe2e9bcDouglas Gregor  // location accordingly.
11276dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  SourceLocation EndLoc = R.getEnd();
113a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (EndLoc.isValid() && EndLoc.isMacroID())
114edc3dccece244a584f8ebdb81da6c962c08e79beChandler Carruth    EndLoc = SM.getExpansionRange(EndLoc).second;
1150a76aae8c03cb7dd7bdbe683485560afaf695959Chris Lattner  if (R.isTokenRange() && !EndLoc.isInvalid() && EndLoc.isFileID()) {
1166a5a23f8e7fb65e028c8092bc1d1a1d9dfe2e9bcDouglas Gregor    unsigned Length = Lexer::MeasureTokenLength(EndLoc, SM, LangOpts);
117a64ccefdf0ea4e03ec88805d71b0af74950c7472Argyrios Kyrtzidis    EndLoc = EndLoc.getLocWithOffset(Length);
11876dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  }
11976dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar
12076dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  CXSourceRange Result = { { (void *)&SM, (void *)&LangOpts },
12176dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                           R.getBegin().getRawEncoding(),
12276dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                           EndLoc.getRawEncoding() };
12376dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  return Result;
12476dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar}
1251db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
1268a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek//===----------------------------------------------------------------------===//
12733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor// Cursor visitor.
1288a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek//===----------------------------------------------------------------------===//
1298a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek
130a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorstatic SourceRange getRawCursorExtent(CXCursor C);
1316653798ff5ce6deb58112777e21307ccc453133dDouglas Gregorstatic SourceRange getFullCursorExtent(CXCursor C, SourceManager &SrcMgr);
1326653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
133a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
13433e9abd21083a0191a7676a04b497006d2da184dDouglas GregorRangeComparisonResult CursorVisitor::CompareRegionOfInterest(SourceRange R) {
135a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return RangeCompare(AU->getSourceManager(), R, RegionOfInterest);
13633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
13733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
138b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \brief Visit the given cursor and, if requested by the visitor,
139b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// its children.
140b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor///
14133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// \param Cursor the cursor to visit.
14233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor///
14333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// \param CheckRegionOfInterest if true, then the caller already checked that
14433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// this cursor is within the region of interest.
14533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor///
146b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \returns true if the visitation should be aborted, false if it
147b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// should continue.
14833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregorbool CursorVisitor::Visit(CXCursor Cursor, bool CheckedRegionOfInterest) {
149b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isInvalid(Cursor.kind))
150b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return false;
151f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
152b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isDeclaration(Cursor.kind)) {
153b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Decl *D = getCursorDecl(Cursor);
154b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    assert(D && "Invalid declaration cursor");
15565ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // Ignore implicit declarations, unless it's an objc method because
15665ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // currently we should report implicit methods for properties when indexing.
15765ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    if (D->isImplicit() && !isa<ObjCMethodDecl>(D))
158b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return false;
159b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
1600d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
16133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // If we have a range of interest, and this cursor doesn't intersect with it,
16233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // we're done.
16333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (RegionOfInterest.isValid() && !CheckedRegionOfInterest) {
164a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    SourceRange Range = getRawCursorExtent(Cursor);
165f408f32aa9ae3d97bc656267dc5d78fa7d03499bDaniel Dunbar    if (Range.isInvalid() || CompareRegionOfInterest(Range))
16633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor      return false;
16733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  }
168f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
169b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  switch (Visitor(Cursor, Parent, ClientData)) {
170b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Break:
171b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
1720d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
173b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Continue:
174b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return false;
1752e331b938b38057e333fab0ba841130ea8467794Douglas Gregor
176b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Recurse:
177b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return VisitChildren(Cursor);
178b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
1790d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
180fd64377225a6a140bddb3f997d52a036486f9360Douglas Gregor  return false;
181b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
1820d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
183f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidisstatic bool visitPreprocessedEntitiesInRange(SourceRange R,
184f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                             PreprocessingRecord &PPRec,
185f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                             CursorVisitor &Visitor) {
186f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  SourceManager &SM = Visitor.getASTUnit()->getSourceManager();
187f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  FileID FID;
188f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
189e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis  if (!Visitor.shouldVisitIncludedEntities()) {
190f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    // If the begin/end of the range lie in the same FileID, do the optimization
191f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    // where we skip preprocessed entities that do not come from the same FileID.
192f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    FID = SM.getFileID(R.getBegin());
193f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (FID != SM.getFileID(R.getEnd()))
194f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      FID = FileID();
195f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  }
196f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
197f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  std::pair<PreprocessingRecord::iterator, PreprocessingRecord::iterator>
198f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    Entities = PPRec.getPreprocessedEntitiesInRange(R);
199f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  return Visitor.visitPreprocessedEntities(Entities.first, Entities.second,
200f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                           PPRec, FID);
201f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis}
202f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
203dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidisvoid CursorVisitor::visitFileRegion() {
204dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (RegionOfInterest.isInvalid())
205dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
206dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
207dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  ASTUnit *Unit = static_cast<ASTUnit *>(TU->TUData);
208dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceManager &SM = Unit->getSourceManager();
209dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
210dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  std::pair<FileID, unsigned>
211dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Begin = SM.getDecomposedLoc(SM.getFileLoc(RegionOfInterest.getBegin())),
212dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End = SM.getDecomposedLoc(SM.getFileLoc(RegionOfInterest.getEnd()));
213dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
214dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (End.first != Begin.first) {
215dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    // If the end does not reside in the same file, try to recover by
216dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    // picking the end of the file of begin location.
217dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End.first = Begin.first;
218dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End.second = SM.getFileIDSize(Begin.first);
219dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
220dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
221dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  assert(Begin.first == End.first);
222dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (Begin.second > End.second)
223dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
224dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
225dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  FileID File = Begin.first;
226dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  unsigned Offset = Begin.second;
227dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  unsigned Length = End.second - Begin.second;
228dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
229b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!VisitDeclsOnly && !VisitPreprocessorLast)
230b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis    if (visitPreprocessedEntitiesInRegion())
231b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis      return; // visitation break.
232dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
233dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  visitDeclsFromFileRegion(File, Offset, Length);
234dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
235b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!VisitDeclsOnly && VisitPreprocessorLast)
236dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    visitPreprocessedEntitiesInRegion();
237dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis}
238dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
239e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidisstatic bool isInLexicalContext(Decl *D, DeclContext *DC) {
240e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  if (!DC)
241e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    return false;
242e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
243e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  for (DeclContext *DeclDC = D->getLexicalDeclContext();
244e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis         DeclDC; DeclDC = DeclDC->getLexicalParent()) {
245e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (DeclDC == DC)
246e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      return true;
247e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  }
248e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  return false;
249e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis}
250e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
251dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidisvoid CursorVisitor::visitDeclsFromFileRegion(FileID File,
252dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis                                             unsigned Offset, unsigned Length) {
253dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  ASTUnit *Unit = static_cast<ASTUnit *>(TU->TUData);
254dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceManager &SM = Unit->getSourceManager();
255dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceRange Range = RegionOfInterest;
256dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
257dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SmallVector<Decl *, 16> Decls;
258dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  Unit->findFileRegionDecls(File, Offset, Length, Decls);
259dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
260dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // If we didn't find any file level decls for the file, try looking at the
261dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // file that it was included from.
262c14a03dffff69b5e1c55cc118fc52d8fd9f3a28dArgyrios Kyrtzidis  while (Decls.empty() || Decls.front()->isTopLevelDeclInObjCContainer()) {
263dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    bool Invalid = false;
264dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    const SrcMgr::SLocEntry &SLEntry = SM.getSLocEntry(File, &Invalid);
265dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (Invalid)
266dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      return;
267dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
268dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    SourceLocation Outer;
269dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (SLEntry.isFile())
270dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      Outer = SLEntry.getFile().getIncludeLoc();
271dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    else
272dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      Outer = SLEntry.getExpansion().getExpansionLocStart();
273dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (Outer.isInvalid())
274dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      return;
275dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
276dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    llvm::tie(File, Offset) = SM.getDecomposedExpansionLoc(Outer);
277dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Length = 0;
278dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Unit->findFileRegionDecls(File, Offset, Length, Decls);
279dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
280dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
281dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  assert(!Decls.empty());
282dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
283dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  bool VisitedAtLeastOnce = false;
284e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  DeclContext *CurDC = 0;
285dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SmallVector<Decl *, 16>::iterator DIt = Decls.begin();
286dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  for (SmallVector<Decl *, 16>::iterator DE = Decls.end(); DIt != DE; ++DIt) {
287dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Decl *D = *DIt;
288ed8bef44c0545fd55a78715606f8d733f6498b21Argyrios Kyrtzidis    if (D->getSourceRange().isInvalid())
289ed8bef44c0545fd55a78715606f8d733f6498b21Argyrios Kyrtzidis      continue;
290dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
291e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (isInLexicalContext(D, CurDC))
292e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      continue;
293e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
294e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    CurDC = dyn_cast<DeclContext>(D);
295e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
296dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    // We handle forward decls via ObjCClassDecl.
297dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (ObjCInterfaceDecl *InterD = dyn_cast<ObjCInterfaceDecl>(D)) {
298dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      if (InterD->isForwardDecl())
299dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis        continue;
300dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      // An interface that started as a forward decl may have changed location
301dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      // because its @interface was parsed.
302dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      if (InterD->isInitiallyForwardDecl() &&
303dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis          !SM.isInFileID(SM.getFileLoc(InterD->getLocation()), File))
304dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis        continue;
305dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    }
306dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
307e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (TagDecl *TD = dyn_cast<TagDecl>(D))
308e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      if (!TD->isFreeStanding())
309e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis        continue;
310e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
311dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    RangeComparisonResult CompRes = RangeCompare(SM, D->getSourceRange(),Range);
312dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CompRes == RangeBefore)
313dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      continue;
314dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CompRes == RangeAfter)
315dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
316dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
317dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    assert(CompRes == RangeOverlap);
318dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    VisitedAtLeastOnce = true;
31903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
32003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (isa<ObjCContainerDecl>(D)) {
32103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDI_current = &DIt;
32203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDE_current = DE;
32303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    } else {
32403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDI_current = 0;
32503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    }
32603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
327ba98617b994864b7554ff75445983ad02a962f45Argyrios Kyrtzidis    if (Visit(MakeCXCursor(D, TU, Range), /*CheckedRegionOfInterest=*/true))
328dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
329dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
330dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
331dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (VisitedAtLeastOnce)
332dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
333dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
334dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // No Decls overlapped with the range. Move up the lexical context until there
335dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // is a context that contains the range or we reach the translation unit
336dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // level.
337dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  DeclContext *DC = DIt == Decls.begin() ? (*DIt)->getLexicalDeclContext()
338dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis                                         : (*(DIt-1))->getLexicalDeclContext();
339dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
340dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  while (DC && !DC->isTranslationUnit()) {
341dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Decl *D = cast<Decl>(DC);
342dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    SourceRange CurDeclRange = D->getSourceRange();
343dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CurDeclRange.isInvalid())
344dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
345dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
346dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (RangeCompare(SM, CurDeclRange, Range) == RangeOverlap) {
347ba98617b994864b7554ff75445983ad02a962f45Argyrios Kyrtzidis      Visit(MakeCXCursor(D, TU, Range), /*CheckedRegionOfInterest=*/true);
348dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
349dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    }
350dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
351dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    DC = D->getLexicalDeclContext();
352dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
353dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis}
354dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
3554c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntitiesInRegion() {
356b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!AU->getPreprocessor().getPreprocessingRecord())
357b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis    return false;
358b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis
359788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  PreprocessingRecord &PPRec
360a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    = *AU->getPreprocessor().getPreprocessingRecord();
361f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  SourceManager &SM = AU->getSourceManager();
362788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
36392ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (RegionOfInterest.isValid()) {
364ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange MappedRange = AU->mapRangeToPreamble(RegionOfInterest);
365f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    SourceLocation B = MappedRange.getBegin();
366f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    SourceLocation E = MappedRange.getEnd();
367f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
368f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (AU->isInPreambleFileID(B)) {
369f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      if (SM.isLoadedSourceLocation(E))
370f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis        return visitPreprocessedEntitiesInRange(SourceRange(B, E),
371f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                                 PPRec, *this);
372f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
373f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // Beginning of range lies in the preamble but it also extends beyond
374f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // it into the main file. Split the range into 2 parts, one covering
375f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // the preamble and another covering the main file. This allows subsequent
376f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // calls to visitPreprocessedEntitiesInRange to accept a source range that
377f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // lies in the same FileID, allowing it to skip preprocessed entities that
378f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // do not come from the same FileID.
379f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      bool breaked =
380f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis        visitPreprocessedEntitiesInRange(
381f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                   SourceRange(B, AU->getEndOfPreambleFileID()),
382f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                          PPRec, *this);
383f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      if (breaked) return true;
384f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      return visitPreprocessedEntitiesInRange(
385f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                    SourceRange(AU->getStartOfMainFileID(), E),
386f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                        PPRec, *this);
387f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    }
388f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
389f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    return visitPreprocessedEntitiesInRange(SourceRange(B, E), PPRec, *this);
39092ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  }
39192ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis
392788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  bool OnlyLocalDecls
39332038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor    = !AU->isMainFileAST() && AU->getOnlyLocalDecls();
39432038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor
39592ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (OnlyLocalDecls)
396f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    return visitPreprocessedEntities(PPRec.local_begin(), PPRec.local_end(),
397f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                     PPRec);
3984c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
399f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  return visitPreprocessedEntities(PPRec.begin(), PPRec.end(), PPRec);
4004c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor}
4014c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4024c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregortemplate<typename InputIterator>
4034c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntities(InputIterator First,
404f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              InputIterator Last,
405f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              PreprocessingRecord &PPRec,
406f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              FileID FID) {
4074c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  for (; First != Last; ++First) {
408f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (!FID.isInvalid() && !PPRec.isEntityInFileID(First, FID))
409f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      continue;
410f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
411f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    PreprocessedEntity *PPE = *First;
412f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (MacroExpansion *ME = dyn_cast<MacroExpansion>(PPE)) {
4134c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroExpansionCursor(ME, TU)))
4144c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4154c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4164c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4174c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4184c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
419f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (MacroDefinition *MD = dyn_cast<MacroDefinition>(PPE)) {
4204c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroDefinitionCursor(MD, TU)))
4214c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
42289d9980bbc2e4a4ac86673e6ec16fb9f5babb63bDouglas Gregor
4234c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4244c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4254c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
426f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (InclusionDirective *ID = dyn_cast<InclusionDirective>(PPE)) {
4274c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeInclusionDirectiveCursor(ID, TU)))
4284c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4294c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4304c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
431788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor    }
432788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  }
433788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
4344c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  return false;
435788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor}
436788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
437b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \brief Visit the children of the given cursor.
438a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek///
439b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \returns true if the visitation should be aborted, false if it
440b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// should continue.
441f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekbool CursorVisitor::VisitChildren(CXCursor Cursor) {
442c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (clang_isReference(Cursor.kind) &&
443c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      Cursor.kind != CXCursor_CXXBaseSpecifier) {
444a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    // By definition, references have no children.
445a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return false;
446a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  }
447f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
448f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Set the Parent field to Cursor, then back to its old value once we're
449b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // done.
4500f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  SetParentRAII SetParent(Parent, StmtParent, Cursor);
451f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
452b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isDeclaration(Cursor.kind)) {
453b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Decl *D = getCursorDecl(Cursor);
45406d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (!D)
45506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return false;
45606d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
457539311e0221df256c70c1c3080c8af847cd29dffTed Kremenek    return VisitAttributes(D) || Visit(D);
458b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
459f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
46006d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isStatement(Cursor.kind)) {
46106d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Stmt *S = getCursorStmt(Cursor))
46206d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(S);
46306d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
46406d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
46506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
46606d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
46706d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isExpression(Cursor.kind)) {
46806d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Expr *E = getCursorExpr(Cursor))
46906d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(E);
47006d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
47106d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
47206d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
473f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
474b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isTranslationUnit(Cursor.kind)) {
475a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    CXTranslationUnit tu = getCursorTU(Cursor);
476a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    ASTUnit *CXXUnit = static_cast<ASTUnit*>(tu->TUData);
47704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
47804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    int VisitOrder[2] = { VisitPreprocessorLast, !VisitPreprocessorLast };
47904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    for (unsigned I = 0; I != 2; ++I) {
48004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      if (VisitOrder[I]) {
48104a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        if (!CXXUnit->isMainFileAST() && CXXUnit->getOnlyLocalDecls() &&
48204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor            RegionOfInterest.isInvalid()) {
48304a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          for (ASTUnit::top_level_iterator TL = CXXUnit->top_level_begin(),
48404a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                        TLEnd = CXXUnit->top_level_end();
48504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor               TL != TLEnd; ++TL) {
486aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis            if (Visit(MakeCXCursor(*TL, tu, RegionOfInterest), true))
48704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor              return true;
48804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          }
48904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        } else if (VisitDeclContext(
49004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                CXXUnit->getASTContext().getTranslationUnitDecl()))
4917b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor          return true;
49204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        continue;
4937b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor      }
4943178cb674ac8c3b59e1791e14d38d48619a1b621Bob Wilson
49504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      // Walk the preprocessing record.
4964c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (CXXUnit->getPreprocessor().getPreprocessingRecord())
4974c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        visitPreprocessedEntitiesInRegion();
4980396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    }
49904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
5007b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor    return false;
501b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
502f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
503c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
504c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    if (CXXBaseSpecifier *Base = getCursorCXXBaseSpecifier(Cursor)) {
505c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      if (TypeSourceInfo *BaseTSInfo = Base->getTypeSourceInfo()) {
506c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor        return Visit(BaseTSInfo->getTypeLoc());
507c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      }
508c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    }
509c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  }
510221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
511221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
512221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    IBOutletCollectionAttr *A =
513221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(Cursor));
514221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    if (const ObjCInterfaceType *InterT = A->getInterface()->getAs<ObjCInterfaceType>())
515221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      return Visit(cxcursor::MakeCursorObjCClassRef(InterT->getInterface(),
516221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis                                                    A->getInterfaceLoc(), TU));
517221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  }
518221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
519b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // Nothing to visit at the moment.
520b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
521dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
522dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5231ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenekbool CursorVisitor::VisitBlockDecl(BlockDecl *B) {
52413c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor  if (TypeSourceInfo *TSInfo = B->getSignatureAsWritten())
52513c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor    if (Visit(TSInfo->getTypeLoc()))
52613c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor        return true;
5271ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
528664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  if (Stmt *Body = B->getBody())
529aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Body, StmtParent, TU, RegionOfInterest));
530664cffd330611d78fc0286f539589920a37ca328Ted Kremenek
531664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  return false;
5321ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek}
5331ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
534d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekllvm::Optional<bool> CursorVisitor::shouldVisitCursor(CXCursor Cursor) {
535d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (RegionOfInterest.isValid()) {
5366653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange Range = getFullCursorExtent(Cursor, AU->getSourceManager());
537d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Range.isInvalid())
538d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
5396653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
540d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    switch (CompareRegionOfInterest(Range)) {
541d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeBefore:
542d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes before the region of interest; skip it.
543d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
54423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
545d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeAfter:
546d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes after the region of interest; we're done.
547d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
548d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar
549d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeOverlap:
550d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration overlaps the region of interest; visit it.
551d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      break;
552d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
553d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
554d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return true;
555d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
556f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
557d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekbool CursorVisitor::VisitDeclContext(DeclContext *DC) {
558d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  DeclContext::decl_iterator I = DC->decls_begin(), E = DC->decls_end();
559f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
560d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually remove.  This part of a hack to support proper
561d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // iteration over all Decls contained lexically within an ObjC container.
562d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator*> DI_saved(DI_current, &I);
563d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator> DE_saved(DE_current, E);
564f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
565d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for ( ; I != E; ++I) {
566d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *D = *I;
567d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (D->getLexicalDeclContext() != DC)
568d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
569aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    CXCursor Cursor = MakeCXCursor(D, TU, RegionOfInterest);
570d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
571d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
572d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
573d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
574d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
575d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar    if (Visit(Cursor, true))
576b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
577b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
578b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
579dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
580dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5811ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTranslationUnitDecl(TranslationUnitDecl *D) {
5821ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  llvm_unreachable("Translation units are visited directly by Visit()");
5831ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
5841ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
5851ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
586162e1c1b487352434552147967c3dd296ebee2f7Richard Smithbool CursorVisitor::VisitTypeAliasDecl(TypeAliasDecl *D) {
587162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
588162e1c1b487352434552147967c3dd296ebee2f7Richard Smith    return Visit(TSInfo->getTypeLoc());
589162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
590162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return false;
591162e1c1b487352434552147967c3dd296ebee2f7Richard Smith}
592162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
5931ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTypedefDecl(TypedefDecl *D) {
5941ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
5951ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(TSInfo->getTypeLoc());
596f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
5971ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
5981ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
5991ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6001ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTagDecl(TagDecl *D) {
6011ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitDeclContext(D);
6021ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6031ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6040ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregorbool CursorVisitor::VisitClassTemplateSpecializationDecl(
6050ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor                                          ClassTemplateSpecializationDecl *D) {
6060ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  bool ShouldVisitBody = false;
6070ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  switch (D->getSpecializationKind()) {
6080ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_Undeclared:
6090ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ImplicitInstantiation:
6100ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    // Nothing to visit
6110ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return false;
6120ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6130ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDeclaration:
6140ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDefinition:
6150ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6160ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6170ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitSpecialization:
6180ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    ShouldVisitBody = true;
6190ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6200ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6210ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6220ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  // Visit the template arguments used in the specialization.
6230ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (TypeSourceInfo *SpecType = D->getTypeAsWritten()) {
6240ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    TypeLoc TL = SpecType->getTypeLoc();
6250ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    if (TemplateSpecializationTypeLoc *TSTLoc
6260ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          = dyn_cast<TemplateSpecializationTypeLoc>(&TL)) {
6270ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor      for (unsigned I = 0, N = TSTLoc->getNumArgs(); I != N; ++I)
6280ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor        if (VisitTemplateArgumentLoc(TSTLoc->getArgLoc(I)))
6290ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          return true;
6300ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    }
6310ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6320ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6330ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (ShouldVisitBody && VisitCXXRecordDecl(D))
6340ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return true;
6350ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6360ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  return false;
6370ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor}
6380ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
63974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregorbool CursorVisitor::VisitClassTemplatePartialSpecializationDecl(
64074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor                                   ClassTemplatePartialSpecializationDecl *D) {
64174dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
64274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // before visiting these template parameters.
64374dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
64474dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return true;
64574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
64674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // Visit the partial specialization arguments.
64774dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  const TemplateArgumentLoc *TemplateArgs = D->getTemplateArgsAsWritten();
64874dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  for (unsigned I = 0, N = D->getNumTemplateArgsAsWritten(); I != N; ++I)
64974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    if (VisitTemplateArgumentLoc(TemplateArgs[I]))
65074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor      return true;
65174dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
65274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  return VisitCXXRecordDecl(D);
65374dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor}
65474dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
655fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmDecl(TemplateTypeParmDecl *D) {
65684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  // Visit the default argument.
65784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
65884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (TypeSourceInfo *DefArg = D->getDefaultArgumentInfo())
65984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      if (Visit(DefArg->getTypeLoc()))
66084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor        return true;
66184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
662fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
663fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
664fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
6651ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitEnumConstantDecl(EnumConstantDecl *D) {
6661ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInitExpr())
667aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Init, StmtParent, TU, RegionOfInterest));
6681ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
6691ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6701ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6717d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitDeclaratorDecl(DeclaratorDecl *DD) {
6727d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  if (TypeSourceInfo *TSInfo = DD->getTypeSourceInfo())
6737d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
6747d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      return true;
6757d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
676c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
677c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = DD->getQualifierLoc())
678c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
679c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
680c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
6817d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  return false;
6827d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
6837d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
684a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor/// \brief Compare two base or member initializers based on their source order.
685cbb67480094b3bcb5b715acd827cbad55e2a204cSean Huntstatic int CompareCXXCtorInitializers(const void* Xp, const void *Yp) {
686cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *X
687cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Xp);
688cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *Y
689cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Yp);
690a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
691a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  if ((*X)->getSourceOrder() < (*Y)->getSourceOrder())
692a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return -1;
693a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else if ((*X)->getSourceOrder() > (*Y)->getSourceOrder())
694a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 1;
695a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else
696a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 0;
697a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor}
698a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
699b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitFunctionDecl(FunctionDecl *ND) {
70001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (TypeSourceInfo *TSInfo = ND->getTypeSourceInfo()) {
70101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function declaration's syntactic components in the order
70201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // written. This requires a bit of work.
703723df245307a530da5433dfb43accf187dc3e243Abramo Bagnara    TypeLoc TL = TSInfo->getTypeLoc().IgnoreParens();
70401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    FunctionTypeLoc *FTL = dyn_cast<FunctionTypeLoc>(&TL);
70501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
70601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // If we have a function declared directly (without the use of a typedef),
70701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // visit just the return type. Otherwise, just visit the function's type
70801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // now.
70901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if ((FTL && !isa<CXXConversionDecl>(ND) && Visit(FTL->getResultLoc())) ||
71001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor        (!FTL && Visit(TL)))
71101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
71201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
713c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // Visit the nested-name-specifier, if present.
714c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (NestedNameSpecifierLoc QualifierLoc = ND->getQualifierLoc())
715c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      if (VisitNestedNameSpecifierLoc(QualifierLoc))
716c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor        return true;
71701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
71801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the declaration name.
71901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (VisitDeclarationNameInfo(ND->getNameInfo()))
72001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
72101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Visit explicitly-specified template arguments!
72301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function parameters, if we have a function type.
72501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (FTL && VisitFunctionTypeLoc(*FTL, true))
72601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
72701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Attributes?
72901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
73001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73110620eb5164e31208fcbf0437cd79ae535ed0559Sean Hunt  if (ND->doesThisDeclarationHaveABody() && !ND->isLateTemplateParsed()) {
732a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(ND)) {
733a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Find the initializers that were written in the source.
7345f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner      SmallVector<CXXCtorInitializer *, 4> WrittenInits;
735a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (CXXConstructorDecl::init_iterator I = Constructor->init_begin(),
736a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                          IEnd = Constructor->init_end();
737a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor           I != IEnd; ++I) {
738a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (!(*I)->isWritten())
739a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor          continue;
740a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
741a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        WrittenInits.push_back(*I);
742a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
743a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
744a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Sort the initializers in source order
745a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      llvm::array_pod_sort(WrittenInits.begin(), WrittenInits.end(),
746cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt                           &CompareCXXCtorInitializers);
747a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
748a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Visit the initializers in source order
749a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (unsigned I = 0, N = WrittenInits.size(); I != N; ++I) {
750cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt        CXXCtorInitializer *Init = WrittenInits[I];
75100eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet        if (Init->isAnyMemberInitializer()) {
75200eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet          if (Visit(MakeCursorMemberRef(Init->getAnyMember(),
753a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                        Init->getMemberLocation(), TU)))
754a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
75576852c218a207ef43583515cb835b6e855353a0fDouglas Gregor        } else if (TypeSourceInfo *TInfo = Init->getTypeSourceInfo()) {
75676852c218a207ef43583515cb835b6e855353a0fDouglas Gregor          if (Visit(TInfo->getTypeLoc()))
757a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
758a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        }
759a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
760a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        // Visit the initializer value.
761a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (Expr *Initializer = Init->getInit())
762aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis          if (Visit(MakeCXCursor(Initializer, ND, TU, RegionOfInterest)))
763a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
764a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
765a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
766a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
767aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(ND->getBody(), StmtParent, TU, RegionOfInterest)))
768a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return true;
769a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  }
770f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
771b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
772b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
773dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
7741ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitFieldDecl(FieldDecl *D) {
7751ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7761ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
777f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7781ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *BitWidth = D->getBitWidth())
779aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(BitWidth, StmtParent, TU, RegionOfInterest));
780f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7811ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7821ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7831ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
7841ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitVarDecl(VarDecl *D) {
7851ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7861ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
787f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7881ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInit())
789aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Init, StmtParent, TU, RegionOfInterest));
790f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7911ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7921ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7931ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
79484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitNonTypeTemplateParmDecl(NonTypeTemplateParmDecl *D) {
79584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitDeclaratorDecl(D))
79684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
79784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
79884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
79984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (Expr *DefArg = D->getDefaultArgument())
800aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(DefArg, StmtParent, TU, RegionOfInterest));
80184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
80284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
80384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
80484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
805fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitFunctionTemplateDecl(FunctionTemplateDecl *D) {
806fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the FunctionDecl
807fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // before visiting these template parameters.
808fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
809fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return true;
810fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
811fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return VisitFunctionDecl(D->getTemplatedDecl());
812fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
813fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
81439d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregorbool CursorVisitor::VisitClassTemplateDecl(ClassTemplateDecl *D) {
81539d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
81639d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // before visiting these template parameters.
81739d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
81839d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return true;
81939d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82039d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  return VisitCXXRecordDecl(D->getTemplatedDecl());
82139d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor}
82239d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitTemplateTemplateParmDecl(TemplateTemplateParmDecl *D) {
82484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
82584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
82684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
82784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited() &&
82884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      VisitTemplateArgumentLoc(D->getDefaultArgument()))
82984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
83084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
83184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
83284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
83384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
8341ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCMethodDecl(ObjCMethodDecl *ND) {
8354bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  if (TypeSourceInfo *TSInfo = ND->getResultTypeSourceInfo())
8364bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
8374bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor      return true;
8384bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor
839f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  for (ObjCMethodDecl::param_iterator P = ND->param_begin(),
8401ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       PEnd = ND->param_end();
8411ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       P != PEnd; ++P) {
842aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(*P, TU, RegionOfInterest)))
8431ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
8441ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  }
845f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8461ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (ND->isThisDeclarationADefinition() &&
847aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Visit(MakeCXCursor(ND->getBody(), StmtParent, TU, RegionOfInterest)))
8481ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
849f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8501ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
8511ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
8521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
85303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidistemplate <typename DeclIt>
85403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidisstatic void addRangedDeclsInContainer(DeclIt *DI_current, DeclIt DE_current,
85503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                      SourceManager &SM, SourceLocation EndLoc,
85603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                      SmallVectorImpl<Decl *> &Decls) {
85703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  DeclIt next = *DI_current;
85803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  while (++next != DE_current) {
85903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    Decl *D_next = *next;
86003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (!D_next)
86103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      break;
86203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    SourceLocation L = D_next->getLocStart();
86303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (!L.isValid())
86403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      break;
86503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (SM.isBeforeInTranslationUnit(L, EndLoc)) {
86603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      *DI_current = next;
86703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      Decls.push_back(D_next);
86803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      continue;
86903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    }
87003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    break;
87103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  }
87203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis}
87303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
874d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremeneknamespace {
875d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  struct ContainerDeclsSort {
876d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    SourceManager &SM;
877d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    ContainerDeclsSort(SourceManager &sm) : SM(sm) {}
878d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    bool operator()(Decl *A, Decl *B) {
879d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_A = A->getLocStart();
880d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_B = B->getLocStart();
881d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      assert(L_A.isValid() && L_B.isValid());
882d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return SM.isBeforeInTranslationUnit(L_A, L_B);
883d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
884d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  };
885d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
886d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
887a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregorbool CursorVisitor::VisitObjCContainerDecl(ObjCContainerDecl *D) {
888d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually convert back to just 'VisitDeclContext()'.  Essentially
889d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // an @implementation can lexically contain Decls that are not properly
890d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // nested in the AST.  When we identify such cases, we need to retrofit
891d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // this nesting here.
89203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  if (!DI_current && !FileDI_current)
893d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
894d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
895d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Scan the Decls that immediately come after the container
896d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // in the current DeclContext.  If any fall within the
897d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // container's lexical region, stash them into a vector
898d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // for later processing.
8995f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<Decl *, 24> DeclsInContainer;
900d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SourceLocation EndLoc = D->getSourceRange().getEnd();
901a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  SourceManager &SM = AU->getSourceManager();
902d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (EndLoc.isValid()) {
90303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (DI_current) {
90403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      addRangedDeclsInContainer(DI_current, DE_current, SM, EndLoc,
90503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                DeclsInContainer);
90603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    } else {
90703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      addRangedDeclsInContainer(FileDI_current, FileDE_current, SM, EndLoc,
90803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                DeclsInContainer);
909d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
910d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
911d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
912d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // The common case.
913d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (DeclsInContainer.empty())
914d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
915d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
916d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Get all the Decls in the DeclContext, and sort them with the
917d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // additional ones we've collected.  Then visit them.
918d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for (DeclContext::decl_iterator I = D->decls_begin(), E = D->decls_end();
919d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek       I!=E; ++I) {
920d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *subDecl = *I;
9210582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek    if (!subDecl || subDecl->getLexicalDeclContext() != D ||
9220582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek        subDecl->getLocStart().isInvalid())
923d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
924d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    DeclsInContainer.push_back(subDecl);
925d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
926d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
927d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now sort the Decls so that they appear in lexical order.
928d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  std::sort(DeclsInContainer.begin(), DeclsInContainer.end(),
929d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek            ContainerDeclsSort(SM));
930d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
931d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now visit the decls.
9325f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  for (SmallVectorImpl<Decl*>::iterator I = DeclsInContainer.begin(),
933d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek         E = DeclsInContainer.end(); I != E; ++I) {
934aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    CXCursor Cursor = MakeCXCursor(*I, TU, RegionOfInterest);
935d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
936d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
937d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
938d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
939d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
940d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Visit(Cursor, true))
941d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return true;
942d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
943d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return false;
944a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor}
945a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor
946b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCCategoryDecl(ObjCCategoryDecl *ND) {
947b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (Visit(MakeCursorObjCClassRef(ND->getClassInterface(), ND->getLocation(),
948b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                   TU)))
949b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
950f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
95178db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCCategoryDecl::protocol_loc_iterator PL = ND->protocol_loc_begin();
95278db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCCategoryDecl::protocol_iterator I = ND->protocol_begin(),
95378db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = ND->protocol_end(); I != E; ++I, ++PL)
954b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
955b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
956f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
957a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(ND);
958dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
959dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
9601ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCProtocolDecl(ObjCProtocolDecl *PID) {
9611ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  ObjCProtocolDecl::protocol_loc_iterator PL = PID->protocol_loc_begin();
9621ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  for (ObjCProtocolDecl::protocol_iterator I = PID->protocol_begin(),
9631ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       E = PID->protocol_end(); I != E; ++I, ++PL)
9641ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
9651ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
966f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
9671ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(PID);
9681ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
9691ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
97023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenekbool CursorVisitor::VisitObjCPropertyDecl(ObjCPropertyDecl *PD) {
97183cb94269015bf2770ade71e616c5322ea7e76e1Douglas Gregor  if (PD->getTypeSourceInfo() && Visit(PD->getTypeSourceInfo()->getTypeLoc()))
972fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall    return true;
973fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall
97423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // FIXME: This implements a workaround with @property declarations also being
97523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // installed in the DeclContext for the @interface.  Eventually this code
97623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // should be removed.
97723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCCategoryDecl *CDecl = dyn_cast<ObjCCategoryDecl>(PD->getDeclContext());
97823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!CDecl || !CDecl->IsClassExtension())
97923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
98023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
98123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCInterfaceDecl *ID = CDecl->getClassInterface();
98223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!ID)
98323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
98423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
98523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  IdentifierInfo *PropertyId = PD->getIdentifier();
98623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCPropertyDecl *prevDecl =
98723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    ObjCPropertyDecl::findPropertyDecl(cast<DeclContext>(ID), PropertyId);
98823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
98923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!prevDecl)
99023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
99123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // Visit synthesized methods since they will be skipped when visiting
99323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // the @interface.
99423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getGetterMethodDecl())
995a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
996aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(MD, TU, RegionOfInterest)))
99723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
99823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getSetterMethodDecl())
1000a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
1001aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(MD, TU, RegionOfInterest)))
100223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
100323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
100423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  return false;
100523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek}
100623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
1007b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
1008dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek  // Issue callbacks for super class.
1009b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (D->getSuperClass() &&
1010b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1011f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
1012b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                        TU)))
1013b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
1014f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
101578db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCInterfaceDecl::protocol_loc_iterator PL = D->protocol_loc_begin();
101678db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCInterfaceDecl::protocol_iterator I = D->protocol_begin(),
101778db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = D->protocol_end(); I != E; ++I, ++PL)
1018b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
1019b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
1020f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
1021a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(D);
1022dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1023dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10241ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplDecl(ObjCImplDecl *D) {
10251ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(D);
10261ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10271ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10281ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) {
1029ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  // 'ID' could be null when dealing with invalid code.
1030ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  if (ObjCInterfaceDecl *ID = D->getClassInterface())
1031ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek    if (Visit(MakeCursorObjCClassRef(ID, D->getLocation(), TU)))
1032ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek      return true;
1033f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10341ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
10351ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10361ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10371ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplementationDecl(ObjCImplementationDecl *D) {
10381ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#if 0
10391ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // Issue callbacks for super class.
10401ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // FIXME: No source location information!
10411ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (D->getSuperClass() &&
10421ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1043f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
10441ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor                                        TU)))
1045a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return true;
10461ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#endif
1047f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10481ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
1049dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1050dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10511ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D) {
10521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  ObjCForwardProtocolDecl::protocol_loc_iterator PL = D->protocol_loc_begin();
10531ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  for (ObjCForwardProtocolDecl::protocol_iterator I = D->protocol_begin(),
10541ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor                                                  E = D->protocol_end();
10551ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       I != E; ++I, ++PL)
1056b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
1057b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
1058f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
1059f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return false;
1060dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1061dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10621ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCClassDecl(ObjCClassDecl *D) {
106395ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian  if (Visit(MakeCursorObjCClassRef(D->getForwardInterfaceDecl(),
106495ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian                                   D->getForwardDecl()->getLocation(), TU)))
10651ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
10661ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
1067dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
10685e4bc590b0ea010e38372d0b4a0aab578a746fe6Benjamin Kramer
1069a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregorbool CursorVisitor::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *PD) {
1070a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  if (ObjCIvarDecl *Ivar = PD->getPropertyIvarDecl())
1071a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor    return Visit(MakeCursorMemberRef(Ivar, PD->getPropertyIvarDeclLoc(), TU));
1072a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
1073a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  return false;
1074a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor}
1075a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
10768f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenekbool CursorVisitor::VisitNamespaceDecl(NamespaceDecl *D) {
10778f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  return VisitDeclContext(D);
10788f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek}
10798f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek
10806931900f43cea558c6974075256c07728dbfecc6Douglas Gregorbool CursorVisitor::VisitNamespaceAliasDecl(NamespaceAliasDecl *D) {
1081c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
10820cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
10830cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1084c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
10856931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10866931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  return Visit(MakeCursorNamespaceRef(D->getAliasedNamespace(),
10876931900f43cea558c6974075256c07728dbfecc6Douglas Gregor                                      D->getTargetNameLoc(), TU));
10886931900f43cea558c6974075256c07728dbfecc6Douglas Gregor}
10896931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10907e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUsingDecl(UsingDecl *D) {
1091c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1092dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1093dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1094c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1095dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
10967e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
10971f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (Visit(MakeCursorOverloadedDeclRef(D, D->getLocation(), TU)))
10981f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return true;
10991f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
11007e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
11017e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11027e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
11030a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregorbool CursorVisitor::VisitUsingDirectiveDecl(UsingDirectiveDecl *D) {
1104c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1105db9924191092b4d426cc066637d81698211846aaDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1106db9924191092b4d426cc066637d81698211846aaDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1107c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
11080a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11090a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  return Visit(MakeCursorNamespaceRef(D->getNominatedNamespaceAsWritten(),
11100a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor                                      D->getIdentLocation(), TU));
11110a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor}
11120a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11137e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingValueDecl(UnresolvedUsingValueDecl *D) {
1114c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1115dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1116dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1117c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1118dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1119c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11207e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
11217e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11227e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
11237e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypenameDecl(
11247e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor                                               UnresolvedUsingTypenameDecl *D) {
1125c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1126dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1127dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1128c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1129c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11307e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return false;
11317e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11327e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
113301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitDeclarationNameInfo(DeclarationNameInfo Name) {
113401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  switch (Name.getName().getNameKind()) {
113501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::Identifier:
113601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXLiteralOperatorName:
113701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXOperatorName:
113801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXUsingDirective:
113901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
114001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
114101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConstructorName:
114201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXDestructorName:
114301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConversionFunctionName:
114401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (TypeSourceInfo *TSInfo = Name.getNamedTypeInfo())
114501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return Visit(TSInfo->getTypeLoc());
114601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
114701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
114801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCZeroArgSelector:
114901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCOneArgSelector:
115001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCMultiArgSelector:
115101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Per-identifier location info?
115201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
115301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
115401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
115501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  return false;
115601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
115701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1158c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregorbool CursorVisitor::VisitNestedNameSpecifier(NestedNameSpecifier *NNS,
1159c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                             SourceRange Range) {
1160c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // FIXME: This whole routine is a hack to work around the lack of proper
1161c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // source information in nested-name-specifiers (PR5791). Since we do have
1162c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // a beginning source location, we can visit the first component of the
1163c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // nested-name-specifier, if it's a single-token component.
1164c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  if (!NNS)
1165c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return false;
1166c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1167c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Get the first component in the nested-name-specifier.
1168c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  while (NestedNameSpecifier *Prefix = NNS->getPrefix())
1169c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    NNS = Prefix;
1170c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1171c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  switch (NNS->getKind()) {
1172c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Namespace:
1173c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(), Range.getBegin(),
1174c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                        TU));
1175c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
117614aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor  case NestedNameSpecifier::NamespaceAlias:
117714aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
117814aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor                                        Range.getBegin(), TU));
117914aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor
1180c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpec: {
1181c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // If the type has a form where we know that the beginning of the source
1182c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // range matches up with a reference cursor. Visit the appropriate reference
1183c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // cursor.
1184f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall    const Type *T = NNS->getAsType();
1185c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TypedefType *Typedef = dyn_cast<TypedefType>(T))
1186c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Typedef->getDecl(), Range.getBegin(), TU));
1187c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TagType *Tag = dyn_cast<TagType>(T))
1188c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Tag->getDecl(), Range.getBegin(), TU));
1189c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TemplateSpecializationType *TST
1190c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                      = dyn_cast<TemplateSpecializationType>(T))
1191c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return VisitTemplateName(TST->getTemplateName(), Range.getBegin());
1192c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1193c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1194c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1195c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpecWithTemplate:
1196c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Global:
1197c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Identifier:
1198c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1199c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1200c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1201c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  return false;
1202c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor}
1203c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1204dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregorbool
1205dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas GregorCursorVisitor::VisitNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier) {
12065f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<NestedNameSpecifierLoc, 4> Qualifiers;
1207dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  for (; Qualifier; Qualifier = Qualifier.getPrefix())
1208dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    Qualifiers.push_back(Qualifier);
1209dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1210dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  while (!Qualifiers.empty()) {
1211dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifierLoc Q = Qualifiers.pop_back_val();
1212dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifier *NNS = Q.getNestedNameSpecifier();
1213dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    switch (NNS->getKind()) {
1214dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Namespace:
1215dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(),
1216c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1217dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1218dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1219dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1220dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1221dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1222dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::NamespaceAlias:
1223dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
1224c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1225dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1226dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1227dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1228dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1229dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1230dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpec:
1231dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpecWithTemplate:
1232dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(Q.getTypeLoc()))
1233dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1234dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1235dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1236dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1237dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Global:
1238dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Identifier:
1239dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1240dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    }
1241dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1242dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1243dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  return false;
1244dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor}
1245dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1246fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateParameters(
1247fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          const TemplateParameterList *Params) {
1248fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (!Params)
1249fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1250fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1251fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (TemplateParameterList::const_iterator P = Params->begin(),
1252fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          PEnd = Params->end();
1253fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor       P != PEnd; ++P) {
1254aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(*P, TU, RegionOfInterest)))
1255fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1256fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
1257fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1258fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1259fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1260fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
12610b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregorbool CursorVisitor::VisitTemplateName(TemplateName Name, SourceLocation Loc) {
12620b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  switch (Name.getKind()) {
12630b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::Template:
12640b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(Name.getAsTemplateDecl(), Loc, TU));
12650b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12660b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::OverloadedTemplate:
12671f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // Visit the overloaded template set.
12681f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (Visit(MakeCursorOverloadedDeclRef(Name, Loc, TU)))
12691f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return true;
12701f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
12710b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12720b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12730b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::DependentTemplate:
12740b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12750b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12760b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12770b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::QualifiedTemplate:
12780b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12790b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(
12800b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                  Name.getAsQualifiedTemplateName()->getDecl(),
12810b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                       Loc, TU));
1282146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall
1283146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall  case TemplateName::SubstTemplateTemplateParm:
1284146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall    return Visit(MakeCursorTemplateRef(
1285146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                         Name.getAsSubstTemplateTemplateParm()->getParameter(),
1286146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                                       Loc, TU));
12871aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor
12881aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor  case TemplateName::SubstTemplateTemplateParmPack:
12891aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor    return Visit(MakeCursorTemplateRef(
12901aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                  Name.getAsSubstTemplateTemplateParmPack()->getParameterPack(),
12911aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                                       Loc, TU));
12920b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  }
12930b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12940b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  return false;
12950b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor}
12960b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
1297fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateArgumentLoc(const TemplateArgumentLoc &TAL) {
1298fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  switch (TAL.getArgument().getKind()) {
1299fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Null:
1300fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Integral:
1301fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Pack:
1302fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
130387dd697dcc8ecb64df73ae64d61b8c80ff0c157cDouglas Gregor
1304fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Type:
1305fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (TypeSourceInfo *TSInfo = TAL.getTypeSourceInfo())
1306fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return Visit(TSInfo->getTypeLoc());
1307fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1308fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1309fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Declaration:
1310fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceDeclExpression())
1311aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(E, StmtParent, TU, RegionOfInterest));
1312fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1313fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1314fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Expression:
1315fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceExpression())
1316aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(E, StmtParent, TU, RegionOfInterest));
1317fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1318fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1319fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Template:
1320a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor  case TemplateArgument::TemplateExpansion:
1321b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor    if (VisitNestedNameSpecifierLoc(TAL.getTemplateQualifierLoc()))
1322b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor      return true;
1323b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor
1324a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor    return VisitTemplateName(TAL.getArgument().getAsTemplateOrTemplatePattern(),
13250b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                             TAL.getTemplateNameLoc());
1326fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
1327fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1328fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1329fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1330fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1331a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenekbool CursorVisitor::VisitLinkageSpecDecl(LinkageSpecDecl *D) {
1332a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  return VisitDeclContext(D);
1333a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek}
1334a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek
133501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitQualifiedTypeLoc(QualifiedTypeLoc TL) {
133601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  return Visit(TL.getUnqualifiedLoc());
133701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
133801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1339f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBuiltinTypeLoc(BuiltinTypeLoc TL) {
1340a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTContext &Context = AU->getASTContext();
1341f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1342f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // Some builtin types (such as Objective-C's "id", "sel", and
1343f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // "Class") have associated declarations. Create cursors for those.
1344f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  QualType VisitType;
1345e0a22d06888c13989b3f72db319f1d498bf69153John McCall  switch (TL.getTypePtr()->getKind()) {
13462dde35bc626153492f5f58202506c88a27fbff5bJohn McCall
13476b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Void:
1348f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::NullPtr:
13496b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Dependent:
13502dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define BUILTIN_TYPE(Id, SingletonId)
13512dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define SIGNED_TYPE(Id, SingletonId) case BuiltinType::Id:
13522dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define UNSIGNED_TYPE(Id, SingletonId) case BuiltinType::Id:
13532dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define FLOATING_TYPE(Id, SingletonId) case BuiltinType::Id:
13542dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define PLACEHOLDER_TYPE(Id, SingletonId) case BuiltinType::Id:
13552dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#include "clang/AST/BuiltinTypes.def"
1356f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13576b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1358f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCId:
1359f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCIdType();
1360f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13616b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
13626b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::ObjCClass:
13636b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    VisitType = Context.getObjCClassType();
13646b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    break;
13656b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1366f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCSel:
1367f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCSelType();
1368f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
1369f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1370f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1371f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (!VisitType.isNull()) {
1372f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (const TypedefType *Typedef = VisitType->getAs<TypedefType>())
1373f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek      return Visit(MakeCursorTypeRef(Typedef->getDecl(), TL.getBuiltinLoc(),
1374f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                     TU));
1375f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1376f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1377f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1378f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1379f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
13807d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitTypedefTypeLoc(TypedefTypeLoc TL) {
1381162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return Visit(MakeCursorTypeRef(TL.getTypedefNameDecl(), TL.getNameLoc(), TU));
13827d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
13837d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
1384f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypeLoc(UnresolvedUsingTypeLoc TL) {
1385f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1386f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1387f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1388f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitTagTypeLoc(TagTypeLoc TL) {
13896f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis  if (TL.isDefinition())
1390aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(TL.getDecl(), TU, RegionOfInterest));
13916f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis
1392f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1393f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1394f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1395fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc TL) {
1396960d13dde337a59dacc9dc3936c26d4aa8478986Chandler Carruth  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1397fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1398fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1399f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc TL) {
1400f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(MakeCursorObjCClassRef(TL.getIFaceDecl(), TL.getNameLoc(), TU)))
1401f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1402f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1403c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return false;
1404c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall}
1405c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1406c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCallbool CursorVisitor::VisitObjCObjectTypeLoc(ObjCObjectTypeLoc TL) {
1407c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  if (TL.hasBaseTypeAsWritten() && Visit(TL.getBaseLoc()))
1408c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall    return true;
1409c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1410f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumProtocols(); I != N; ++I) {
1411f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (Visit(MakeCursorObjCProtocolRef(TL.getProtocol(I), TL.getProtocolLoc(I),
1412f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                        TU)))
1413f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor      return true;
1414f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1415f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1416f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1417f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1418f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1419f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCObjectPointerTypeLoc(ObjCObjectPointerTypeLoc TL) {
1420c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return Visit(TL.getPointeeLoc());
1421f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1422f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1423075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnarabool CursorVisitor::VisitParenTypeLoc(ParenTypeLoc TL) {
1424075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara  return Visit(TL.getInnerLoc());
1425075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara}
1426075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara
1427f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitPointerTypeLoc(PointerTypeLoc TL) {
1428f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1429f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1430f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1431f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBlockPointerTypeLoc(BlockPointerTypeLoc TL) {
1432f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1433f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1434f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1435f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitMemberPointerTypeLoc(MemberPointerTypeLoc TL) {
1436f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1437f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1438f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1439f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitLValueReferenceTypeLoc(LValueReferenceTypeLoc TL) {
1440f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1441f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1442f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1443f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitRValueReferenceTypeLoc(RValueReferenceTypeLoc TL) {
1444f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1445f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1446f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
14473422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidisbool CursorVisitor::VisitAttributedTypeLoc(AttributedTypeLoc TL) {
14483422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis  return Visit(TL.getModifiedLoc());
14493422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis}
14503422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis
145101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitFunctionTypeLoc(FunctionTypeLoc TL,
145201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor                                         bool SkipResultType) {
145301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (!SkipResultType && Visit(TL.getResultLoc()))
1454f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1455f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1456f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
14575dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek    if (Decl *D = TL.getArg(I))
1458aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(D, TU, RegionOfInterest)))
14595dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek        return true;
1460f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1461f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1462f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1463f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1464f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitArrayTypeLoc(ArrayTypeLoc TL) {
1465f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(TL.getElementLoc()))
1466f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1467f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1468f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Expr *Size = TL.getSizeExpr())
1469aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Size, StmtParent, TU, RegionOfInterest));
1470f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1471f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1472f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1473f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1474fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateSpecializationTypeLoc(
1475fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                             TemplateSpecializationTypeLoc TL) {
14760b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  // Visit the template name.
14770b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  if (VisitTemplateName(TL.getTypePtr()->getTemplateName(),
14780b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                        TL.getTemplateNameLoc()))
14790b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return true;
1480fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1481fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // Visit the template arguments.
1482fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
1483fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
1484fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1485fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1486fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1487fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1488fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
14892332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfExprTypeLoc(TypeOfExprTypeLoc TL) {
14902332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return Visit(MakeCXCursor(TL.getUnderlyingExpr(), StmtParent, TU));
14912332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
14922332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
14932332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfTypeLoc(TypeOfTypeLoc TL) {
14942332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
1495ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt    return Visit(TSInfo->getTypeLoc());
1496ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1497ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  return false;
1498ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt}
1499ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1500ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Huntbool CursorVisitor::VisitUnaryTransformTypeLoc(UnaryTransformTypeLoc TL) {
1501ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
15022332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor    return Visit(TSInfo->getTypeLoc());
15032332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15042332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return false;
15052332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
15062332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15072494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregorbool CursorVisitor::VisitDependentNameTypeLoc(DependentNameTypeLoc TL) {
15082494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15092494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    return true;
15102494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
15112494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  return false;
15122494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor}
15132494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
151494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregorbool CursorVisitor::VisitDependentTemplateSpecializationTypeLoc(
151594fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor                                    DependentTemplateSpecializationTypeLoc TL) {
151694fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the nested-name-specifier, if there is one.
151794fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  if (TL.getQualifierLoc() &&
151894fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
151994fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    return true;
152094fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
152194fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the template arguments.
152294fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
152394fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
152494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      return true;
152594fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
152694fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  return false;
152794fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor}
152894fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
15299e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregorbool CursorVisitor::VisitElaboratedTypeLoc(ElaboratedTypeLoc TL) {
15309e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15319e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor    return true;
15329e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15339e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  return Visit(TL.getNamedTypeLoc());
15349e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor}
15359e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15367536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregorbool CursorVisitor::VisitPackExpansionTypeLoc(PackExpansionTypeLoc TL) {
15377536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor  return Visit(TL.getPatternLoc());
15387536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor}
15397536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor
1540427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitDecltypeTypeLoc(DecltypeTypeLoc TL) {
1541427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  if (Expr *E = TL.getUnderlyingExpr())
1542427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis    return Visit(MakeCXCursor(E, StmtParent, TU));
1543427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1544427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return false;
1545427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1546427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1547427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
1548427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1549427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1550427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1551b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedmanbool CursorVisitor::VisitAtomicTypeLoc(AtomicTypeLoc TL) {
1552b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman  return Visit(TL.getValueLoc());
1553b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman}
1554b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman
1555427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#define DEFAULT_TYPELOC_IMPL(CLASS, PARENT) \
1556427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::Visit##CLASS##TypeLoc(CLASS##TypeLoc TL) { \
1557427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit##PARENT##Loc(TL); \
1558427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1559427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1560427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Complex, Type)
1561427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ConstantArray, ArrayType)
1562427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(IncompleteArray, ArrayType)
1563427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(VariableArray, ArrayType)
1564427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedArray, ArrayType)
1565427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedExtVector, Type)
1566427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Vector, Type)
1567427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ExtVector, VectorType)
1568427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionProto, FunctionType)
1569427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionNoProto, FunctionType)
1570427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Record, TagType)
1571427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Enum, TagType)
1572427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParm, Type)
1573427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParmPack, Type)
1574427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Auto, Type)
1575427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
15763064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenekbool CursorVisitor::VisitCXXRecordDecl(CXXRecordDecl *D) {
1577c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
1578c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1579c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1580c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
1581c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
15825e1cdac63c3d9c9b32fa41fa0b2d242a58a20d49John McCall  if (D->isCompleteDefinition()) {
15833064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    for (CXXRecordDecl::base_class_iterator I = D->bases_begin(),
15843064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek         E = D->bases_end(); I != E; ++I) {
15853064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      if (Visit(cxcursor::MakeCursorCXXBaseSpecifier(I, TU)))
15863064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        return true;
15873064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
15883064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  }
15893064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
15903064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  return VisitTagDecl(D);
15913064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek}
15923064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
159309dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenekbool CursorVisitor::VisitAttributes(Decl *D) {
1594cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt  for (AttrVec::const_iterator i = D->attr_begin(), e = D->attr_end();
1595cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt       i != e; ++i)
1596cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt    if (Visit(MakeCXCursor(*i, D, TU)))
159709dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek        return true;
159809dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
159909dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek  return false;
160009dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek}
160109dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
1602c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1603c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Data-recursive visitor methods.
1604c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1605c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
160628a719433411ef782b582946823bc648ddcc4533Ted Kremeneknamespace {
1607035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#define DEF_JOB(NAME, DATA, KIND)\
1608035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass NAME : public VisitorJob {\
1609035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:\
1610035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  NAME(DATA *d, CXCursor parent) : VisitorJob(parent, VisitorJob::KIND, d) {} \
1611035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) { return VJ->getKind() == KIND; }\
1612f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DATA *get() const { return static_cast<DATA*>(data[0]); }\
1613035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1614035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1615035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(StmtVisit, Stmt, StmtVisitKind)
1616035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(MemberExprParts, MemberExpr, MemberExprPartsKind)
1617e4979ccb5960608edce73f3b274eb7c2de15dac5Ted KremenekDEF_JOB(DeclRefExprParts, DeclRefExpr, DeclRefExprPartsKind)
1618035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(OverloadExprParts, OverloadExpr, OverloadExprPartsKind)
1619b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios KyrtzidisDEF_JOB(ExplicitTemplateArgsVisit, ASTTemplateArgumentListInfo,
162060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        ExplicitTemplateArgsVisitKind)
162194d96291cd041adc5731a2294828a9c20e450b74Douglas GregorDEF_JOB(SizeOfPackExprParts, SizeOfPackExpr, SizeOfPackExprPartsKind)
1622035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#undef DEF_JOB
1623035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1624035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass DeclVisit : public VisitorJob {
1625035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1626035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  DeclVisit(Decl *d, CXCursor parent, bool isFirst) :
1627035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::DeclVisitKind,
1628035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               d, isFirst ? (void*) 1 : (void*) 0) {}
1629035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
163082f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    return VJ->getKind() == DeclVisitKind;
1631035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1632f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  Decl *get() const { return static_cast<Decl*>(data[0]); }
1633f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  bool isFirst() const { return data[1] ? true : false; }
1634035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1635035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass TypeLocVisit : public VisitorJob {
1636035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1637035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  TypeLocVisit(TypeLoc tl, CXCursor parent) :
1638035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::TypeLocVisitKind,
1639035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               tl.getType().getAsOpaquePtr(), tl.getOpaqueData()) {}
1640035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1641035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1642035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return VJ->getKind() == TypeLocVisitKind;
1643035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1644035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
164582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek  TypeLoc get() const {
1646f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    QualType T = QualType::getFromOpaquePtr(data[0]);
1647f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return TypeLoc(T, data[1]);
1648035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1649035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1650035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1651ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekclass LabelRefVisit : public VisitorJob {
1652ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekpublic:
1653ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelRefVisit(LabelDecl *LD, SourceLocation labelLoc, CXCursor parent)
1654ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner    : VisitorJob(parent, VisitorJob::LabelRefVisitKind, LD,
1655dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 labelLoc.getPtrEncoding()) {}
1656ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek
1657ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1658ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek    return VJ->getKind() == VisitorJob::LabelRefVisitKind;
1659ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  }
1660ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelDecl *get() const { return static_cast<LabelDecl*>(data[0]); }
1661ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  SourceLocation getLoc() const {
1662dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin    return SourceLocation::getFromPtrEncoding(data[1]); }
1663f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek};
1664f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1665f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorclass NestedNameSpecifierLocVisit : public VisitorJob {
1666f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorpublic:
1667f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLocVisit(NestedNameSpecifierLoc Qualifier, CXCursor parent)
1668f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    : VisitorJob(parent, VisitorJob::NestedNameSpecifierLocVisitKind,
1669f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getNestedNameSpecifier(),
1670f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getOpaqueData()) { }
1671f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1672f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  static bool classof(const VisitorJob *VJ) {
1673f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return VJ->getKind() == VisitorJob::NestedNameSpecifierLocVisitKind;
1674f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1675f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1676f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLoc get() const {
1677f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return NestedNameSpecifierLoc(static_cast<NestedNameSpecifier*>(data[0]),
1678f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                                  data[1]);
1679f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1680f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor};
1681f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1682f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekclass DeclarationNameInfoVisit : public VisitorJob {
1683f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekpublic:
1684f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfoVisit(Stmt *S, CXCursor parent)
1685f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    : VisitorJob(parent, VisitorJob::DeclarationNameInfoVisitKind, S) {}
1686f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1687f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return VJ->getKind() == VisitorJob::DeclarationNameInfoVisitKind;
1688f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1689f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfo get() const {
1690f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    Stmt *S = static_cast<Stmt*>(data[0]);
1691f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    switch (S->getStmtClass()) {
1692f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    default:
1693f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      llvm_unreachable("Unhandled Stmt");
1694ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor    case clang::Stmt::MSDependentExistsStmtClass:
1695ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor      return cast<MSDependentExistsStmt>(S)->getNameInfo();
1696f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::CXXDependentScopeMemberExprClass:
1697f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<CXXDependentScopeMemberExpr>(S)->getMemberNameInfo();
1698f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::DependentScopeDeclRefExprClass:
1699f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<DependentScopeDeclRefExpr>(S)->getNameInfo();
1700f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    }
1701f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1702ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek};
1703cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekclass MemberRefVisit : public VisitorJob {
1704cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekpublic:
1705cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  MemberRefVisit(FieldDecl *D, SourceLocation L, CXCursor parent)
1706cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    : VisitorJob(parent, VisitorJob::MemberRefVisitKind, D,
1707dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 L.getPtrEncoding()) {}
1708cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  static bool classof(const VisitorJob *VJ) {
1709cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return VJ->getKind() == VisitorJob::MemberRefVisitKind;
1710cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1711cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  FieldDecl *get() const {
1712cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return static_cast<FieldDecl*>(data[0]);
1713cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1714cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  SourceLocation getLoc() const {
1715cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return SourceLocation::getFromRawEncoding((unsigned)(uintptr_t) data[1]);
1716cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1717cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek};
171828a719433411ef782b582946823bc648ddcc4533Ted Kremenekclass EnqueueVisitor : public StmtVisitor<EnqueueVisitor, void> {
171928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitorWorkList &WL;
172028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  CXCursor Parent;
172128a719433411ef782b582946823bc648ddcc4533Ted Kremenekpublic:
172228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueVisitor(VisitorWorkList &wl, CXCursor parent)
172328a719433411ef782b582946823bc648ddcc4533Ted Kremenek    : WL(wl), Parent(parent) {}
172428a719433411ef782b582946823bc648ddcc4533Ted Kremenek
1725ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitAddrLabelExpr(AddrLabelExpr *E);
172673d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitBlockExpr(BlockExpr *B);
172728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCompoundLiteralExpr(CompoundLiteralExpr *E);
1728083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  void VisitCompoundStmt(CompoundStmt *S);
172911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXDefaultArgExpr(CXXDefaultArgExpr *E) { /* Do nothing. */ }
1730ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor  void VisitMSDependentExistsStmt(MSDependentExistsStmt *S);
1731f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitCXXDependentScopeMemberExpr(CXXDependentScopeMemberExpr *E);
173211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXNewExpr(CXXNewExpr *E);
17336d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  void VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E);
173428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E);
1735cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E);
173673d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E);
1737b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  void VisitCXXTypeidExpr(CXXTypeidExpr *E);
173855b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  void VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr *E);
17391e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  void VisitCXXUuidofExpr(CXXUuidofExpr *E);
1740e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  void VisitDeclRefExpr(DeclRefExpr *D);
1741035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  void VisitDeclStmt(DeclStmt *S);
1742f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E);
1743cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitDesignatedInitExpr(DesignatedInitExpr *E);
174428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitExplicitCastExpr(ExplicitCastExpr *E);
174528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitForStmt(ForStmt *FS);
1746ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitGotoStmt(GotoStmt *GS);
174728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitIfStmt(IfStmt *If);
174828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitInitListExpr(InitListExpr *IE);
174928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitMemberExpr(MemberExpr *M);
1750cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitOffsetOfExpr(OffsetOfExpr *E);
175173d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitObjCEncodeExpr(ObjCEncodeExpr *E);
175228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitObjCMessageExpr(ObjCMessageExpr *M);
175328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitOverloadExpr(OverloadExpr *E);
1754f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne  void VisitUnaryExprOrTypeTraitExpr(UnaryExprOrTypeTraitExpr *E);
175528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitStmt(Stmt *S);
175628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitSwitchStmt(SwitchStmt *S);
175728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitWhileStmt(WhileStmt *W);
17582939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  void VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E);
17596ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  void VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E);
176021ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  void VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E);
1761552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  void VisitExpressionTraitExpr(ExpressionTraitExpr *E);
176228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U);
17639d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  void VisitVAArgExpr(VAArgExpr *E);
176494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  void VisitSizeOfPackExpr(SizeOfPackExpr *E);
17654b9c2d235fb9449e249d74f48ecfec601650de93John McCall  void VisitPseudoObjectExpr(PseudoObjectExpr *E);
17664b9c2d235fb9449e249d74f48ecfec601650de93John McCall  void VisitOpaqueValueExpr(OpaqueValueExpr *E);
1767ee8aff06f6a96214731de17b2cb6df407c6c1820Douglas Gregor
176828a719433411ef782b582946823bc648ddcc4533Ted Kremenekprivate:
1769f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void AddDeclarationNameInfo(Stmt *S);
1770f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  void AddNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier);
1771b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis  void AddExplicitTemplateArgs(const ASTTemplateArgumentListInfo *A);
1772cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void AddMemberRef(FieldDecl *D, SourceLocation L);
177328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void AddStmt(Stmt *S);
1774035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  void AddDecl(Decl *D, bool isFirst = true);
177528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void AddTypeLoc(TypeSourceInfo *TI);
177628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void EnqueueChildren(Stmt *S);
177728a719433411ef782b582946823bc648ddcc4533Ted Kremenek};
177828a719433411ef782b582946823bc648ddcc4533Ted Kremenek} // end anonyous namespace
177928a719433411ef782b582946823bc648ddcc4533Ted Kremenek
1780f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::AddDeclarationNameInfo(Stmt *S) {
1781f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  // 'S' should always be non-null, since it comes from the
1782f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  // statement we are visiting.
1783f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  WL.push_back(DeclarationNameInfoVisit(S, Parent));
1784f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
1785f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1786f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorvoid
1787f3db29fff6a583ecda823cf909ab7737d8d30129Douglas GregorEnqueueVisitor::AddNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier) {
1788f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  if (Qualifier)
1789f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    WL.push_back(NestedNameSpecifierLocVisit(Qualifier, Parent));
1790f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor}
1791f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
179228a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::AddStmt(Stmt *S) {
179328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (S)
179428a719433411ef782b582946823bc648ddcc4533Ted Kremenek    WL.push_back(StmtVisit(S, Parent));
179528a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1796035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekvoid EnqueueVisitor::AddDecl(Decl *D, bool isFirst) {
179728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (D)
1798035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    WL.push_back(DeclVisit(D, Parent, isFirst));
179928a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
180060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenekvoid EnqueueVisitor::
1801b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis  AddExplicitTemplateArgs(const ASTTemplateArgumentListInfo *A) {
180260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  if (A)
180360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek    WL.push_back(ExplicitTemplateArgsVisit(
1804b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis                        const_cast<ASTTemplateArgumentListInfo*>(A), Parent));
180560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek}
1806cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::AddMemberRef(FieldDecl *D, SourceLocation L) {
1807cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  if (D)
1808cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    WL.push_back(MemberRefVisit(D, L, Parent));
1809cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
181028a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::AddTypeLoc(TypeSourceInfo *TI) {
181128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (TI)
181228a719433411ef782b582946823bc648ddcc4533Ted Kremenek    WL.push_back(TypeLocVisit(TI->getTypeLoc(), Parent));
181328a719433411ef782b582946823bc648ddcc4533Ted Kremenek }
181428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::EnqueueChildren(Stmt *S) {
1815a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  unsigned size = WL.size();
18167502c1d3ce8bb97bcc4f7bebef507040bd93b26fJohn McCall  for (Stmt::child_range Child = S->children(); Child; ++Child) {
181728a719433411ef782b582946823bc648ddcc4533Ted Kremenek    AddStmt(*Child);
1818a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  }
1819a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  if (size == WL.size())
1820a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek    return;
1821a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  // Now reverse the entries we just added.  This will match the DFS
1822a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  // ordering performed by the worklist.
1823a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  VisitorWorkList::iterator I = WL.begin() + size, E = WL.end();
1824a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  std::reverse(I, E);
1825a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek}
1826ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekvoid EnqueueVisitor::VisitAddrLabelExpr(AddrLabelExpr *E) {
1827ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  WL.push_back(LabelRefVisit(E->getLabel(), E->getLabelLoc(), Parent));
1828ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek}
182973d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitBlockExpr(BlockExpr *B) {
183073d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddDecl(B->getBlockDecl());
183173d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
183228a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitCompoundLiteralExpr(CompoundLiteralExpr *E) {
183328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(E);
183428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(E->getTypeSourceInfo());
183528a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1836083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenekvoid EnqueueVisitor::VisitCompoundStmt(CompoundStmt *S) {
1837083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  for (CompoundStmt::reverse_body_iterator I = S->body_rbegin(),
1838083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek        E = S->body_rend(); I != E; ++I) {
1839083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek    AddStmt(*I);
1840083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  }
184111b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek}
1842f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::
1843ba0513de93d2fab6db5ab30b6927209fcc883078Douglas GregorVisitMSDependentExistsStmt(MSDependentExistsStmt *S) {
1844ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor  AddStmt(S->getSubStmt());
1845ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor  AddDeclarationNameInfo(S);
1846ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = S->getQualifierLoc())
1847ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1848ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor}
1849ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor
1850ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregorvoid EnqueueVisitor::
1851f64d80306144f978148ba92f36f7cea7b671dd34Ted KremenekVisitCXXDependentScopeMemberExpr(CXXDependentScopeMemberExpr *E) {
1852f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
1853f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddDeclarationNameInfo(E);
18547c3179cf463c3b3b8c21dbb955f933ba50b74f28Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = E->getQualifierLoc())
18557c3179cf463c3b3b8c21dbb955f933ba50b74f28Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1856f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  if (!E->isImplicitAccess())
1857f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    AddStmt(E->getBase());
1858f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
185911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenekvoid EnqueueVisitor::VisitCXXNewExpr(CXXNewExpr *E) {
186011b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the initializer or constructor arguments.
186111b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  for (unsigned I = E->getNumConstructorArgs(); I > 0; --I)
186211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek    AddStmt(E->getConstructorArg(I-1));
186311b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the array size, if any.
186411b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddStmt(E->getArraySize());
186511b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the allocated type.
186611b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddTypeLoc(E->getAllocatedTypeSourceInfo());
186711b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the placement arguments.
186811b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  for (unsigned I = E->getNumPlacementArgs(); I > 0; --I)
186911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek    AddStmt(E->getPlacementArg(I-1));
187011b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek}
187128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *CE) {
18728b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek  for (unsigned I = CE->getNumArgs(); I > 1 /* Yes, this is 1 */; --I)
18738b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek    AddStmt(CE->getArg(I-1));
187428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getCallee());
187528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getArg(0));
187628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1877cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E) {
1878cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the name of the type being destroyed.
1879cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getDestroyedTypeInfo());
1880cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the scope type that looks disturbingly like the nested-name-specifier
1881cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // but isn't.
1882cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getScopeTypeInfo());
1883cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the nested-name-specifier.
1884f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = E->getQualifierLoc())
1885f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1886cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit base expression.
1887cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getBase());
1888cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
18896d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenekvoid EnqueueVisitor::VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E) {
18906d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
18916d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
189273d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E) {
189373d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  EnqueueChildren(E);
189473d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
189573d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
1896b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenekvoid EnqueueVisitor::VisitCXXTypeidExpr(CXXTypeidExpr *E) {
1897b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  EnqueueChildren(E);
1898b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  if (E->isTypeOperand())
1899b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
1900b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek}
190155b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek
190255b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenekvoid EnqueueVisitor::VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr
190355b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek                                                     *E) {
190455b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  EnqueueChildren(E);
190555b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
190655b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek}
19071e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenekvoid EnqueueVisitor::VisitCXXUuidofExpr(CXXUuidofExpr *E) {
19081e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  EnqueueChildren(E);
19091e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  if (E->isTypeOperand())
19101e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
19111e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek}
1912e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenekvoid EnqueueVisitor::VisitDeclRefExpr(DeclRefExpr *DR) {
191360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  if (DR->hasExplicitTemplateArgs()) {
191460608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek    AddExplicitTemplateArgs(&DR->getExplicitTemplateArgs());
191560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  }
1916e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  WL.push_back(DeclRefExprParts(DR, Parent));
1917e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek}
1918f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E) {
1919f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
1920f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddDeclarationNameInfo(E);
192100cf3cc2718671aa48e8da264a523b0058a8591eDouglas Gregor  AddNestedNameSpecifierLoc(E->getQualifierLoc());
1922f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
1923035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekvoid EnqueueVisitor::VisitDeclStmt(DeclStmt *S) {
1924035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  unsigned size = WL.size();
1925035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  bool isFirst = true;
1926035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  for (DeclStmt::decl_iterator D = S->decl_begin(), DEnd = S->decl_end();
1927035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek       D != DEnd; ++D) {
1928035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    AddDecl(*D, isFirst);
1929035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    isFirst = false;
1930035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1931035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  if (size == WL.size())
1932035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return;
1933035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // Now reverse the entries we just added.  This will match the DFS
1934035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // ordering performed by the worklist.
1935035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  VisitorWorkList::iterator I = WL.begin() + size, E = WL.end();
1936035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  std::reverse(I, E);
1937035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek}
1938cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitDesignatedInitExpr(DesignatedInitExpr *E) {
1939cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getInit());
1940cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  typedef DesignatedInitExpr::Designator Designator;
1941cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (DesignatedInitExpr::reverse_designators_iterator
1942cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D = E->designators_rbegin(), DEnd = E->designators_rend();
1943cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D != DEnd; ++D) {
1944cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isFieldDesignator()) {
1945cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      if (FieldDecl *Field = D->getField())
1946cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        AddMemberRef(Field, D->getFieldLoc());
1947cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1948cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1949cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isArrayDesignator()) {
1950cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getArrayIndex(*D));
1951cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1952cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1953cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    assert(D->isArrayRangeDesignator() && "Unknown designator kind");
1954cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeEnd(*D));
1955cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeStart(*D));
1956cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1957cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
195828a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitExplicitCastExpr(ExplicitCastExpr *E) {
195928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(E);
196028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(E->getTypeInfoAsWritten());
196128a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
196228a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitForStmt(ForStmt *FS) {
196328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getBody());
196428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInc());
196528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getCond());
196628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(FS->getConditionVariable());
196728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInit());
196828a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1969ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekvoid EnqueueVisitor::VisitGotoStmt(GotoStmt *GS) {
1970ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  WL.push_back(LabelRefVisit(GS->getLabel(), GS->getLabelLoc(), Parent));
1971ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek}
197228a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitIfStmt(IfStmt *If) {
197328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getElse());
197428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getThen());
197528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getCond());
197628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(If->getConditionVariable());
197728a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
197828a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitInitListExpr(InitListExpr *IE) {
197928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  // We care about the syntactic form of the initializer list, only.
198028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (InitListExpr *Syntactic = IE->getSyntacticForm())
198128a719433411ef782b582946823bc648ddcc4533Ted Kremenek    IE = Syntactic;
198228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(IE);
198328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
198428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitMemberExpr(MemberExpr *M) {
198589629a746019a42797495b091711a1d68467e88aDouglas Gregor  WL.push_back(MemberExprParts(M, Parent));
198689629a746019a42797495b091711a1d68467e88aDouglas Gregor
198789629a746019a42797495b091711a1d68467e88aDouglas Gregor  // If the base of the member access expression is an implicit 'this', don't
198889629a746019a42797495b091711a1d68467e88aDouglas Gregor  // visit it.
198989629a746019a42797495b091711a1d68467e88aDouglas Gregor  // FIXME: If we ever want to show these implicit accesses, this will be
199089629a746019a42797495b091711a1d68467e88aDouglas Gregor  // unfortunate. However, clang_getCursor() relies on this behavior.
199175e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor  if (!M->isImplicitAccess())
199275e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor    AddStmt(M->getBase());
199328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
199473d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitObjCEncodeExpr(ObjCEncodeExpr *E) {
199573d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getEncodedTypeSourceInfo());
199673d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
199728a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitObjCMessageExpr(ObjCMessageExpr *M) {
199828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(M);
199928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(M->getClassReceiverTypeInfo());
200028a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
2001cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitOffsetOfExpr(OffsetOfExpr *E) {
2002cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the components of the offsetof expression.
2003cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (unsigned N = E->getNumComponents(), I = N; I > 0; --I) {
2004cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    typedef OffsetOfExpr::OffsetOfNode OffsetOfNode;
2005cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    const OffsetOfNode &Node = E->getComponent(I-1);
2006cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    switch (Node.getKind()) {
2007cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Array:
2008cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getIndexExpr(Node.getArrayExprIndex()));
2009cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2010cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Field:
201106dec892b5300b43263d25c5476b506c9d6cfbadAbramo Bagnara      AddMemberRef(Node.getField(), Node.getSourceRange().getEnd());
2012cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2013cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Identifier:
2014cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Base:
2015cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
2016cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
2017cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
2018cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the type into which we're computing the offset.
2019cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
2020cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
202128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitOverloadExpr(OverloadExpr *E) {
202260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
20236045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek  WL.push_back(OverloadExprParts(E, Parent));
20246045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek}
2025f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbournevoid EnqueueVisitor::VisitUnaryExprOrTypeTraitExpr(
2026f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne                                              UnaryExprOrTypeTraitExpr *E) {
20276d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  EnqueueChildren(E);
20286d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  if (E->isArgumentType())
20296d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek    AddTypeLoc(E->getArgumentTypeInfo());
20306d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
203128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitStmt(Stmt *S) {
203228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(S);
203328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
203428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitSwitchStmt(SwitchStmt *S) {
203528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getBody());
203628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getCond());
203728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(S->getConditionVariable());
203828a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
2039fafa75aebadef8d6b44a920e3f40529f150a5574Ted Kremenek
204028a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitWhileStmt(WhileStmt *W) {
204128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getBody());
204228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getCond());
204328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(W->getConditionVariable());
204428a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
204521ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
20462939b6f356161f572712d4d6310b65f9599e3675Ted Kremenekvoid EnqueueVisitor::VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E) {
20472939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  AddTypeLoc(E->getQueriedTypeSourceInfo());
20482939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek}
20496ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
20506ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichetvoid EnqueueVisitor::VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E) {
20516ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  AddTypeLoc(E->getRhsTypeSourceInfo());
20520a03a3f98b14006a54bcac9e8908a7c9f50e519fFrancois Pichet  AddTypeLoc(E->getLhsTypeSourceInfo());
20536ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet}
20546ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
205521ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegleyvoid EnqueueVisitor::VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E) {
205621ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  AddTypeLoc(E->getQueriedTypeSourceInfo());
205721ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley}
205821ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
2059552622067dc45013d240f73952fece703f5e63bdJohn Wiegleyvoid EnqueueVisitor::VisitExpressionTraitExpr(ExpressionTraitExpr *E) {
2060552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  EnqueueChildren(E);
2061552622067dc45013d240f73952fece703f5e63bdJohn Wiegley}
2062552622067dc45013d240f73952fece703f5e63bdJohn Wiegley
206328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U) {
206428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitOverloadExpr(U);
206528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (!U->isImplicitAccess())
206628a719433411ef782b582946823bc648ddcc4533Ted Kremenek    AddStmt(U->getBase());
206728a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
20689d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenekvoid EnqueueVisitor::VisitVAArgExpr(VAArgExpr *E) {
20699d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddStmt(E->getSubExpr());
20709d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddTypeLoc(E->getWrittenTypeInfo());
20719d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek}
207294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregorvoid EnqueueVisitor::VisitSizeOfPackExpr(SizeOfPackExpr *E) {
207394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  WL.push_back(SizeOfPackExprParts(E, Parent));
207494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor}
20754b9c2d235fb9449e249d74f48ecfec601650de93John McCallvoid EnqueueVisitor::VisitOpaqueValueExpr(OpaqueValueExpr *E) {
20764b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // If the opaque value has a source expression, just transparently
20774b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // visit that.  This is useful for (e.g.) pseudo-object expressions.
20784b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (Expr *SourceExpr = E->getSourceExpr())
20794b9c2d235fb9449e249d74f48ecfec601650de93John McCall    return Visit(SourceExpr);
20804b9c2d235fb9449e249d74f48ecfec601650de93John McCall  AddStmt(E);
20814b9c2d235fb9449e249d74f48ecfec601650de93John McCall}
20824b9c2d235fb9449e249d74f48ecfec601650de93John McCallvoid EnqueueVisitor::VisitPseudoObjectExpr(PseudoObjectExpr *E) {
20834b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // Treat the expression like its syntactic form.
20844b9c2d235fb9449e249d74f48ecfec601650de93John McCall  Visit(E->getSyntacticForm());
20854b9c2d235fb9449e249d74f48ecfec601650de93John McCall}
20866045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek
2087c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekvoid CursorVisitor::EnqueueWorkList(VisitorWorkList &WL, Stmt *S) {
2088aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  EnqueueVisitor(WL, MakeCXCursor(S, StmtParent, TU,RegionOfInterest)).Visit(S);
2089c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2090c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2091c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::IsInRegionOfInterest(CXCursor C) {
2092c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  if (RegionOfInterest.isValid()) {
2093c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SourceRange Range = getRawCursorExtent(C);
2094c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    if (Range.isInvalid() || CompareRegionOfInterest(Range))
2095c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      return false;
2096c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2097c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return true;
2098c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2099c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2100c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::RunVisitorWorkList(VisitorWorkList &WL) {
2101c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  while (!WL.empty()) {
2102c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Dequeue the worklist item.
210382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    VisitorJob LI = WL.back();
210482f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    WL.pop_back();
210582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek
2106c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Set the Parent field, then back to its old value once we're done.
2107c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SetParentRAII SetParent(Parent, StmtParent, LI.getParent());
2108c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2109c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    switch (LI.getKind()) {
2110f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      case VisitorJob::DeclVisitKind: {
211182f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Decl *D = cast<DeclVisit>(&LI)->get();
2112f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        if (!D)
2113f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek          continue;
2114f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2115f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // For now, perform default visitation for Decls.
2116aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        if (Visit(MakeCXCursor(D, TU, RegionOfInterest,
2117aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               cast<DeclVisit>(&LI)->isFirst())))
2118f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek            return true;
2119f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2120f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        continue;
2121f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      }
212260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      case VisitorJob::ExplicitTemplateArgsVisitKind: {
2123b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis        const ASTTemplateArgumentListInfo *ArgList =
212460608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          cast<ExplicitTemplateArgsVisit>(&LI)->get();
212560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        for (const TemplateArgumentLoc *Arg = ArgList->getTemplateArgs(),
212660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               *ArgEnd = Arg + ArgList->NumTemplateArgs;
212760608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               Arg != ArgEnd; ++Arg) {
212860608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          if (VisitTemplateArgumentLoc(*Arg))
212960608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek            return true;
213060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        }
213160608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        continue;
213260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      }
2133cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      case VisitorJob::TypeLocVisitKind: {
2134cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        // Perform default visitation for TypeLocs.
213582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        if (Visit(cast<TypeLocVisit>(&LI)->get()))
2136cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek          return true;
2137cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        continue;
2138cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      }
2139ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      case VisitorJob::LabelRefVisitKind: {
2140ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner        LabelDecl *LS = cast<LabelRefVisit>(&LI)->get();
2141e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        if (LabelStmt *stmt = LS->getStmt()) {
2142e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          if (Visit(MakeCursorLabelRef(stmt, cast<LabelRefVisit>(&LI)->getLoc(),
2143e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek                                       TU))) {
2144e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek            return true;
2145e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          }
2146e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        }
2147ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek        continue;
2148ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      }
214947695c8ad8424851f62e0d4a983b45b15daee1c5Ted Kremenek
2150f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      case VisitorJob::NestedNameSpecifierLocVisitKind: {
2151f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        NestedNameSpecifierLocVisit *V = cast<NestedNameSpecifierLocVisit>(&LI);
2152f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        if (VisitNestedNameSpecifierLoc(V->get()))
2153f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor          return true;
2154f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        continue;
2155f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      }
2156f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
2157f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      case VisitorJob::DeclarationNameInfoVisitKind: {
2158f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        if (VisitDeclarationNameInfo(cast<DeclarationNameInfoVisit>(&LI)
2159f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek                                     ->get()))
2160f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek          return true;
2161f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        continue;
2162f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      }
2163cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      case VisitorJob::MemberRefVisitKind: {
2164cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        MemberRefVisit *V = cast<MemberRefVisit>(&LI);
2165cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (Visit(MakeCursorMemberRef(V->get(), V->getLoc(), TU)))
2166cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          return true;
2167cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        continue;
2168cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      }
2169c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::StmtVisitKind: {
217082f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Stmt *S = cast<StmtVisit>(&LI)->get();
21718c269ac75569454a049385b1246140db5f2b6faaTed Kremenek        if (!S)
21728c269ac75569454a049385b1246140db5f2b6faaTed Kremenek          continue;
21738c269ac75569454a049385b1246140db5f2b6faaTed Kremenek
2174f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // Update the current cursor.
2175aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        CXCursor Cursor = MakeCXCursor(S, StmtParent, TU, RegionOfInterest);
2176cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (!IsInRegionOfInterest(Cursor))
2177cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          continue;
2178cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        switch (Visitor(Cursor, Parent, ClientData)) {
2179cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Break: return true;
2180cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Continue: break;
2181cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Recurse:
2182cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek            EnqueueWorkList(WL, S);
218382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek            break;
2184c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
218582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        continue;
2186c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2187c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::MemberExprPartsKind: {
2188c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Handle the other pieces in the MemberExpr besides the base.
218982f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        MemberExpr *M = cast<MemberExprParts>(&LI)->get();
2190c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2191c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the nested-name-specifier
219240d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = M->getQualifierLoc())
219340d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2194c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            return true;
2195c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2196c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the declaration name.
2197c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (VisitDeclarationNameInfo(M->getMemberNameInfo()))
2198c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          return true;
2199c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2200c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the explicitly-specified template arguments, if any.
2201c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (M->hasExplicitTemplateArgs()) {
2202c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          for (const TemplateArgumentLoc *Arg = M->getTemplateArgs(),
2203c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               *ArgEnd = Arg + M->getNumTemplateArgs();
2204c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               Arg != ArgEnd; ++Arg) {
2205c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            if (VisitTemplateArgumentLoc(*Arg))
2206c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek              return true;
2207c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          }
2208c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
2209c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        continue;
2210c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2211e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      case VisitorJob::DeclRefExprPartsKind: {
221282f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        DeclRefExpr *DR = cast<DeclRefExprParts>(&LI)->get();
2213e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit nested-name-specifier, if present.
221440d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = DR->getQualifierLoc())
221540d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2216e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek            return true;
2217e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit declaration name.
2218e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        if (VisitDeclarationNameInfo(DR->getNameInfo()))
2219e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek          return true;
2220e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        continue;
2221e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      }
22226045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      case VisitorJob::OverloadExprPartsKind: {
222382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        OverloadExpr *O = cast<OverloadExprParts>(&LI)->get();
22246045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the nested-name-specifier.
22254c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = O->getQualifierLoc())
22264c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
22276045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek            return true;
22286045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the declaration name.
22296045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (VisitDeclarationNameInfo(O->getNameInfo()))
22306045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22316045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the overloaded declaration reference.
22326045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (Visit(MakeCursorOverloadedDeclRef(O, TU)))
22336045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22346045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        continue;
22356045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      }
223694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      case VisitorJob::SizeOfPackExprPartsKind: {
223794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        SizeOfPackExpr *E = cast<SizeOfPackExprParts>(&LI)->get();
223894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        NamedDecl *Pack = E->getPack();
223994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTypeParmDecl>(Pack)) {
224094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTypeRef(cast<TemplateTypeParmDecl>(Pack),
224194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                      E->getPackLoc(), TU)))
224294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
224394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
224494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
224594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
224694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
224794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTemplateParmDecl>(Pack)) {
224894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTemplateRef(cast<TemplateTemplateParmDecl>(Pack),
224994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                          E->getPackLoc(), TU)))
225094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
225194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
225294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
225394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
225494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
225594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // Non-type template parameter packs and function parameter packs are
225694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // treated like DeclRefExpr cursors.
225794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        continue;
225894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      }
2259c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    }
2260c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2261c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return false;
2262c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2263c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2264cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekbool CursorVisitor::Visit(Stmt *S) {
2265d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  VisitorWorkList *WL = 0;
2266d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  if (!WorkListFreeList.empty()) {
2267d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = WorkListFreeList.back();
2268d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL->clear();
2269d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListFreeList.pop_back();
2270d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2271d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  else {
2272d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = new VisitorWorkList();
2273d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListCache.push_back(WL);
2274d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2275d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  EnqueueWorkList(*WL, S);
2276d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  bool result = RunVisitorWorkList(*WL);
2277d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  WorkListFreeList.push_back(WL);
2278d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  return result;
2279c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2280c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
228148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichetnamespace {
228248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichettypedef llvm::SmallVector<SourceRange, 4> RefNamePieces;
228348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois PichetRefNamePieces buildPieces(unsigned NameFlags, bool IsMemberRefExpr,
228448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const DeclarationNameInfo &NI,
228548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const SourceRange &QLoc,
2286b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis                          const ASTTemplateArgumentListInfo *TemplateArgs = 0){
228748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantQualifier = NameFlags & CXNameRange_WantQualifier;
228848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantTemplateArgs = NameFlags & CXNameRange_WantTemplateArgs;
228948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantSinglePiece = NameFlags & CXNameRange_WantSinglePiece;
229048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const DeclarationName::NameKind Kind = NI.getName().getNameKind();
229248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  RefNamePieces Pieces;
229448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantQualifier && QLoc.isValid())
229648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(QLoc);
229748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind != DeclarationName::CXXOperatorName || IsMemberRefExpr)
229948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(NI.getLoc());
230048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
230148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantTemplateArgs && TemplateArgs)
230248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceRange(TemplateArgs->LAngleLoc,
230348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                                 TemplateArgs->RAngleLoc));
230448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
230548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind == DeclarationName::CXXOperatorName) {
230648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
230748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.BeginOpNameLoc));
230848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
230948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.EndOpNameLoc));
231048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
231148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
231248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantSinglePiece) {
231348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    SourceRange R(Pieces.front().getBegin(), Pieces.back().getEnd());
231448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.clear();
231548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(R);
231648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
231748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
231848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  return Pieces;
231948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
232048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
232148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
2322c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2323c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Misc. API hooks.
2324c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2325c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
23268c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic llvm::sys::Mutex EnableMultithreadingMutex;
23278c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic bool EnabledMultithreading;
23288c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
23295e4bc590b0ea010e38372d0b4a0aab578a746fe6Benjamin Kramerextern "C" {
23300a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas GregorCXIndex clang_createIndex(int excludeDeclarationsFromPCH,
23310a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          int displayDiagnostics) {
233248615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // Disable pretty stack trace functionality, which will otherwise be a very
233348615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // poor citizen of the world and set up all sorts of signal handlers.
233448615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  llvm::DisablePrettyStackTrace = true;
233548615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar
2336c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // We use crash recovery to make some of our APIs more reliable, implicitly
2337c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // enable it.
2338c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  llvm::CrashRecoveryContext::Enable();
2339c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar
23408c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  // Enable support for multithreading in LLVM.
23418c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  {
23428c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    llvm::sys::ScopedLock L(EnableMultithreadingMutex);
23438c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    if (!EnabledMultithreading) {
23448c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      llvm::llvm_start_multithreaded();
23458c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      EnabledMultithreading = true;
23468c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    }
23478c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  }
23488c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
2349a030b7cf5e6aad5889b1b662b6979840bc75f87fDouglas Gregor  CIndexer *CIdxr = new CIndexer();
2350e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  if (excludeDeclarationsFromPCH)
2351e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    CIdxr->setOnlyLocalDecls();
23520a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (displayDiagnostics)
23530a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    CIdxr->setDisplayDiagnostics();
2354e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  return CIdxr;
2355600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2356600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
23579ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeIndex(CXIndex CIdx) {
23582b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (CIdx)
23592b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    delete static_cast<CIndexer *>(CIdx);
23602bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
23612bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff
2362d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenekvoid clang_toggleCrashRecovery(unsigned isEnabled) {
2363d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  if (isEnabled)
2364d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Enable();
2365d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  else
2366d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Disable();
2367d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek}
2368d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek
23699ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit clang_createTranslationUnit(CXIndex CIdx,
2370a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                              const char *ast_filename) {
23712b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
23722b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    return 0;
2373f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
23747d1d49d2971b20a97b3c2a301470b9eaaa130137Douglas Gregor  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2375389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOptions FileSystemOpts;
2376389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOpts.WorkingDir = CXXIdx->getWorkingDirectory();
23770d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
2378d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie  llvm::IntrusiveRefCntPtr<DiagnosticsEngine> Diags;
2379a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *TU = ASTUnit::LoadFromASTFile(ast_filename, Diags, FileSystemOpts,
2380a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                  CXXIdx->getOnlyLocalDecls(),
2381a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                  0, 0, true);
2382a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return MakeCXTranslationUnit(TU);
2383600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2384600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2385b1c031be513705d924038f497279b9b599868ba1Douglas Gregorunsigned clang_defaultEditingTranslationUnitOptions() {
23862a2c50b330e7754499f42173616a36865b5f313bDouglas Gregor  return CXTranslationUnit_PrecompiledPreamble |
2387b5af843a20e237ad1a13ad66a867e200695b8c8eDouglas Gregor         CXTranslationUnit_CacheCompletionResults;
2388b1c031be513705d924038f497279b9b599868ba1Douglas Gregor}
2389b1c031be513705d924038f497279b9b599868ba1Douglas Gregor
23909ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit
23919ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarclang_createTranslationUnitFromSourceFile(CXIndex CIdx,
23929ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          const char *source_filename,
23939ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          int num_command_line_args,
23942ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                          const char * const *command_line_args,
23954db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                                          unsigned num_unsaved_files,
2396a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                          struct CXUnsavedFile *unsaved_files) {
2397dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  unsigned Options = CXTranslationUnit_DetailedPreprocessingRecord |
2398ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth                     CXTranslationUnit_NestedMacroExpansions;
23995a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor  return clang_parseTranslationUnit(CIdx, source_filename,
24005a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    command_line_args, num_command_line_args,
24015a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    unsaved_files, num_unsaved_files,
2402dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                    Options);
24035a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor}
240419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
240519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbarstruct ParseTranslationUnitInfo {
240619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx;
240719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename;
24082ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char *const *command_line_args;
240919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args;
241019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
241119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files;
241219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options;
241319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXTranslationUnit result;
241419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar};
2415b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_parseTranslationUnit_Impl(void *UserData) {
241619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo *PTUI =
241719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    static_cast<ParseTranslationUnitInfo*>(UserData);
241819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx = PTUI->CIdx;
241919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename = PTUI->source_filename;
24202ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char * const *command_line_args = PTUI->command_line_args;
242119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args = PTUI->num_command_line_args;
242219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files = PTUI->unsaved_files;
242319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files = PTUI->num_unsaved_files;
242419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options = PTUI->options;
242519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  PTUI->result = 0;
24265a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor
24272b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
242819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return;
2429f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2430e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2431e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff
243244c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  bool PrecompilePreamble = options & CXTranslationUnit_PrecompiledPreamble;
2433467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  // FIXME: Add a flag for modules.
2434467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  TranslationUnitKind TUKind
2435467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor    = (options & CXTranslationUnit_Incomplete)? TU_Prefix : TU_Complete;
243687c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  bool CacheCodeCompetionResults
243787c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    = options & CXTranslationUnit_CacheCompletionResults;
243887c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor
24395352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  // Configure the diagnostics.
24405352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  DiagnosticOptions DiagOpts;
2441d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie  llvm::IntrusiveRefCntPtr<DiagnosticsEngine>
244225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Diags(CompilerInstance::createDiagnostics(DiagOpts, num_command_line_args,
244325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                                command_line_args));
244425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
244525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
2446d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie  llvm::CrashRecoveryContextCleanupRegistrar<DiagnosticsEngine,
2447d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie    llvm::CrashRecoveryContextReleaseRefCleanup<DiagnosticsEngine> >
244825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    DiagCleanup(Diags.getPtr());
244925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
245025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<ASTUnit::RemappedFile> >
245125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
245225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
245325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
245425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
245525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
2456f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
24574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
24585f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2459f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    const llvm::MemoryBuffer *Buffer
2460a0a270c0f1c0a4e3482438bdc5f4a7bd3d25f0a6Chris Lattner      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
246125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
246225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
24634db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
2464f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
246525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<const char *> >
246625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args(new std::vector<const char*>());
246725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
246825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this method.
246925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<std::vector<const char*> >
247025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    ArgsCleanup(Args.get());
247125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
247252ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // Since the Clang C library is primarily used by batch tools dealing with
247352ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // (often very broken) source code, where spell-checking can have a
247452ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // significant negative impact on performance (particularly when
247552ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // precompiled headers are involved), we disable it by default.
2476b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  // Only do this if we haven't found a spell-checking-related argument.
2477b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  bool FoundSpellCheckingArgument = false;
2478b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  for (int I = 0; I != num_command_line_args; ++I) {
2479b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (strcmp(command_line_args[I], "-fno-spell-checking") == 0 ||
2480b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        strcmp(command_line_args[I], "-fspell-checking") == 0) {
2481b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      FoundSpellCheckingArgument = true;
2482b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      break;
2483e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    }
2484b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  }
2485b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  if (!FoundSpellCheckingArgument)
248625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-fno-spell-checking");
2487b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
248825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  Args->insert(Args->end(), command_line_args,
248925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek               command_line_args + num_command_line_args);
2490d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2491c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // The 'source_filename' argument is optional.  If the caller does not
2492c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // specify it then it is assumed that the source file is specified
2493c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // in the actual argument list.
2494c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // Put the source file after command_line_args otherwise if '-x' flag is
2495c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // present it will be unused.
2496c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  if (source_filename)
249725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back(source_filename);
2498c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis
249944c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  // Do we need the detailed preprocessing record?
2500ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth  bool NestedMacroExpansions = false;
250144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (options & CXTranslationUnit_DetailedPreprocessingRecord) {
250225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-Xclang");
250325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-detailed-preprocessing-record");
2504ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth    NestedMacroExpansions
2505ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth      = (options & CXTranslationUnit_NestedMacroExpansions);
250644c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  }
250744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
2508026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  unsigned NumErrors = Diags->getClient()->getNumErrors();
2509b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  llvm::OwningPtr<ASTUnit> Unit(
25104ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek    ASTUnit::LoadFromCommandLine(Args->size() ? &(*Args)[0] : 0
25114ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 /* vector::data() not portable */,
25124ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 Args->size() ? (&(*Args)[0] + Args->size()) :0,
2513b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 Diags,
2514b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getClangResourcesPath(),
2515b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getOnlyLocalDecls(),
2516e47be3e9682e82da15059006f43c7f3c021e4fffDouglas Gregor                                 /*CaptureDiagnostics=*/true,
25174ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 RemappedFiles->size() ? &(*RemappedFiles)[0]:0,
251825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                 RemappedFiles->size(),
2519299a4a967b02c9f0d0d94ad8560e3ced893f9116Argyrios Kyrtzidis                                 /*RemappedFilesKeepOriginalName=*/true,
2520b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 PrecompilePreamble,
2521467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor                                 TUKind,
252299ba202f659e1885fa5ee114f97c97cf6a857491Douglas Gregor                                 CacheCodeCompetionResults,
2523ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth                                 NestedMacroExpansions));
2524b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
2525026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  if (NumErrors != Diags->getClient()->getNumErrors()) {
2526b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    // Make sure to check that 'Unit' is non-NULL.
2527b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (CXXIdx->getDisplayDiagnostics() && Unit.get()) {
2528b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      for (ASTUnit::stored_diag_iterator D = Unit->stored_diag_begin(),
2529b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                      DEnd = Unit->stored_diag_end();
2530b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor           D != DEnd; ++D) {
2531b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        CXStoredDiagnostic Diag(*D, Unit->getASTContext().getLangOptions());
2532b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        CXString Msg = clang_formatDiagnostic(&Diag,
2533b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                    clang_defaultDiagnosticDisplayOptions());
2534b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        fprintf(stderr, "%s\n", clang_getCString(Msg));
2535b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        clang_disposeString(Msg);
2536b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      }
2537274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor#ifdef LLVM_ON_WIN32
2538b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // On Windows, force a flush, since there may be multiple copies of
2539b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // stderr and stdout in the file system, all with different buffers
2540b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // but writing to the same device.
2541b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      fflush(stderr);
2542b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor#endif
2543b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    }
2544a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  }
2545d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2546a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  PTUI->result = MakeCXTranslationUnit(Unit.take());
254719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar}
254819ffd492a31a25fb691098bf79f317e5f3edf177Daniel DunbarCXTranslationUnit clang_parseTranslationUnit(CXIndex CIdx,
254919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             const char *source_filename,
25502ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                         const char * const *command_line_args,
255119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             int num_command_line_args,
25529e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                            struct CXUnsavedFile *unsaved_files,
255319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned num_unsaved_files,
255419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned options) {
255519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo PTUI = { CIdx, source_filename, command_line_args,
25569e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_command_line_args, unsaved_files,
25579e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_unsaved_files, options, 0 };
255819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  llvm::CrashRecoveryContext CRC;
255919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
2560bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_parseTranslationUnit_Impl, &PTUI)) {
256160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "libclang: crash detected during parsing: {\n");
256260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'source_filename' : '%s'\n", source_filename);
256360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'command_line_args' : [");
256460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (int i = 0; i != num_command_line_args; ++i) {
256560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
256660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
256760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "'%s'", command_line_args[i]);
256860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
256960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
257060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'unsaved_files' : [");
257160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (unsigned i = 0; i != num_unsaved_files; ++i) {
257260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
257360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
257460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "('%s', '...', %ld)", unsaved_files[i].Filename,
257560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar              unsaved_files[i].Length);
257660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
257760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
257860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'options' : %d,\n", options);
257960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "}\n");
258060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar
258119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return 0;
25826df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE")) {
25836df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(PTUI.result);
258419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  }
25856df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
258619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  return PTUI.result;
25875b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff}
25885b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff
25891999844e7a18786e61e619e1dc6c789827541863Douglas Gregorunsigned clang_defaultSaveOptions(CXTranslationUnit TU) {
25901999844e7a18786e61e619e1dc6c789827541863Douglas Gregor  return CXSaveTranslationUnit_None;
25911999844e7a18786e61e619e1dc6c789827541863Douglas Gregor}
25921999844e7a18786e61e619e1dc6c789827541863Douglas Gregor
25931999844e7a18786e61e619e1dc6c789827541863Douglas Gregorint clang_saveTranslationUnit(CXTranslationUnit TU, const char *FileName,
25941999844e7a18786e61e619e1dc6c789827541863Douglas Gregor                              unsigned options) {
25957ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU)
259639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    return CXSaveError_InvalidTU;
25977ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
259839c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  CXSaveError result = static_cast<ASTUnit *>(TU->TUData)->Save(FileName);
25996df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  if (getenv("LIBCLANG_RESOURCE_USAGE"))
26006df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
26016df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  return result;
26027ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
260319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
26049ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeTranslationUnit(CXTranslationUnit CTUnit) {
2605ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  if (CTUnit) {
2606ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // If the translation unit has been marked as unsafe to free, just discard
2607ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // it.
2608a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    if (static_cast<ASTUnit *>(CTUnit->TUData)->isUnsafeToFree())
2609ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar      return;
2610ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2611a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete static_cast<ASTUnit *>(CTUnit->TUData);
2612a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    disposeCXStringPool(CTUnit->StringPool);
2613153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    delete static_cast<CXDiagnosticSetImpl *>(CTUnit->Diagnostics);
2614a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete CTUnit;
2615ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  }
26162bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
26170d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
2618e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregorunsigned clang_defaultReparseOptions(CXTranslationUnit TU) {
2619e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor  return CXReparse_None;
2620e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor}
2621e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor
2622ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarstruct ReparseTranslationUnitInfo {
2623ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU;
2624ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files;
2625ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
2626ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options;
2627ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  int result;
2628ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar};
2629593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2630b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_reparseTranslationUnit_Impl(void *UserData) {
2631ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo *RTUI =
2632ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    static_cast<ReparseTranslationUnitInfo*>(UserData);
2633ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU = RTUI->TU;
2634153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2635153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  // Reset the associated diagnostics.
2636153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  delete static_cast<CXDiagnosticSetImpl*>(TU->Diagnostics);
2637153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  TU->Diagnostics = 0;
2638153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2639ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files = RTUI->num_unsaved_files;
2640ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files = RTUI->unsaved_files;
2641ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options = RTUI->options;
2642ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  (void) options;
2643ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  RTUI->result = 1;
2644ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2645abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU)
2646ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return;
2647593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2648a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
2649593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
2650abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
265125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<ASTUnit::RemappedFile> >
265225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
265325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
265425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
265525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
265625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
265725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
2658abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
26595f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2660abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    const llvm::MemoryBuffer *Buffer
26611abc6bca3d7fb0e7b1e40fbcad6cfb5e10594548Douglas Gregor      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
266225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
266325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
2664abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
2665abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
26664ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek  if (!CXXUnit->Reparse(RemappedFiles->size() ? &(*RemappedFiles)[0] : 0,
26674ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                        RemappedFiles->size()))
2668593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor    RTUI->result = 0;
2669abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
2670593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2671ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarint clang_reparseTranslationUnit(CXTranslationUnit TU,
2672ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned num_unsaved_files,
2673ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 struct CXUnsavedFile *unsaved_files,
2674ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned options) {
2675ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo RTUI = { TU, num_unsaved_files, unsaved_files,
2676ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                      options, 0 };
26778c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis
2678e7de9b4a1f4a15620ab15bc8159018df7d54080aArgyrios Kyrtzidis  if (getenv("LIBCLANG_NOTHREADS")) {
26798c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis    clang_reparseTranslationUnit_Impl(&RTUI);
26808c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis    return RTUI.result;
26818c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis  }
26828c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis
2683ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  llvm::CrashRecoveryContext CRC;
2684ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2685bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_reparseTranslationUnit_Impl, &RTUI)) {
2686b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbar    fprintf(stderr, "libclang: crash detected during reparsing\n");
2687a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    static_cast<ASTUnit *>(TU->TUData)->setUnsafeToFree(true);
2688ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return 1;
26896df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE"))
26906df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
26911dfb26af4d6aa4f7818e256659a79f1ec2cba784Ted Kremenek
2692ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  return RTUI.result;
2693ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar}
2694ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2695df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
26969ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getTranslationUnitSpelling(CXTranslationUnit CTUnit) {
26972b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CTUnit)
2698ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
2699f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2700a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(CTUnit->TUData);
2701ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(CXXUnit->getOriginalSourceFileName(), true);
2702af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
27031eb79b58e56b99cf557d5d353586a10c5360364dDaniel Dunbar
27047eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas GregorCXCursor clang_getTranslationUnitCursor(CXTranslationUnit TU) {
2705aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Result = { CXCursor_TranslationUnit, 0, { 0, 0, TU } };
27067eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return Result;
27077eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
27087eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
2709fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2710600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2711fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2712fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXFile Operations.
2713fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2714fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2715fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
271674844072411bae91d5dbb89955d200cbe1e0a1c8Ted KremenekCXString clang_getFileName(CXFile SFile) {
271798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
2718a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return createCXString((const char*)NULL);
2719f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
272088145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
272174844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  return createCXString(FEnt->getName());
272288145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
272388145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff
272488145034694ed5267fa6fa5febc54fadc02bd479Steve Narofftime_t clang_getFileTime(CXFile SFile) {
272598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
272698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    return 0;
2727f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
272888145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
272988145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  return FEnt->getModificationTime();
2730ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff}
2731f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2732b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXFile clang_getFile(CXTranslationUnit tu, const char *file_name) {
2733b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!tu)
2734b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor    return 0;
2735f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2736a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2737f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2738b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  FileManager &FMgr = CXXUnit->getFileManager();
273939b49bcaaddb1049234fca9500c0ac02c088e23dChris Lattner  return const_cast<FileEntry *>(FMgr.getFile(file_name));
2740b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2741f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2742dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorunsigned clang_isFileMultipleIncludeGuarded(CXTranslationUnit tu, CXFile file) {
2743dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  if (!tu || !file)
2744dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor    return 0;
2745dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2746dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2747dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  FileEntry *FEnt = static_cast<FileEntry *>(file);
2748dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  return CXXUnit->getPreprocessor().getHeaderSearchInfo()
2749dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor                                          .isFileMultipleIncludeGuarded(FEnt);
2750dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor}
2751dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2752fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2753fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2754fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2755fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXCursor Operations.
2756fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2757fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2758fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekstatic Decl *getDeclFromExpr(Stmt *E) {
2759c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2760db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return getDeclFromExpr(CE->getSubExpr());
2761db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2762fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (DeclRefExpr *RefExpr = dyn_cast<DeclRefExpr>(E))
2763fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RefExpr->getDecl();
276438f28c1189142429384996409fffbc57f67b5c60Douglas Gregor  if (BlockDeclRefExpr *RefExpr = dyn_cast<BlockDeclRefExpr>(E))
276538f28c1189142429384996409fffbc57f67b5c60Douglas Gregor    return RefExpr->getDecl();
2766fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (MemberExpr *ME = dyn_cast<MemberExpr>(E))
2767fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return ME->getMemberDecl();
2768fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCIvarRefExpr *RE = dyn_cast<ObjCIvarRefExpr>(E))
2769fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RE->getDecl();
2770db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCPropertyRefExpr *PRE = dyn_cast<ObjCPropertyRefExpr>(E))
277112f78a6741a4cb3d904340f8d3d2714568b50e7aJohn McCall    return PRE->isExplicitProperty() ? PRE->getExplicitProperty() : 0;
27724b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (PseudoObjectExpr *POE = dyn_cast<PseudoObjectExpr>(E))
27734b9c2d235fb9449e249d74f48ecfec601650de93John McCall    return getDeclFromExpr(POE->getSyntacticForm());
27744b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (OpaqueValueExpr *OVE = dyn_cast<OpaqueValueExpr>(E))
27754b9c2d235fb9449e249d74f48ecfec601650de93John McCall    if (Expr *Src = OVE->getSourceExpr())
27764b9c2d235fb9449e249d74f48ecfec601650de93John McCall      return getDeclFromExpr(Src);
2777db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2778fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (CallExpr *CE = dyn_cast<CallExpr>(E))
2779fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return getDeclFromExpr(CE->getCallee());
27805f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  if (CXXConstructExpr *CE = dyn_cast<CXXConstructExpr>(E))
278193798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    if (!CE->isElidable())
278293798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CE->getConstructor();
2783fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCMessageExpr *OME = dyn_cast<ObjCMessageExpr>(E))
2784fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return OME->getMethodDecl();
2785f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2786db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCProtocolExpr *PE = dyn_cast<ObjCProtocolExpr>(E))
2787db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return PE->getProtocol();
2788c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor  if (SubstNonTypeTemplateParmPackExpr *NTTP
2789c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor                              = dyn_cast<SubstNonTypeTemplateParmPackExpr>(E))
2790c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor    return NTTP->getParameterPack();
279194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
279294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    if (isa<NonTypeTemplateParmDecl>(SizeOfPack->getPack()) ||
279394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        isa<ParmVarDecl>(SizeOfPack->getPack()))
279494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      return SizeOfPack->getPack();
2795db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2796fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  return 0;
2797fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek}
2798ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff
2799c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbarstatic SourceLocation getLocationFromExpr(Expr *E) {
2800c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2801c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis    return getLocationFromExpr(CE->getSubExpr());
2802c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis
2803c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCMessageExpr *Msg = dyn_cast<ObjCMessageExpr>(E))
2804c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return /*FIXME:*/Msg->getLeftLoc();
2805c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E))
2806c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return DRE->getLocation();
280738f28c1189142429384996409fffbc57f67b5c60Douglas Gregor  if (BlockDeclRefExpr *RefExpr = dyn_cast<BlockDeclRefExpr>(E))
280838f28c1189142429384996409fffbc57f67b5c60Douglas Gregor    return RefExpr->getLocation();
2809c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (MemberExpr *Member = dyn_cast<MemberExpr>(E))
2810c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Member->getMemberLoc();
2811c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCIvarRefExpr *Ivar = dyn_cast<ObjCIvarRefExpr>(E))
2812c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Ivar->getLocation();
281394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
281494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    return SizeOfPack->getPackLoc();
281594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
2816c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  return E->getLocStart();
2817c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar}
2818c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar
2819fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
2820f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2821f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekunsigned clang_visitChildren(CXCursor parent,
2822b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXCursorVisitor visitor,
2823b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXClientData client_data) {
2824f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  CursorVisitor CursorVis(getCursorTU(parent), visitor, client_data,
2825f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                          /*VisitPreprocessorLast=*/false);
2826b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return CursorVis.VisitChildren(parent);
2827b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
2828b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor
28293387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#ifndef __has_feature
28303387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#define __has_feature(x) 0
28313387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
28323387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#if __has_feature(blocks)
28333387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef enum CXChildVisitResult
28343387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall     (^CXCursorVisitorBlock)(CXCursor cursor, CXCursor parent);
28353387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
28363387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
28373387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
28383387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
28393387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block(cursor, parent);
28403387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
28413387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#else
28423387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// If we are compiled with a compiler that doesn't have native blocks support,
28433387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// define and call the block manually, so the
28443387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef struct _CXChildVisitResult
28453387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall{
28463387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	void *isa;
28473387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int flags;
28483387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int reserved;
28499e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar	enum CXChildVisitResult(*invoke)(struct _CXChildVisitResult*, CXCursor,
28509e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                         CXCursor);
28513387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall} *CXCursorVisitorBlock;
28523387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
28533387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
28543387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
28553387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
28563387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block->invoke(block, cursor, parent);
28573387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
28583387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
28593387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
28603387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
28619e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbarunsigned clang_visitChildrenWithBlock(CXCursor parent,
28629e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                      CXCursorVisitorBlock block) {
28633387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return clang_visitChildren(parent, visitWithBlock, block);
28643387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
28653387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
286678205d4bada39d95097e766af9eb30cdd0159461Douglas Gregorstatic CXString getDeclSpelling(Decl *D) {
286778205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  NamedDecl *ND = dyn_cast_or_null<NamedDecl>(D);
2868e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  if (!ND) {
28695f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
2870e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
2871e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return createCXString(Property->getIdentifier()->getName());
2872e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
2873ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
2874e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  }
2875e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
287678205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCMethodDecl *OMD = dyn_cast<ObjCMethodDecl>(ND))
2877ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(OMD->getSelector().getAsString());
2878f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
287978205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCCategoryImplDecl *CIMP = dyn_cast<ObjCCategoryImplDecl>(ND))
288078205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // No, this isn't the same as the code below. getIdentifier() is non-virtual
288178205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // and returns different names. NamedDecl returns the class name and
288278205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // ObjCCategoryImplDecl returns the category name.
2883ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(CIMP->getIdentifier()->getNameStart());
2884f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
28850a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  if (isa<UsingDirectiveDecl>(D))
28860a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("");
28870a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
288850aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  llvm::SmallString<1024> S;
288950aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  llvm::raw_svector_ostream os(S);
289050aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  ND->printName(os);
289150aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek
289250aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  return createCXString(os.str());
289378205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor}
2894f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
28959ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getCursorSpelling(CXCursor C) {
28967eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  if (clang_isTranslationUnit(C.kind))
2897a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return clang_getTranslationUnitSpelling(
2898a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                            static_cast<CXTranslationUnit>(C.data[2]));
28997eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
2900f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  if (clang_isReference(C.kind)) {
2901f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    switch (C.kind) {
2902acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCSuperClassRef: {
29032e331b938b38057e333fab0ba841130ea8467794Douglas Gregor      ObjCInterfaceDecl *Super = getCursorObjCSuperClassRef(C).first;
2904ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Super->getIdentifier()->getNameStart());
2905acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
2906acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCClassRef: {
29071adb082a709f7b588f03672999294e061234b2cfDouglas Gregor      ObjCInterfaceDecl *Class = getCursorObjCClassRef(C).first;
2908ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Class->getIdentifier()->getNameStart());
2909acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
2910acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCProtocolRef: {
291178db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor      ObjCProtocolDecl *OID = getCursorObjCProtocolRef(C).first;
2912f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      assert(OID && "getCursorSpelling(): Missing protocol decl");
2913ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(OID->getIdentifier()->getNameStart());
2914acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
29153064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
29163064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = getCursorCXXBaseSpecifier(C);
29173064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return createCXString(B->getType().getAsString());
29183064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
29197d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    case CXCursor_TypeRef: {
29207d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      TypeDecl *Type = getCursorTypeRef(C).first;
29217d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      assert(Type && "Missing type decl");
29227d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
2923ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(getCursorContext(C).getTypeDeclType(Type).
2924ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                              getAsString());
29257d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
29260b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
29270b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      TemplateDecl *Template = getCursorTemplateRef(C).first;
29286931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(Template && "Missing template decl");
29290b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
29300b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString(Template->getNameAsString());
29310b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
29326931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
29336931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
29346931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      NamedDecl *NS = getCursorNamespaceRef(C).first;
29356931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(NS && "Missing namespace decl");
29366931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
29376931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return createCXString(NS->getNameAsString());
29386931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
29397d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
2940a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
2941a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      FieldDecl *Field = getCursorMemberRef(C).first;
2942a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      assert(Field && "Missing member decl");
2943a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
2944a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return createCXString(Field->getNameAsString());
2945a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
2946a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
294736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
294836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      LabelStmt *Label = getCursorLabelRef(C).first;
294936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      assert(Label && "Missing label");
295036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
2951ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
295236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
295336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
29541f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef: {
29551f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
29561f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Decl *D = Storage.dyn_cast<Decl *>()) {
29571f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        if (NamedDecl *ND = dyn_cast<NamedDecl>(D))
29581f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          return createCXString(ND->getNameAsString());
29591f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
29601f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
29611f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
29621f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString(E->getName().getAsString());
29631f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedTemplateStorage *Ovl
29641f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        = Storage.get<OverloadedTemplateStorage*>();
29651f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Ovl->size() == 0)
29661f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
29671f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return createCXString((*Ovl->begin())->getNameAsString());
29681f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    }
29691f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
2970acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    default:
2971ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString("<not implemented>");
2972f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    }
2973f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  }
297497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
297597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
297697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    Decl *D = getDeclFromExpr(getCursorExpr(C));
297797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    if (D)
297878205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor      return getDeclSpelling(D);
2979ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
298097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
298197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
298236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
298336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
298436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (LabelStmt *Label = dyn_cast_or_null<LabelStmt>(S))
2985ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
298636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
298736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("");
298836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
298936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
29909b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
29919e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    return createCXString(getCursorMacroExpansion(C)->getName()
29924ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor                                                           ->getNameStart());
29934ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor
2994572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition)
2995572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString(getCursorMacroDefinition(C)->getName()
2996572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor                                                           ->getNameStart());
2997572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
2998ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective)
2999ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString(getCursorInclusionDirective(C)->getFileName());
3000ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
300160cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor  if (clang_isDeclaration(C.kind))
300260cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor    return getDeclSpelling(getCursorDecl(C));
3003e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
30045f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  if (C.kind == CXCursor_AnnotateAttr) {
30055f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    AnnotateAttr *AA = cast<AnnotateAttr>(cxcursor::getCursorAttr(C));
30065f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    return createCXString(AA->getAnnotation());
30075f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  }
30085f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen
3009ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString("");
3010f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3011f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
3012358559d8d7b458c5f64941842383a16e61f0828dDouglas GregorCXString clang_getCursorDisplayName(CXCursor C) {
3013358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!clang_isDeclaration(C.kind))
3014358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return clang_getCursorSpelling(C);
3015358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3016358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  Decl *D = getCursorDecl(C);
3017358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!D)
3018358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString("");
3019358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
302030c42404202d2e2512e51efc6066bd614cfdb5a4Douglas Gregor  PrintingPolicy Policy = getCursorContext(C).getPrintingPolicy();
3021358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast<FunctionTemplateDecl>(D))
3022358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    D = FunTmpl->getTemplatedDecl();
3023358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3024358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionDecl *Function = dyn_cast<FunctionDecl>(D)) {
3025358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3026358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3027358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << Function->getNameAsString();
3028358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->getPrimaryTemplate())
3029358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "<>";
3030358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "(";
3031358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Function->getNumParams(); I != N; ++I) {
3032358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3033358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3034358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << Function->getParamDecl(I)->getType().getAsString(Policy);
3035358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3036358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3037358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->isVariadic()) {
3038358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Function->getNumParams())
3039358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3040358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "...";
3041358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3042358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ")";
3043358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3044358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3045358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3046358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateDecl *ClassTemplate = dyn_cast<ClassTemplateDecl>(D)) {
3047358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3048358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3049358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ClassTemplate->getNameAsString();
3050358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "<";
3051358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    TemplateParameterList *Params = ClassTemplate->getTemplateParameters();
3052358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Params->size(); I != N; ++I) {
3053358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3054358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3055358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3056358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      NamedDecl *Param = Params->getParam(I);
3057358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Param->getIdentifier()) {
3058358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << Param->getIdentifier()->getName();
3059358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        continue;
3060358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      }
3061358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3062358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // There is no parameter name, which makes this tricky. Try to come up
3063358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // with something useful that isn't too long.
3064358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(Param))
3065358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << (TTP->wasDeclaredWithTypename()? "typename" : "class");
3066358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else if (NonTypeTemplateParmDecl *NTTP
3067358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                    = dyn_cast<NonTypeTemplateParmDecl>(Param))
3068358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << NTTP->getType().getAsString(Policy);
3069358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else
3070358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << "template<...> class";
3071358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3072358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3073358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ">";
3074358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3075358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3076358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3077358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateSpecializationDecl *ClassSpec
3078358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
3079358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    // If the type was explicitly written, use that.
3080358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (TypeSourceInfo *TSInfo = ClassSpec->getTypeAsWritten())
3081358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      return createCXString(TSInfo->getType().getAsString(Policy));
3082358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3083358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3084358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3085358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ClassSpec->getNameAsString();
3086358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << TemplateSpecializationType::PrintTemplateArgumentList(
3087910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().data(),
3088910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().size(),
3089358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                                                Policy);
3090358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3091358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3092358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3093358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  return clang_getCursorSpelling(C);
3094358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor}
3095358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3096e68fff6fc083c6270d835216a3de0b82c6ef0310Ted KremenekCXString clang_getCursorKindSpelling(enum CXCursorKind Kind) {
309789922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  switch (Kind) {
3098e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FunctionDecl:
3099e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FunctionDecl");
3100e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypedefDecl:
3101e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypedefDecl");
3102e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumDecl:
3103e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumDecl");
3104e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumConstantDecl:
3105e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumConstantDecl");
3106e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_StructDecl:
3107e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("StructDecl");
3108e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnionDecl:
3109e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnionDecl");
3110e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ClassDecl:
3111e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ClassDecl");
3112e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FieldDecl:
3113e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FieldDecl");
3114e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_VarDecl:
3115e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("VarDecl");
3116e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ParmDecl:
3117e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ParmDecl");
3118e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInterfaceDecl:
3119e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInterfaceDecl");
3120e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryDecl:
3121e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryDecl");
3122e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolDecl:
3123e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolDecl");
3124e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCPropertyDecl:
3125e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCPropertyDecl");
3126e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCIvarDecl:
3127e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCIvarDecl");
3128e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInstanceMethodDecl:
3129e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInstanceMethodDecl");
3130e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassMethodDecl:
3131e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassMethodDecl");
3132e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCImplementationDecl:
3133e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCImplementationDecl");
3134e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryImplDecl:
3135e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryImplDecl");
31368bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek  case CXCursor_CXXMethod:
31378bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek      return createCXString("CXXMethod");
3138e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedDecl:
3139e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedDecl");
3140e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCSuperClassRef:
3141e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCSuperClassRef");
3142e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolRef:
3143e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolRef");
3144e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassRef:
3145e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassRef");
3146e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypeRef:
3147e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypeRef");
31480b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case CXCursor_TemplateRef:
31490b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString("TemplateRef");
31506931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceRef:
31516931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceRef");
3152a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  case CXCursor_MemberRef:
3153a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return createCXString("MemberRef");
315436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelRef:
315536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("LabelRef");
31561f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case CXCursor_OverloadedDeclRef:
31571f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return createCXString("OverloadedDeclRef");
315842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IntegerLiteral:
315942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IntegerLiteral");
316042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_FloatingLiteral:
316142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("FloatingLiteral");
316242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ImaginaryLiteral:
316342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ImaginaryLiteral");
316442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_StringLiteral:
316542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("StringLiteral");
316642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CharacterLiteral:
316742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CharacterLiteral");
316842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ParenExpr:
316942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ParenExpr");
317042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnaryOperator:
317142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnaryOperator");
317242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ArraySubscriptExpr:
317342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ArraySubscriptExpr");
317442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_BinaryOperator:
317542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("BinaryOperator");
317642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundAssignOperator:
317742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundAssignOperator");
317842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ConditionalOperator:
317942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ConditionalOperator");
318042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CStyleCastExpr:
318142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CStyleCastExpr");
318242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundLiteralExpr:
318342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundLiteralExpr");
318442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_InitListExpr:
318542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("InitListExpr");
318642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_AddrLabelExpr:
318742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("AddrLabelExpr");
318842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_StmtExpr:
318942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("StmtExpr");
319042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GenericSelectionExpr:
319142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GenericSelectionExpr");
319242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GNUNullExpr:
319342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GNUNullExpr");
319442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXStaticCastExpr:
319542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXStaticCastExpr");
319642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXDynamicCastExpr:
319742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXDynamicCastExpr");
319842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXReinterpretCastExpr:
319942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXReinterpretCastExpr");
320042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXConstCastExpr:
320142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXConstCastExpr");
320242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXFunctionalCastExpr:
320342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXFunctionalCastExpr");
320442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXTypeidExpr:
320542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXTypeidExpr");
320642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXBoolLiteralExpr:
320742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXBoolLiteralExpr");
320842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXNullPtrLiteralExpr:
320942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXNullPtrLiteralExpr");
321042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXThisExpr:
321142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXThisExpr");
321242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXThrowExpr:
321342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXThrowExpr");
321442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXNewExpr:
321542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXNewExpr");
321642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXDeleteExpr:
321742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXDeleteExpr");
321842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnaryExpr:
321942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnaryExpr");
322042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCStringLiteral:
322142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCStringLiteral");
322242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCEncodeExpr:
322342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCEncodeExpr");
322442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCSelectorExpr:
322542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCSelectorExpr");
322642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCProtocolExpr:
322742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCProtocolExpr");
322842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCBridgedCastExpr:
322942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCBridgedCastExpr");
32301ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek  case CXCursor_BlockExpr:
32311ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek      return createCXString("BlockExpr");
323242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_PackExpansionExpr:
323342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("PackExpansionExpr");
323442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SizeOfPackExpr:
323542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SizeOfPackExpr");
323642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnexposedExpr:
323742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnexposedExpr");
3238e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_DeclRefExpr:
3239e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("DeclRefExpr");
3240e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_MemberRefExpr:
3241e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("MemberRefExpr");
3242e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_CallExpr:
3243e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("CallExpr");
3244e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCMessageExpr:
3245e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCMessageExpr");
3246e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedStmt:
3247e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedStmt");
324842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DeclStmt:
324942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DeclStmt");
325036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelStmt:
325136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return createCXString("LabelStmt");
325242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundStmt:
325342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundStmt");
325442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CaseStmt:
325542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CaseStmt");
325642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DefaultStmt:
325742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DefaultStmt");
325842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IfStmt:
325942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IfStmt");
326042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SwitchStmt:
326142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SwitchStmt");
326242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_WhileStmt:
326342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("WhileStmt");
326442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DoStmt:
326542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DoStmt");
326642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ForStmt:
326742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ForStmt");
326842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GotoStmt:
326942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GotoStmt");
327042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IndirectGotoStmt:
327142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IndirectGotoStmt");
327242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ContinueStmt:
327342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ContinueStmt");
327442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_BreakStmt:
327542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("BreakStmt");
327642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ReturnStmt:
327742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ReturnStmt");
327842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_AsmStmt:
327942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("AsmStmt");
328042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtTryStmt:
328142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtTryStmt");
328242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtCatchStmt:
328342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtCatchStmt");
328442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtFinallyStmt:
328542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtFinallyStmt");
328642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtThrowStmt:
328742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtThrowStmt");
328842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtSynchronizedStmt:
328942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtSynchronizedStmt");
329042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAutoreleasePoolStmt:
329142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAutoreleasePoolStmt");
329242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCForCollectionStmt:
329342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCForCollectionStmt");
329442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXCatchStmt:
329542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXCatchStmt");
329642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXTryStmt:
329742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXTryStmt");
329842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXForRangeStmt:
329942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXForRangeStmt");
330042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHTryStmt:
330142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHTryStmt");
330242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHExceptStmt:
330342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHExceptStmt");
330442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHFinallyStmt:
330542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHFinallyStmt");
330642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_NullStmt:
330742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("NullStmt");
3308e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_InvalidFile:
3309e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("InvalidFile");
3310292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek  case CXCursor_InvalidCode:
3311292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek    return createCXString("InvalidCode");
3312e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NoDeclFound:
3313e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NoDeclFound");
3314e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NotImplemented:
3315e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NotImplemented");
3316e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TranslationUnit:
3317e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TranslationUnit");
3318e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_UnexposedAttr:
3319e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("UnexposedAttr");
3320e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_IBActionAttr:
3321e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("attribute(ibaction)");
33229f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_IBOutletAttr:
33239f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor     return createCXString("attribute(iboutlet)");
3324857e918a8a40deb128840308a318bf623d68295fTed Kremenek  case CXCursor_IBOutletCollectionAttr:
3325857e918a8a40deb128840308a318bf623d68295fTed Kremenek      return createCXString("attribute(iboutletcollection)");
33266639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXFinalAttr:
33276639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(final)");
33286639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXOverrideAttr:
33296639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(override)");
33305f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  case CXCursor_AnnotateAttr:
33315f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    return createCXString("attribute(annotate)");
33329f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_PreprocessingDirective:
33339f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return createCXString("preprocessing directive");
3334572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  case CXCursor_MacroDefinition:
3335572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString("macro definition");
33369b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  case CXCursor_MacroExpansion:
33379b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    return createCXString("macro expansion");
3338ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  case CXCursor_InclusionDirective:
3339ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString("inclusion directive");
33408f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  case CXCursor_Namespace:
33418f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek    return createCXString("Namespace");
3342a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  case CXCursor_LinkageSpec:
3343a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek    return createCXString("LinkageSpec");
33443064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  case CXCursor_CXXBaseSpecifier:
33453064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    return createCXString("C++ base class specifier");
334601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Constructor:
334701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConstructor");
334801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Destructor:
334901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXDestructor");
335001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_ConversionFunction:
335101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConversion");
3352fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTypeParameter:
3353fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTypeParameter");
3354fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_NonTypeTemplateParameter:
3355fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("NonTypeTemplateParameter");
3356fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTemplateParameter:
3357fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTemplateParameter");
3358fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_FunctionTemplate:
3359fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("FunctionTemplate");
336039d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  case CXCursor_ClassTemplate:
336139d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return createCXString("ClassTemplate");
336274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  case CXCursor_ClassTemplatePartialSpecialization:
336374dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return createCXString("ClassTemplatePartialSpecialization");
33646931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceAlias:
33656931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceAlias");
33660a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  case CXCursor_UsingDirective:
33670a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("UsingDirective");
33687e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  case CXCursor_UsingDeclaration:
33697e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor    return createCXString("UsingDeclaration");
3370162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case CXCursor_TypeAliasDecl:
3371352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("TypeAliasDecl");
3372352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCSynthesizeDecl:
3373352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCSynthesizeDecl");
3374352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCDynamicDecl:
3375352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCDynamicDecl");
33762dfdb948bef51a601e763191e4becfe59880d382Argyrios Kyrtzidis  case CXCursor_CXXAccessSpecifier:
33772dfdb948bef51a601e763191e4becfe59880d382Argyrios Kyrtzidis    return createCXString("CXXAccessSpecifier");
337889922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  }
3379e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3380deb06bd3566e18f677e76bc435d478b033fe328bTed Kremenek  llvm_unreachable("Unhandled CXCursorKind");
3381a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return createCXString((const char*) 0);
3382600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
338389922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
3384064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidisstruct GetCursorData {
3385064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  SourceLocation TokenBeginLoc;
33864b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  bool PointsAtMacroArgExpansion;
3387064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor &BestCursor;
3388064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
33894b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  GetCursorData(SourceManager &SM,
33904b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                SourceLocation tokenBegin, CXCursor &outputCursor)
33914b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    : TokenBeginLoc(tokenBegin), BestCursor(outputCursor) {
33924b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    PointsAtMacroArgExpansion = SM.isMacroArgExpansion(tokenBegin);
33934b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  }
3394064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis};
3395064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
33964b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidisstatic enum CXChildVisitResult GetCursorVisitor(CXCursor cursor,
33974b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXCursor parent,
33984b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXClientData client_data) {
3399064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  GetCursorData *Data = static_cast<GetCursorData *>(client_data);
3400064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor *BestCursor = &Data->BestCursor;
34014b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis
34024b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // If we point inside a macro argument we should provide info of what the
34034b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // token is so use the actual cursor, don't replace it with a macro expansion
34044b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // cursor.
34054b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  if (cursor.kind == CXCursor_MacroExpansion && Data->PointsAtMacroArgExpansion)
34064b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    return CXChildVisit_Recurse;
340765ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis
340865ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis  if (clang_isDeclaration(cursor.kind)) {
340965ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // Avoid having the implicit methods override the property decls.
341065ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(getCursorDecl(cursor)))
341165ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis      if (MD->isImplicit())
341265ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis        return CXChildVisit_Break;
341365ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis  }
3414064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
3415064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  if (clang_isExpression(cursor.kind) &&
3416064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis      clang_isDeclaration(BestCursor->kind)) {
3417064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    Decl *D = getCursorDecl(*BestCursor);
3418064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
3419064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // Avoid having the cursor of an expression replace the declaration cursor
3420064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // when the expression source range overlaps the declaration range.
3421064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // This can happen for C++ constructor expressions whose range generally
3422064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // include the variable declaration, e.g.:
3423064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    //  MyCXXClass foo; // Make sure pointing at 'foo' returns a VarDecl cursor.
3424064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    if (D->getLocation().isValid() && Data->TokenBeginLoc.isValid() &&
3425064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis        D->getLocation() == Data->TokenBeginLoc)
3426064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis      return CXChildVisit_Break;
3427064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  }
3428064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
342993798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // If our current best cursor is the construction of a temporary object,
343093798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // don't replace that cursor with a type reference, because we want
343193798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // clang_getCursor() to point at the constructor.
343293798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  if (clang_isExpression(BestCursor->kind) &&
343393798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor      isa<CXXTemporaryObjectExpr>(getCursorExpr(*BestCursor)) &&
3434aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      cursor.kind == CXCursor_TypeRef) {
3435aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    // Keep the cursor pointing at CXXTemporaryObjectExpr but also mark it
3436aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    // as having the actual point on the type reference.
3437aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    *BestCursor = getTypeRefedCallExprCursor(*BestCursor);
343893798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CXChildVisit_Recurse;
3439aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
344093798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor
344133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  *BestCursor = cursor;
344233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  return CXChildVisit_Recurse;
344333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
3444e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3445b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXCursor clang_getCursor(CXTranslationUnit TU, CXSourceLocation Loc) {
3446b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!TU)
3447f462989fe8d6f59ab2d7d0fe2b4b96292ce706eaTed Kremenek    return clang_getNullCursor();
3448e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3449a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
3450bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
3451bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor
3452a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek  SourceLocation SLoc = cxloc::translateSourceLocation(Loc);
3453671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  CXCursor Result = cxcursor::getCursor(TU, SLoc);
3454a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek
345540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  bool Logging = getenv("LIBCLANG_LOGGING");
345640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  if (Logging) {
345740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile SearchFile;
345840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned SearchLine, SearchColumn;
345940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile ResultFile;
346040749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned ResultLine, ResultColumn;
34616653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    CXString SearchFileName, ResultFileName, KindSpelling, USR;
34626653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    const char *IsDef = clang_isCursorDefinition(Result)? " (Definition)" : "";
346340749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXSourceLocation ResultLoc = clang_getCursorLocation(Result);
346440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
346520174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(Loc, &SearchFile, &SearchLine, &SearchColumn, 0);
346620174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(ResultLoc, &ResultFile, &ResultLine,
346720174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                               &ResultColumn, 0);
346840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    SearchFileName = clang_getFileName(SearchFile);
346940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    ResultFileName = clang_getFileName(ResultFile);
347040749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    KindSpelling = clang_getCursorKindSpelling(Result.kind);
34716653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    USR = clang_getCursorUSR(Result);
34726653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    fprintf(stderr, "clang_getCursor(%s:%d:%d) = %s(%s:%d:%d):%s%s\n",
347340749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(SearchFileName), SearchLine, SearchColumn,
347440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(KindSpelling),
34756653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(ResultFileName), ResultLine, ResultColumn,
34766653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(USR), IsDef);
347740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(SearchFileName);
347840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(ResultFileName);
347940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(KindSpelling);
34806653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    clang_disposeString(USR);
34810aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor
34820aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    CXCursor Definition = clang_getCursorDefinition(Result);
34830aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    if (!clang_equalCursors(Definition, clang_getNullCursor())) {
34840aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXSourceLocation DefinitionLoc = clang_getCursorLocation(Definition);
34850aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionKindSpelling
34860aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor                                = clang_getCursorKindSpelling(Definition.kind);
34870aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXFile DefinitionFile;
34880aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      unsigned DefinitionLine, DefinitionColumn;
348920174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth      clang_getExpansionLocation(DefinitionLoc, &DefinitionFile,
349020174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                 &DefinitionLine, &DefinitionColumn, 0);
34910aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionFileName = clang_getFileName(DefinitionFile);
34920aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      fprintf(stderr, "  -> %s(%s:%d:%d)\n",
34930aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionKindSpelling),
34940aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionFileName),
34950aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              DefinitionLine, DefinitionColumn);
34960aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionFileName);
34970aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionKindSpelling);
34980aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    }
349940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  }
350040749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
3501e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  return Result;
350277128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
350377128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
3504738855554394a6afcf39cc8345fd22c3756b8dd0Ted KremenekCXCursor clang_getNullCursor(void) {
35055bfb8c128c2ac8eb4032afc180cdc400a0f953caDouglas Gregor  return MakeCXCursorInvalid(CXCursor_InvalidFile);
3506738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
3507738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek
3508738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenekunsigned clang_equalCursors(CXCursor X, CXCursor Y) {
3509283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  return X == Y;
3510738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
35110d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
35129ce5584553054d0cb934940586aca0186e87fa57Douglas Gregorunsigned clang_hashCursor(CXCursor C) {
35139ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  unsigned Index = 0;
35149ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  if (clang_isExpression(C.kind) || clang_isStatement(C.kind))
35159ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor    Index = 1;
35169ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35179ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  return llvm::DenseMapInfo<std::pair<unsigned, void*> >::getHashValue(
35189ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor                                        std::make_pair(C.kind, C.data[Index]));
35199ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor}
35209ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35219ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isInvalid(enum CXCursorKind K) {
352277128ddd3077fc045751a55bb3226802b15d5510Steve Naroff  return K >= CXCursor_FirstInvalid && K <= CXCursor_LastInvalid;
352377128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
352477128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
35259ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isDeclaration(enum CXCursorKind K) {
352689922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  return K >= CXCursor_FirstDecl && K <= CXCursor_LastDecl;
352789922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
35282d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff
35299ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isReference(enum CXCursorKind K) {
3530f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  return K >= CXCursor_FirstRef && K <= CXCursor_LastRef;
3531f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3532f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
353397b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isExpression(enum CXCursorKind K) {
353497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstExpr && K <= CXCursor_LastExpr;
353597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
353697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
353797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isStatement(enum CXCursorKind K) {
353897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstStmt && K <= CXCursor_LastStmt;
353997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
354097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
35418be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregorunsigned clang_isAttribute(enum CXCursorKind K) {
35428be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor    return K >= CXCursor_FirstAttr && K <= CXCursor_LastAttr;
35438be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor}
35448be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor
35457eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregorunsigned clang_isTranslationUnit(enum CXCursorKind K) {
35467eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return K == CXCursor_TranslationUnit;
35477eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
35487eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
35499f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregorunsigned clang_isPreprocessing(enum CXCursorKind K) {
35509f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  return K >= CXCursor_FirstPreprocessing && K <= CXCursor_LastPreprocessing;
35519f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor}
35529f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor
3553ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenekunsigned clang_isUnexposed(enum CXCursorKind K) {
3554ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  switch (K) {
3555ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedDecl:
3556ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedExpr:
3557ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedStmt:
3558ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedAttr:
3559ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return true;
3560ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    default:
3561ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return false;
3562ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  }
3563ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek}
3564ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek
35659ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXCursorKind clang_getCursorKind(CXCursor C) {
35669efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff  return C.kind;
35679efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff}
35689efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff
356998258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas GregorCXSourceLocation clang_getCursorLocation(CXCursor C) {
357098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (clang_isReference(C.kind)) {
3571f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    switch (C.kind) {
3572f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCSuperClassRef: {
3573f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3574f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCSuperClassRef(C);
3575a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3576f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3577f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3578f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
3579f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCProtocolDecl *, SourceLocation> P
3580f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCProtocolRef(C);
3581a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3582f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3583f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3584f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCClassRef: {
3585f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3586f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCClassRef(C);
3587a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3588f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
35897d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3590f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef: {
35917d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      std::pair<TypeDecl *, SourceLocation> P = getCursorTypeRef(C);
3592a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
35937d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
35940b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
35950b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
35960b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      std::pair<TemplateDecl *, SourceLocation> P = getCursorTemplateRef(C);
35970b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
35980b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
35990b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
36006931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
36016931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      std::pair<NamedDecl *, SourceLocation> P = getCursorNamespaceRef(C);
36026931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36036931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
36046931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
3605a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
3606a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      std::pair<FieldDecl *, SourceLocation> P = getCursorMemberRef(C);
3607a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3608a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
3609a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
36103064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
36111b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      CXXBaseSpecifier *BaseSpec = getCursorCXXBaseSpecifier(C);
36121b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (!BaseSpec)
36131b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return clang_getNullLocation();
36141b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
36151b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (TypeSourceInfo *TSInfo = BaseSpec->getTypeSourceInfo())
36161b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return cxloc::translateSourceLocation(getCursorContext(C),
36171b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor                                            TSInfo->getTypeLoc().getBeginLoc());
36181b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
36191b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
36201b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor                                        BaseSpec->getSourceRange().getBegin());
36213064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
3622f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
362336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
362436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      std::pair<LabelStmt *, SourceLocation> P = getCursorLabelRef(C);
362536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C), P.second);
362636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
362736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
36281f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
36291f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
36301f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                                          getCursorOverloadedDeclRef(C).second);
36311f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3632f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    default:
3633f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3634f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      llvm_unreachable("Missed a reference kind");
3635f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
363698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  }
363797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
363897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3639f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    return cxloc::translateSourceLocation(getCursorContext(C),
364097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor                                   getLocationFromExpr(getCursorExpr(C)));
364197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
364236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind))
364336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C),
364436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor                                          getCursorStmt(C)->getLocStart());
364536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
36469f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  if (C.kind == CXCursor_PreprocessingDirective) {
36479f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    SourceLocation L = cxcursor::getCursorPreprocessingDirective(C).getBegin();
36489f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
36499f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  }
36504807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
36519b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
36524ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor    SourceLocation L
36539e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth      = cxcursor::getCursorMacroExpansion(C)->getSourceRange().getBegin();
36544807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
36554807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor  }
3656572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3657572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition) {
3658572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    SourceLocation L = cxcursor::getCursorMacroDefinition(C)->getLocation();
3659572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3660572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  }
3661ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3662ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective) {
3663ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    SourceLocation L
3664ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      = cxcursor::getCursorInclusionDirective(C)->getSourceRange().getBegin();
3665ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3666ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  }
3667ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
36689a700d277c38d9afaa7cb3fe93a714bfe9b62eecTed Kremenek  if (C.kind < CXCursor_FirstDecl || C.kind > CXCursor_LastDecl)
36695352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullLocation();
367098258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor
3671f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  Decl *D = getCursorDecl(C);
3672f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  SourceLocation Loc = D->getLocation();
3673007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // FIXME: Multiple variables declared in a single declaration
3674007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // currently lack the information needed to correctly determine their
3675007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // ranges when accounting for the type-specifier.  We use context
3676007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3677007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // and if so, whether it is the first decl.
3678007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3679007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (!cxcursor::isFirstInDeclGroup(C))
3680007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      Loc = VD->getLocation();
3681007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
3682007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek
36832ca54feee89d7277fb967e3247a64f40ef155a82Douglas Gregor  return cxloc::translateSourceLocation(getCursorContext(C), Loc);
368488145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
3685a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor
3686a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor} // end extern "C"
3687a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3688671436e9e2794c56f3c2e62739d225571493af37Argyrios KyrtzidisCXCursor cxcursor::getCursor(CXTranslationUnit TU, SourceLocation SLoc) {
3689671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  assert(TU);
3690671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3691671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // Guard against an invalid SourceLocation, or we may assert in one
3692671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // of the following calls.
3693671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  if (SLoc.isInvalid())
3694671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    return clang_getNullCursor();
3695671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3696671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
3697671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3698671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // Translate the given source location to make it point at the beginning of
3699671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // the token under the cursor.
3700671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  SLoc = Lexer::GetBeginningOfToken(SLoc, CXXUnit->getSourceManager(),
3701671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis                                    CXXUnit->getASTContext().getLangOptions());
3702671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3703671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  CXCursor Result = MakeCXCursorInvalid(CXCursor_NoDeclFound);
3704671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  if (SLoc.isValid()) {
3705671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    GetCursorData ResultData(CXXUnit->getSourceManager(), SLoc, Result);
3706671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    CursorVisitor CursorVis(TU, GetCursorVisitor, &ResultData,
3707671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis                            /*VisitPreprocessorLast=*/true,
3708e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                            /*VisitIncludedEntities=*/false,
3709671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis                            SourceLocation(SLoc));
3710dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    CursorVis.visitFileRegion();
3711671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  }
3712671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3713671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  return Result;
3714671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis}
3715671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3716a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorstatic SourceRange getRawCursorExtent(CXCursor C) {
3717a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  if (clang_isReference(C.kind)) {
3718a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    switch (C.kind) {
3719a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCSuperClassRef:
3720a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return  getCursorObjCSuperClassRef(C).second;
3721f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3722a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCProtocolRef:
3723a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCProtocolRef(C).second;
3724f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3725a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCClassRef:
3726a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCClassRef(C).second;
37277d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3728a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_TypeRef:
3729a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorTypeRef(C).second;
37300b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
37310b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
37320b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return getCursorTemplateRef(C).second;
37330b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
37346931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
37356931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return getCursorNamespaceRef(C).second;
3736a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3737a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
3738a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return getCursorMemberRef(C).second;
3739a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
37403064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier:
37411b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return getCursorCXXBaseSpecifier(C)->getSourceRange();
3742f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
374336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
374436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return getCursorLabelRef(C).second;
374536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
37461f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
37471f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return getCursorOverloadedDeclRef(C).second;
37481f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3749a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    default:
3750a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3751a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      llvm_unreachable("Missed a reference kind");
3752a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    }
3753a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  }
375497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
375597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3756a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorExpr(C)->getSourceRange();
375733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
375833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (clang_isStatement(C.kind))
3759a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorStmt(C)->getSourceRange();
3760f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
37616639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  if (clang_isAttribute(C.kind))
37626639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis    return getCursorAttr(C)->getRange();
37636639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
3764a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (C.kind == CXCursor_PreprocessingDirective)
3765a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return cxcursor::getCursorPreprocessingDirective(C);
37664807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
3767ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_MacroExpansion) {
3768ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3769ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorMacroExpansion(C)->getSourceRange();
3770ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3771ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3772572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3773ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_MacroDefinition) {
3774ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3775ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorMacroDefinition(C)->getSourceRange();
3776ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3777ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3778ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3779ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_InclusionDirective) {
3780ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3781ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorInclusionDirective(C)->getSourceRange();
3782ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3783ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3784ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3785007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
3786007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    Decl *D = cxcursor::getCursorDecl(C);
3787007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    SourceRange R = D->getSourceRange();
3788007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // FIXME: Multiple variables declared in a single declaration
3789007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // currently lack the information needed to correctly determine their
3790007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // ranges when accounting for the type-specifier.  We use context
3791007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3792007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // and if so, whether it is the first decl.
3793007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3794007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      if (!cxcursor::isFirstInDeclGroup(C))
3795007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek        R.setBegin(VD->getLocation());
3796007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    }
3797007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    return R;
3798007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
37996653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return SourceRange();
38006653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
38016653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38026653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// \brief Retrieves the "raw" cursor extent, which is then extended to include
38036653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// the decl-specifier-seq for declarations.
38046653798ff5ce6deb58112777e21307ccc453133dDouglas Gregorstatic SourceRange getFullCursorExtent(CXCursor C, SourceManager &SrcMgr) {
38056653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
38066653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    Decl *D = cxcursor::getCursorDecl(C);
38076653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange R = D->getSourceRange();
38082494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
38092494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // Adjust the start of the location for declarations preceded by
38102494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // declaration specifiers.
38112494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
38126653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    if (const DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) {
38132494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
38142494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
38152494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    } else if (TypedefDecl *Typedef = dyn_cast<TypedefDecl>(D)) {
38162494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
38172494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
38182494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    }
38196653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38202494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && R.getBegin().isValid() &&
38212494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, R.getBegin()))
38222494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      R.setBegin(StartLoc);
38232494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
38242494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // FIXME: Multiple variables declared in a single declaration
38252494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // currently lack the information needed to correctly determine their
38262494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // ranges when accounting for the type-specifier.  We use context
38272494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
38282494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // and if so, whether it is the first decl.
38292494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
38302494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (!cxcursor::isFirstInDeclGroup(C))
38312494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        R.setBegin(VD->getLocation());
38326653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    }
38336653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38346653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    return R;
38356653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  }
38366653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38376653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return getRawCursorExtent(C);
38386653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
3839a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3840a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorextern "C" {
3841a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3842a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas GregorCXSourceRange clang_getCursorExtent(CXCursor C) {
3843a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  SourceRange R = getRawCursorExtent(C);
3844a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R.isInvalid())
38455352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
3846f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3847a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  return cxloc::translateSourceRange(getCursorContext(C), R);
3848a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor}
3849c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
3850c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas GregorCXCursor clang_getCursorReferenced(CXCursor C) {
3851b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
3852b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
3853f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3854a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit tu = getCursorTU(C);
38551f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (clang_isDeclaration(C.kind)) {
38561f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getCursorDecl(C);
38571f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
3858a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Using, D->getLocation(), tu);
38591f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (ObjCClassDecl *Classes = dyn_cast<ObjCClassDecl>(D))
3860a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Classes, D->getLocation(), tu);
38611f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (ObjCForwardProtocolDecl *Protocols
38621f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                                        = dyn_cast<ObjCForwardProtocolDecl>(D))
3863a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Protocols, D->getLocation(), tu);
38645f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
3865e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
3866e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return MakeCXCursor(Property, tu);
3867e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
3868c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return C;
38691f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
38701f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
387197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
38721f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Expr *E = getCursorExpr(C);
38731f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getDeclFromExpr(E);
3874aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (D) {
3875aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXCursor declCursor = MakeCXCursor(D, tu);
3876aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      declCursor = getSelectorIdentifierCursor(getSelectorIdentifierIndex(C),
3877aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                               declCursor);
3878aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return declCursor;
3879aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
38801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
38811f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (OverloadExpr *Ovl = dyn_cast_or_null<OverloadExpr>(E))
3882a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Ovl, tu);
38831f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
388497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    return clang_getNullCursor();
388597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
388697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
388736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
388836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
388936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (GotoStmt *Goto = dyn_cast_or_null<GotoStmt>(S))
389037c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek      if (LabelDecl *label = Goto->getLabel())
389137c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        if (LabelStmt *labelS = label->getStmt())
389237c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        return MakeCXCursor(labelS, getCursorDecl(C), tu);
389336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
389436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return clang_getNullCursor();
389536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
389636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
38979b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
38989e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    if (MacroDefinition *Def = getCursorMacroExpansion(C)->getDefinition())
3899a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeMacroDefinitionCursor(Def, tu);
3900bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor  }
3901bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
3902c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  if (!clang_isReference(C.kind))
3903c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return clang_getNullCursor();
3904f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3905c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  switch (C.kind) {
3906c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    case CXCursor_ObjCSuperClassRef:
3907a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCSuperClassRef(C).first, tu);
3908f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3909f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
3910a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCProtocolRef(C).first, tu);
3911f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3912f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCClassRef:
3913a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCClassRef(C).first, tu );
39147d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3915f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef:
3916a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTypeRef(C).first, tu );
39170b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
39180b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
3919a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTemplateRef(C).first, tu );
39200b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
39216931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
3922a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorNamespaceRef(C).first, tu );
39236931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
3924a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
3925a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorMemberRef(C).first, tu );
3926a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
39273064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
39283064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = cxcursor::getCursorCXXBaseSpecifier(C);
39293064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return clang_getTypeDeclaration(cxtype::MakeCXType(B->getType(),
3930a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                                         tu ));
39313064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
3932f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
393336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
393436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // FIXME: We end up faking the "parent" declaration here because we
393536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // don't want to make CXCursor larger.
393636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return MakeCXCursor(getCursorLabelRef(C).first,
3937a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek               static_cast<ASTUnit*>(tu->TUData)->getASTContext()
3938a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          .getTranslationUnitDecl(),
3939a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          tu);
394036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
39411f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
39421f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return C;
39431f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3944c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    default:
3945c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      // We would prefer to enumerate all non-reference cursor kinds here.
3946c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      llvm_unreachable("Unhandled reference cursor kind");
3947c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      break;
3948c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
3949c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  }
3950f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3951c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  return clang_getNullCursor();
3952c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor}
3953c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
3954b699866820102a69d83d6ac6941985c5ef4e8c40Douglas GregorCXCursor clang_getCursorDefinition(CXCursor C) {
3955b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
3956b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
3957f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3958a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(C);
3959f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3960b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  bool WasReference = false;
396197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isReference(C.kind) || clang_isExpression(C.kind)) {
3962b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    C = clang_getCursorReferenced(C);
3963b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    WasReference = true;
3964b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
3965b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
39669b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
3967bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor    return clang_getCursorReferenced(C);
3968bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
3969b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
3970b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
3971b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
3972b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  Decl *D = getCursorDecl(C);
3973b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!D)
3974b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
3975f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3976b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  switch (D->getKind()) {
3977b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't really separate the notions of
3978b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // declaration and definition.
3979b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Namespace:
3980b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Typedef:
3981162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case Decl::TypeAlias:
39823e4c6c4c79a03f5cb0c4671d7c282d623c6dc35eRichard Smith  case Decl::TypeAliasTemplate:
3983b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTypeParm:
3984b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::EnumConstant:
3985b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Field:
3986d98114647e16796a976b04af79975b4f0eacf22bBenjamin Kramer  case Decl::IndirectField:
3987b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCIvar:
3988b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCAtDefsField:
3989b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ImplicitParam:
3990b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ParmVar:
3991b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NonTypeTemplateParm:
3992b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTemplateParm:
3993b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategoryImpl:
3994b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCImplementation:
39956206d53f67613958ae1b023aba337ebb46f11a8bAbramo Bagnara  case Decl::AccessSpec:
3996b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::LinkageSpec:
3997b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCPropertyImpl:
3998b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FileScopeAsm:
3999b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::StaticAssert:
4000b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Block:
4001ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  case Decl::Label:  // FIXME: Is this right??
4002af0f4d0b2e38c810effc8b024ad2fb6604eec5d3Francois Pichet  case Decl::ClassScopeFunctionSpecialization:
4003b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return C;
4004b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4005b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't make any sense here, but are
4006b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // nonetheless harmless.
4007b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TranslationUnit:
4008b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4009b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4010b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds for which the definition is not resolvable.
4011b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingTypename:
4012b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingValue:
4013b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4014b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4015b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingDirective:
4016b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return MakeCXCursor(cast<UsingDirectiveDecl>(D)->getNominatedNamespace(),
4017a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                        TU);
4018b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4019b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NamespaceAlias:
4020a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<NamespaceAliasDecl>(D)->getNamespace(), TU);
4021b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4022b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Enum:
4023b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Record:
4024b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXRecord:
4025b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplateSpecialization:
4026b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplatePartialSpecialization:
4027952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor    if (TagDecl *Def = cast<TagDecl>(D)->getDefinition())
4028a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
4029b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4030b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4031b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Function:
4032b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXMethod:
4033b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConstructor:
4034b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXDestructor:
4035b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConversion: {
4036b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4037b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionDecl>(D)->getBody(Def))
4038a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(const_cast<FunctionDecl *>(Def), TU);
4039b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4040b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4041b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4042b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Var: {
404331310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    // Ask the variable if it has a definition.
404431310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    if (VarDecl *Def = cast<VarDecl>(D)->getDefinition())
4045a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
404631310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    return clang_getNullCursor();
4047b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4048f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4049b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FunctionTemplate: {
4050b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4051b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionTemplateDecl>(D)->getTemplatedDecl()->getBody(Def))
4052a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def->getDescribedFunctionTemplate(), TU);
4053b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4054b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4055f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4056b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplate: {
4057b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (RecordDecl *Def = cast<ClassTemplateDecl>(D)->getTemplatedDecl()
4058952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor                                                            ->getDefinition())
40590b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return MakeCXCursor(cast<CXXRecordDecl>(Def)->getDescribedClassTemplate(),
4060a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          TU);
4061b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4062b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4063b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
40641f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::Using:
40651f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return MakeCursorOverloadedDeclRef(cast<UsingDecl>(D),
4066a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       D->getLocation(), TU);
4067b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4068b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingShadow:
4069b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getCursorDefinition(
4070f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                       MakeCXCursor(cast<UsingShadowDecl>(D)->getTargetDecl(),
4071a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                    TU));
4072b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4073b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCMethod: {
4074b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    ObjCMethodDecl *Method = cast<ObjCMethodDecl>(D);
4075b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (Method->isThisDeclarationADefinition())
4076b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      return C;
4077b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4078b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // Dig out the method definition in the associated
4079b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // @implementation, if we have it.
4080b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: The ASTs should make finding the definition easier.
4081b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4082b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                       = dyn_cast<ObjCInterfaceDecl>(Method->getDeclContext()))
4083b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (ObjCImplementationDecl *ClassImpl = Class->getImplementation())
4084b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor        if (ObjCMethodDecl *Def = ClassImpl->getMethod(Method->getSelector(),
4085b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                                                  Method->isInstanceMethod()))
4086b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          if (Def->isThisDeclarationADefinition())
4087a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            return MakeCXCursor(Def, TU);
4088b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4089b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4090b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4091b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4092b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategory:
4093b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCCategoryImplDecl *Impl
4094b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                               = cast<ObjCCategoryDecl>(D)->getImplementation())
4095a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4096b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4097b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4098b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProtocol:
4099b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (!cast<ObjCProtocolDecl>(D)->isForwardDecl())
4100b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      return C;
4101b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4102b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4103b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCInterface:
4104b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // There are two notions of a "definition" for an Objective-C
4105b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // class: the interface and its implementation. When we resolved a
4106b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // reference to an Objective-C class, produce the @interface as
4107b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // the definition; when we were provided with the interface,
4108b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // produce the @implementation as the definition.
4109b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (WasReference) {
4110b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (!cast<ObjCInterfaceDecl>(D)->isForwardDecl())
4111b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor        return C;
4112b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    } else if (ObjCImplementationDecl *Impl
4113b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                              = cast<ObjCInterfaceDecl>(D)->getImplementation())
4114a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4115b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4116f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4117b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProperty:
4118b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: We don't really know where to find the
4119b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // ObjCPropertyImplDecls that implement this property.
4120b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4121b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4122b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCompatibleAlias:
4123b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4124b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          = cast<ObjCCompatibleAliasDecl>(D)->getClassInterface())
4125b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (!Class->isForwardDecl())
4126a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek        return MakeCXCursor(Class, TU);
4127f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4128b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4129b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41301f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::ObjCForwardProtocol:
41311f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return MakeCursorOverloadedDeclRef(cast<ObjCForwardProtocolDecl>(D),
4132a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       D->getLocation(), TU);
4133b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41341f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::ObjCClass:
41359e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar    return MakeCursorOverloadedDeclRef(cast<ObjCClassDecl>(D), D->getLocation(),
4136a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       TU);
4137b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4138b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Friend:
4139b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendDecl>(D)->getFriendDecl())
4140a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4141b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4142b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4143b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FriendTemplate:
4144b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendTemplateDecl>(D)->getFriendDecl())
4145a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4146b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4147b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4148b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4149b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getNullCursor();
4150b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4151b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4152b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregorunsigned clang_isCursorDefinition(CXCursor C) {
4153b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
4154b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return 0;
4155b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4156b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getCursorDefinition(C) == C;
4157b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4158b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41591a9d0503b67a499797141af0fd6d315d5045f0eaDouglas GregorCXCursor clang_getCanonicalCursor(CXCursor C) {
41601a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  if (!clang_isDeclaration(C.kind))
41611a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return C;
41621a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
4163e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  if (Decl *D = getCursorDecl(C)) {
4164debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis    if (ObjCCategoryImplDecl *CatImplD = dyn_cast<ObjCCategoryImplDecl>(D))
4165debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis      if (ObjCCategoryDecl *CatD = CatImplD->getCategoryDecl())
4166debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis        return MakeCXCursor(CatD, getCursorTU(C));
4167debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis
4168e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis    if (ObjCImplDecl *ImplD = dyn_cast<ObjCImplDecl>(D))
4169e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis      if (ObjCInterfaceDecl *IFD = ImplD->getClassInterface())
4170e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis        return MakeCXCursor(IFD, getCursorTU(C));
4171e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis
41721a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return MakeCXCursor(D->getCanonicalDecl(), getCursorTU(C));
4173e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  }
41741a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
41751a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  return C;
41761a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor}
41771a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
41781f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregorunsigned clang_getNumOverloadedDecls(CXCursor C) {
41797c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (C.kind != CXCursor_OverloadedDeclRef)
41801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return 0;
41811f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
41821f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
41831f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
41841f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return E->getNumDecls();
41851f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
41861f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
41871f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
41881f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return S->size();
41891f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
41901f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
41911f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
4192826faa22bae112e01293a58534a40711043cce65Argyrios Kyrtzidis    return Using->shadow_size();
419395ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian  if (isa<ObjCClassDecl>(D))
419495ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian    return 1;
41951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCForwardProtocolDecl *Protocols =dyn_cast<ObjCForwardProtocolDecl>(D))
41961f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return Protocols->protocol_size();
41971f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
41981f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return 0;
41991f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
42001f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42011f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas GregorCXCursor clang_getOverloadedDecl(CXCursor cursor, unsigned index) {
42027c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (cursor.kind != CXCursor_OverloadedDeclRef)
42031f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42041f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42051f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (index >= clang_getNumOverloadedDecls(cursor))
42061f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
4208a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(cursor);
42091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(cursor).first;
42101f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
4211a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(E->decls_begin()[index], TU);
42121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
42141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
4215a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(S->begin()[index], TU);
42161f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42171f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
42181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D)) {
42191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // FIXME: This is, unfortunately, linear time.
42201f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    UsingDecl::shadow_iterator Pos = Using->shadow_begin();
42211f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    std::advance(Pos, index);
4222a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<UsingShadowDecl>(*Pos)->getTargetDecl(), TU);
42231f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
42241f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCClassDecl *Classes = dyn_cast<ObjCClassDecl>(D))
422595ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian    return MakeCXCursor(Classes->getForwardInterfaceDecl(), TU);
42261f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCForwardProtocolDecl *Protocols = dyn_cast<ObjCForwardProtocolDecl>(D))
4227a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(Protocols->protocol_begin()[index], TU);
42281f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42291f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return clang_getNullCursor();
42301f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
42311f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42320d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbarvoid clang_getDefinitionSpellingAndExtent(CXCursor C,
42334ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **startBuf,
42344ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **endBuf,
42354ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startLine,
42364ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startColumn,
42374ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *endLine,
42389ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          unsigned *endColumn) {
4239283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  assert(getCursorDecl(C) && "CXCursor has null decl");
4240283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  NamedDecl *ND = static_cast<NamedDecl *>(getCursorDecl(C));
42414ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  FunctionDecl *FD = dyn_cast<FunctionDecl>(ND);
42424ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  CompoundStmt *Body = dyn_cast<CompoundStmt>(FD->getBody());
4243f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
42444ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  SourceManager &SM = FD->getASTContext().getSourceManager();
42454ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startBuf = SM.getCharacterData(Body->getLBracLoc());
42464ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endBuf = SM.getCharacterData(Body->getRBracLoc());
42474ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startLine = SM.getSpellingLineNumber(Body->getLBracLoc());
42484ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startColumn = SM.getSpellingColumnNumber(Body->getLBracLoc());
42494ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endLine = SM.getSpellingLineNumber(Body->getRBracLoc());
42504ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endColumn = SM.getSpellingColumnNumber(Body->getRBracLoc());
42514ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff}
4252f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4253430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4254430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas GregorCXSourceRange clang_getCursorReferenceNameRange(CXCursor C, unsigned NameFlags,
4255430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                unsigned PieceIndex) {
4256430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  RefNamePieces Pieces;
4257430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4258430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  switch (C.kind) {
4259430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_MemberRefExpr:
4260430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (MemberExpr *E = dyn_cast<MemberExpr>(getCursorExpr(C)))
4261430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, true, E->getMemberNameInfo(),
4262430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange());
4263430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4264430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4265430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_DeclRefExpr:
4266430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (DeclRefExpr *E = dyn_cast<DeclRefExpr>(getCursorExpr(C)))
4267430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, false, E->getNameInfo(),
4268430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange(),
4269430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getExplicitTemplateArgsOpt());
4270430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4271430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4272430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_CallExpr:
4273430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (CXXOperatorCallExpr *OCE =
4274430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        dyn_cast<CXXOperatorCallExpr>(getCursorExpr(C))) {
4275430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Expr *Callee = OCE->getCallee();
4276430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(Callee))
4277430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Callee = ICE->getSubExpr();
4278430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4279430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Callee))
4280430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Pieces = buildPieces(NameFlags, false, DRE->getNameInfo(),
4281430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                             DRE->getQualifierLoc().getSourceRange());
4282430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
4283430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4284430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4285430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  default:
4286430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4287430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4288430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4289430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (Pieces.empty()) {
4290430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (PieceIndex == 0)
4291430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      return clang_getCursorExtent(C);
4292430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  } else if (PieceIndex < Pieces.size()) {
4293430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      SourceRange R = Pieces[PieceIndex];
4294430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (R.isValid())
4295430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        return cxloc::translateSourceRange(getCursorContext(C), R);
4296430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4297430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4298430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  return clang_getNullRange();
4299430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
4300430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
43010a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregorvoid clang_enableStackTraces(void) {
43020a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  llvm::sys::PrintStackTraceOnErrorSignal();
43030a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor}
43040a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor
4305995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbarvoid clang_executeOnThread(void (*fn)(void*), void *user_data,
4306995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar                           unsigned stack_size) {
4307995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar  llvm::llvm_execute_on_thread(fn, user_data, stack_size);
4308995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar}
4309995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar
4310fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
4311fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
4312fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
4313fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor// Token-based Operations.
4314fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
4315fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4316fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor/* CXToken layout:
4317fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[0]: a CXTokenKind
4318fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[1]: starting token location
4319fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[2]: token length
4320fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[3]: reserved
4321f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek *   ptr_data: for identifiers and keywords, an IdentifierInfo*.
4322fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   otherwise unused.
4323fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor */
4324fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorextern "C" {
4325fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4326fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXTokenKind clang_getTokenKind(CXToken CXTok) {
4327fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return static_cast<CXTokenKind>(CXTok.int_data[0]);
4328fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4329fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4330fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXString clang_getTokenSpelling(CXTranslationUnit TU, CXToken CXTok) {
4331fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  switch (clang_getTokenKind(CXTok)) {
4332fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Identifier:
4333fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Keyword:
4334fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We know we have an IdentifierInfo*, so use that.
4335ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(static_cast<IdentifierInfo *>(CXTok.ptr_data)
4336ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                            ->getNameStart());
4337fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4338fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Literal: {
4339fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We have stashed the starting pointer in the ptr_data field. Use it.
4340fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *Text = static_cast<const char *>(CXTok.ptr_data);
43415f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    return createCXString(StringRef(Text, CXTok.int_data[2]));
4342fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4343f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4344fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Punctuation:
4345fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Comment:
4346fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    break;
4347fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4348f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4349f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // We have to find the starting buffer pointer the hard way, by
4350fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // deconstructing the source location.
4351a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4352fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4353ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
4354f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4355fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceLocation Loc = SourceLocation::getFromRawEncoding(CXTok.int_data[1]);
4356fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> LocInfo
4357a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    = CXXUnit->getSourceManager().getDecomposedSpellingLoc(Loc);
4358f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
43595f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4360f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = CXXUnit->getSourceManager().getBufferData(LocInfo.first, &Invalid);
4361f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  if (Invalid)
4362aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor    return createCXString("");
4363fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4364f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  return createCXString(Buffer.substr(LocInfo.second, CXTok.int_data[2]));
4365fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4366f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4367fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceLocation clang_getTokenLocation(CXTranslationUnit TU, CXToken CXTok) {
4368a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4369fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4370fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return clang_getNullLocation();
4371f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4372fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return cxloc::translateSourceLocation(CXXUnit->getASTContext(),
4373fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4374fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4375fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4376fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceRange clang_getTokenExtent(CXTranslationUnit TU, CXToken CXTok) {
4377a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
43785352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  if (!CXXUnit)
43795352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
4380f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4381f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return cxloc::translateSourceRange(CXXUnit->getASTContext(),
4382fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4383fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4384f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4385ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisstatic void getTokens(ASTUnit *CXXUnit, SourceRange Range,
4386ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                      SmallVectorImpl<CXToken> &CXTokens) {
4387fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceManager &SourceMgr = CXXUnit->getSourceManager();
4388fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> BeginLocInfo
4389ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(Range.getBegin());
4390fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> EndLocInfo
4391ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(Range.getEnd());
4392f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4393fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Cannot tokenize across files.
4394fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (BeginLocInfo.first != EndLocInfo.first)
4395fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4396f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4397f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Create a lexer
4398f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
43995f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4400f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid);
440147a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor  if (Invalid)
440247a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor    return;
4403aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor
4404fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
4405fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            CXXUnit->getASTContext().getLangOptions(),
4406f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer            Buffer.begin(), Buffer.data() + BeginLocInfo.second, Buffer.end());
4407fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lex.SetCommentRetentionState(true);
4408f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4409fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Lex tokens until we hit the end of the range.
4410f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  const char *EffectiveBufferEnd = Buffer.data() + EndLocInfo.second;
4411fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Token Tok;
4412096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall  bool previousWasAt = false;
4413fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  do {
4414fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Lex the next token
4415fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    Lex.LexFromRawLexer(Tok);
4416fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.is(tok::eof))
4417fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
4418f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4419fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Initialize the CXToken.
4420fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXToken CXTok;
4421f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4422fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Common fields
4423fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[1] = Tok.getLocation().getRawEncoding();
4424fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[2] = Tok.getLength();
4425fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[3] = 0;
4426f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4427fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Kind-specific fields
4428fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.isLiteral()) {
4429fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Literal;
4430fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = (void *)Tok.getLiteralData();
4431c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara    } else if (Tok.is(tok::raw_identifier)) {
4432aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor      // Lookup the identifier to determine whether we have a keyword.
4433fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      IdentifierInfo *II
4434c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        = CXXUnit->getPreprocessor().LookUpIdentifierInfo(Tok);
4435aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek
4436096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall      if ((II->getObjCKeywordID() != tok::objc_not_keyword) && previousWasAt) {
4437aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek        CXTok.int_data[0] = CXToken_Keyword;
4438aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4439aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      else {
4440c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        CXTok.int_data[0] = Tok.is(tok::identifier)
4441c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          ? CXToken_Identifier
4442c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          : CXToken_Keyword;
4443aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4444fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = II;
4445fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else if (Tok.is(tok::comment)) {
4446fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Comment;
4447fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4448fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else {
4449fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Punctuation;
4450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4451fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
4452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTokens.push_back(CXTok);
4453096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall    previousWasAt = Tok.is(tok::at);
4454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  } while (Lex.getBufferLocation() <= EffectiveBufferEnd);
4455ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis}
4456ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4457ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisvoid clang_tokenize(CXTranslationUnit TU, CXSourceRange Range,
4458ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                    CXToken **Tokens, unsigned *NumTokens) {
4459ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (Tokens)
4460ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    *Tokens = 0;
4461ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (NumTokens)
4462ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    *NumTokens = 0;
4463ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4464ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4465ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (!CXXUnit || !Tokens || !NumTokens)
4466ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4467ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4468ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
4469ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4470ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SourceRange R = cxloc::translateCXSourceRange(Range);
4471ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (R.isInvalid())
4472ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4473ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4474ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SmallVector<CXToken, 32> CXTokens;
4475ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  getTokens(CXXUnit, R, CXTokens);
4476f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4477fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (CXTokens.empty())
4478fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4479f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4480fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *Tokens = (CXToken *)malloc(sizeof(CXToken) * CXTokens.size());
4481fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memmove(*Tokens, CXTokens.data(), sizeof(CXToken) * CXTokens.size());
4482fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *NumTokens = CXTokens.size();
4483fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
44840045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
44856db610934bedc6896393c1e1099525b35380acd6Ted Kremenekvoid clang_disposeTokens(CXTranslationUnit TU,
44866db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                         CXToken *Tokens, unsigned NumTokens) {
44876db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  free(Tokens);
44886db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
44896db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
44906db610934bedc6896393c1e1099525b35380acd6Ted Kremenek} // end: extern "C"
44916db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
44926db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
44936db610934bedc6896393c1e1099525b35380acd6Ted Kremenek// Token annotation APIs.
44946db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
44956db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
44960045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregortypedef llvm::DenseMap<unsigned, CXCursor> AnnotateTokensData;
4497fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
4498fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXCursor parent,
4499fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXClientData client_data);
45006db610934bedc6896393c1e1099525b35380acd6Ted Kremeneknamespace {
45016db610934bedc6896393c1e1099525b35380acd6Ted Kremenekclass AnnotateTokensWorker {
45026db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  AnnotateTokensData &Annotated;
450311949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXToken *Tokens;
450411949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXCursor *Cursors;
450511949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  unsigned NumTokens;
4506fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned TokIdx;
45074419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  unsigned PreprocessingTokIdx;
4508fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CursorVisitor AnnotateVis;
4509fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceManager &SrcMgr;
4510f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool HasContextSensitiveKeywords;
4511f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4512fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  bool MoreTokens() const { return TokIdx < NumTokens; }
4513fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned NextToken() const { return TokIdx; }
4514fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void AdvanceToken() { ++TokIdx; }
4515fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation GetTokenLoc(unsigned tokI) {
4516fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4517fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
45185f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  bool isFunctionMacroToken(unsigned tokI) const {
4519a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return Tokens[tokI].int_data[3] != 0;
4520a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
45215f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  SourceLocation getFunctionMacroTokenLoc(unsigned tokI) const {
4522a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[3]);
4523a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4524a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4525a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  void annotateAndAdvanceTokens(CXCursor, RangeComparisonResult, SourceRange);
45265f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void annotateAndAdvanceFunctionMacroTokens(CXCursor, RangeComparisonResult,
45275f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                             SourceRange);
4528fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
45296db610934bedc6896393c1e1099525b35380acd6Ted Kremenekpublic:
453011949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  AnnotateTokensWorker(AnnotateTokensData &annotated,
4531fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                       CXToken *tokens, CXCursor *cursors, unsigned numTokens,
4532a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                       CXTranslationUnit tu, SourceRange RegionOfInterest)
453311949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    : Annotated(annotated), Tokens(tokens), Cursors(cursors),
45344419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      NumTokens(numTokens), TokIdx(0), PreprocessingTokIdx(0),
4535a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      AnnotateVis(tu,
4536f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  AnnotateTokensVisitor, this,
4537f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  /*VisitPreprocessorLast=*/true,
4538e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                  /*VisitIncludedEntities=*/false,
4539f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  RegionOfInterest),
4540f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      SrcMgr(static_cast<ASTUnit*>(tu->TUData)->getSourceManager()),
4541f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      HasContextSensitiveKeywords(false) { }
454211949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4543fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void VisitChildren(CXCursor C) { AnnotateVis.VisitChildren(C); }
45446db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  enum CXChildVisitResult Visit(CXCursor cursor, CXCursor parent);
454503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  void AnnotateTokens();
4546f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4547f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// \brief Determine whether the annotator saw any cursors that have
4548f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// context-sensitive keywords.
4549f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool hasContextSensitiveKeywords() const {
4550f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    return HasContextSensitiveKeywords;
4551f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
45526db610934bedc6896393c1e1099525b35380acd6Ted Kremenek};
45536db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
45540045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
455503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidisvoid AnnotateTokensWorker::AnnotateTokens() {
4556fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Walk the AST within the region of interest, annotating tokens
4557fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // along the way.
455803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  AnnotateVis.visitFileRegion();
4559fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4560fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = 0 ; I < TokIdx ; ++I) {
456111949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
45624419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    if (Pos != Annotated.end() &&
45634419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        (clang_isInvalid(Cursors[I].kind) ||
45644419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor         Pos->second.kind != CXCursor_PreprocessingDirective))
4565fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      Cursors[I] = Pos->second;
4566fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4567fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4568fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Finish up annotating any tokens left.
4569fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (!MoreTokens())
4570fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return;
457111949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4572fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor &C = clang_getNullCursor();
4573fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = TokIdx ; I < NumTokens ; ++I) {
457403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (I < PreprocessingTokIdx && clang_isPreprocessing(Cursors[I].kind))
457503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      continue;
457603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
4577fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
4578fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = (Pos == Annotated.end()) ? C : Pos->second;
457911949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  }
458011949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek}
458111949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4582a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief It annotates and advances tokens with a cursor until the comparison
4583a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis//// between the cursor location and the source range is the same as
4584a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \arg compResult.
4585a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis///
4586a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeBefore to annotate tokens with a cursor until a range is reached.
4587a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeOverlap to annotate tokens inside a range.
4588a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceTokens(CXCursor updateC,
4589a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4590a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
4591a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  while (MoreTokens()) {
4592a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    const unsigned I = NextToken();
45935f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    if (isFunctionMacroToken(I))
45945f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      return annotateAndAdvanceFunctionMacroTokens(updateC, compResult, range);
4595a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4596a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceLocation TokLoc = GetTokenLoc(I);
4597a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4598a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      Cursors[I] = updateC;
4599a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      AdvanceToken();
4600a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue;
4601a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4602a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    break;
4603a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4604a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4605a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4606a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Special annotation handling for macro argument tokens.
46075f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceFunctionMacroTokens(
46085f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                               CXCursor updateC,
4609a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4610a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
46115f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(MoreTokens());
46125f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(isFunctionMacroToken(NextToken()) &&
4613a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis         "Should be called only for macro arg tokens");
4614a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4615a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // This works differently than annotateAndAdvanceTokens; because expanded
4616a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // macro arguments can have arbitrary translation-unit source order, we do not
4617a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // advance the token index one by one until a token fails the range test.
4618a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // We only advance once past all of the macro arg tokens if all of them
4619a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // pass the range test. If one of them fails we keep the token index pointing
4620a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // at the start of the macro arg tokens so that the failing token will be
4621a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // annotated by a subsequent annotation try.
4622a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4623a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  bool atLeastOneCompFail = false;
4624a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4625a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned I = NextToken();
46265f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  for (; I < NumTokens && isFunctionMacroToken(I); ++I) {
46275f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    SourceLocation TokLoc = getFunctionMacroTokenLoc(I);
4628a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (TokLoc.isFileID())
4629a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue; // not macro arg token, it's parens or comma.
4630a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4631a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (clang_isInvalid(clang_getCursorKind(Cursors[I])))
4632a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        Cursors[I] = updateC;
4633a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    } else
4634a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      atLeastOneCompFail = true;
4635a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4636a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4637a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (!atLeastOneCompFail)
4638a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    TokIdx = I; // All of the tokens were handled, advance beyond all of them.
4639a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4640a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
46416db610934bedc6896393c1e1099525b35380acd6Ted Kremenekenum CXChildVisitResult
46424419b675577d7c281a659fab1fec10e1bfbe04c5Douglas GregorAnnotateTokensWorker::Visit(CXCursor cursor, CXCursor parent) {
4643fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CXSourceLocation Loc = clang_getCursorLocation(cursor);
46444419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  SourceRange cursorRange = getRawCursorExtent(cursor);
464581d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor  if (cursorRange.isInvalid())
464681d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor    return CXChildVisit_Recurse;
4647f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4648f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (!HasContextSensitiveKeywords) {
4649f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C properties can have context-sensitive keywords.
4650f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    if (cursor.kind == CXCursor_ObjCPropertyDecl) {
4651f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCPropertyDecl *Property
4652f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(cursor)))
4653f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        HasContextSensitiveKeywords = Property->getPropertyAttributesAsWritten() != 0;
4654f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4655f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C methods can have context-sensitive keywords.
4656f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_ObjCInstanceMethodDecl ||
4657f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ObjCClassMethodDecl) {
4658f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCMethodDecl *Method
4659f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            = dyn_cast_or_null<ObjCMethodDecl>(getCursorDecl(cursor))) {
4660f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->getObjCDeclQualifier())
4661f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4662f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        else {
4663f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          for (ObjCMethodDecl::param_iterator P = Method->param_begin(),
4664f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                                           PEnd = Method->param_end();
4665f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor               P != PEnd; ++P) {
4666f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            if ((*P)->getObjCDeclQualifier()) {
4667f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              HasContextSensitiveKeywords = true;
4668f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              break;
4669f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            }
4670f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          }
4671f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
4672f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4673f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4674f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ methods can have context-sensitive keywords.
4675f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_CXXMethod) {
4676f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (CXXMethodDecl *Method
4677f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<CXXMethodDecl>(getCursorDecl(cursor))) {
4678f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->hasAttr<FinalAttr>() || Method->hasAttr<OverrideAttr>())
4679f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4680f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4681f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4682f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ classes can have context-sensitive keywords.
4683f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_StructDecl ||
4684f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassDecl ||
4685f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplate ||
4686f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplatePartialSpecialization) {
4687f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Decl *D = getCursorDecl(cursor))
4688f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (D->hasAttr<FinalAttr>())
4689f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4690f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4691f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
4692f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
46934419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  if (clang_isPreprocessing(cursor.kind)) {
4694cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // For macro expansions, just note where the beginning of the macro
4695cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // expansion occurs.
46969b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    if (cursor.kind == CXCursor_MacroExpansion) {
46974419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      Annotated[Loc.int_data] = cursor;
46984419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      return CXChildVisit_Recurse;
46994419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47004419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47014419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Items in the preprocessing record are kept separate from items in
47024419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // declarations, so we keep a separate token index.
47034419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    unsigned SavedTokIdx = TokIdx;
47044419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = PreprocessingTokIdx;
47054419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47064419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Skip tokens up until we catch up to the beginning of the preprocessing
47074419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // entry.
47084419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
47094419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
47104419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
47114419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
47124419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
47134419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
47144419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
47154419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
47164419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
47174419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
47184419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
47194419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
47204419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47214419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47224419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Look at all of the tokens within this range.
47234419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
47244419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
47254419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
47264419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
47274419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
4728b219cfc4d75f0a03630b7c4509ef791b7e97b2c8David Blaikie        llvm_unreachable("Infeasible");
47294419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
47304419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
47314419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
47324419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        Cursors[I] = cursor;
47334419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
47344419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
47354419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
47364419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
47374419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47384419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47394419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Save the preprocessing token index; restore the non-preprocessing
47404419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // token index.
47414419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    PreprocessingTokIdx = TokIdx;
47424419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = SavedTokIdx;
47430045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    return CXChildVisit_Recurse;
47440045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  }
4745fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4746fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (cursorRange.isInvalid())
4747fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return CXChildVisit_Continue;
4748a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek
4749fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation L = SourceLocation::getFromRawEncoding(Loc.int_data);
4750fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4751a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  // Adjust the annotated range based specific declarations.
4752a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  const enum CXCursorKind cursorK = clang_getCursorKind(cursor);
4753a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  if (cursorK >= CXCursor_FirstDecl && cursorK <= CXCursor_LastDecl) {
475423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    Decl *D = cxcursor::getCursorDecl(cursor);
47552494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
47562494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
475723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    if (const DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) {
47582494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
47592494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
47602494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    } else if (TypedefDecl *Typedef = dyn_cast<TypedefDecl>(D)) {
47612494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
47622494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
4763a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek    }
47642494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
47652494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && L.isValid() &&
47662494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, L))
47672494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      cursorRange.setBegin(StartLoc);
4768a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  }
476981d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor
47703f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // If the location of the cursor occurs within a macro instantiation, record
47713f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // the spelling location of the cursor in our annotation map.  We can then
47723f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // paper over the token labelings during a post-processing step to try and
47733f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // get cursor mappings for tokens that are the *arguments* of a macro
47743f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // instantiation.
47753f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  if (L.isMacroID()) {
47763f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    unsigned rawEncoding = SrcMgr.getSpellingLoc(L).getRawEncoding();
47773f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // Only invalidate the old annotation if it isn't part of a preprocessing
47783f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // directive.  Here we assume that the default construction of CXCursor
47793f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // results in CXCursor.kind being an initialized value (i.e., 0).  If
47803f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // this isn't the case, we can fix by doing lookup + insertion.
47814419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47823f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    CXCursor &oldC = Annotated[rawEncoding];
47833f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    if (!clang_isPreprocessing(oldC.kind))
47843f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek      oldC = cursor;
47853f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  }
47863f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek
4787fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const enum CXCursorKind K = clang_getCursorKind(parent);
4788fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor updateC =
4789d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek    (clang_isInvalid(K) || K == CXCursor_TranslationUnit)
4790d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek     ? clang_getNullCursor() : parent;
4791fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4792a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(updateC, RangeBefore, cursorRange);
4793fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
47945517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // Avoid having the cursor of an expression "overwrite" the annotation of the
47955517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // variable declaration that it belongs to.
47965517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // This can happen for C++ constructor expressions whose range generally
47975517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // include the variable declaration, e.g.:
47985517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  //  MyCXXClass foo; // Make sure we don't annotate 'foo' as a CallExpr cursor.
47995517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  if (clang_isExpression(cursorK)) {
48005517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    Expr *E = getCursorExpr(cursor);
48018ccac3de1335f1cfd7cea56ba1cefcf0b724ce3fArgyrios Kyrtzidis    if (Decl *D = getCursorParentDecl(cursor)) {
48025517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      const unsigned I = NextToken();
48035517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      if (E->getLocStart().isValid() && D->getLocation().isValid() &&
48045517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == D->getLocation() &&
48055517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == GetTokenLoc(I)) {
48065517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        Cursors[I] = updateC;
48075517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        AdvanceToken();
48085517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      }
48095517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    }
48105517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  }
48115517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis
4812fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Visit children to get their cursor information.
4813fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned BeforeChildren = NextToken();
4814fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  VisitChildren(cursor);
4815fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned AfterChildren = NextToken();
4816fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4817a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // Scan the tokens that are at the end of the cursor, but are not captured
4818a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // but the child cursors.
4819a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(cursor, RangeOverlap, cursorRange);
48206db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
4821fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Scan the tokens that are at the beginning of the cursor, but are not
4822fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // capture by the child cursors.
4823fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = BeforeChildren; I != AfterChildren; ++I) {
4824fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    if (!clang_isInvalid(clang_getCursorKind(Cursors[I])))
4825fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      break;
48264419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
4827fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = cursor;
4828fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4829fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4830fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  return CXChildVisit_Continue;
48310045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor}
48320045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
48336db610934bedc6896393c1e1099525b35380acd6Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
48346db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXCursor parent,
48356db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXClientData client_data) {
48366db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  return static_cast<AnnotateTokensWorker*>(client_data)->Visit(cursor, parent);
48376db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
48386db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
48396628a614c504263ae539462f049d523dd07ac1baTed Kremeneknamespace {
4840a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4841a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Uses the macro expansions in the preprocessing record to find
4842a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// and mark tokens that are macro arguments. This info is used by the
4843a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// AnnotateTokensWorker.
4844a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisclass MarkMacroArgTokensVisitor {
4845a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceManager &SM;
4846a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXToken *Tokens;
4847a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned NumTokens;
4848a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned CurIdx;
4849a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4850a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidispublic:
4851a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  MarkMacroArgTokensVisitor(SourceManager &SM,
4852a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                            CXToken *tokens, unsigned numTokens)
4853a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    : SM(SM), Tokens(tokens), NumTokens(numTokens), CurIdx(0) { }
4854a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4855a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXChildVisitResult visit(CXCursor cursor, CXCursor parent) {
4856a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (cursor.kind != CXCursor_MacroExpansion)
4857a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue;
4858a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4859a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceRange macroRange = getCursorMacroExpansion(cursor)->getSourceRange();
4860a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (macroRange.getBegin() == macroRange.getEnd())
4861a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue; // it's not a function macro.
4862a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4863a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4864a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(getTokenLoc(CurIdx),
4865a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                        macroRange.getBegin()))
4866a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4867a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4868a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4869a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4870a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4871a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4872a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4873a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      SourceLocation tokLoc = getTokenLoc(CurIdx);
4874a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(tokLoc, macroRange.getEnd()))
4875a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4876a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
48775f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      setFunctionMacroTokenLoc(CurIdx, SM.getMacroArgExpandedLocation(tokLoc));
4878a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4879a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4880a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4881a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4882a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4883a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return CXChildVisit_Continue;
4884a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4885a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4886a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisprivate:
4887a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceLocation getTokenLoc(unsigned tokI) {
4888a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4889a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4890a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
48915f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void setFunctionMacroTokenLoc(unsigned tokI, SourceLocation loc) {
4892a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // The third field is reserved and currently not used. Use it here
4893a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // to mark macro arg expanded tokens with their expanded locations.
4894a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    Tokens[tokI].int_data[3] = loc.getRawEncoding();
4895a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4896a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis};
4897a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4898a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis} // end anonymous namespace
4899a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4900a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisstatic CXChildVisitResult
4901a676379b26edc959193f9f919ba9c6d296a57824Argyrios KyrtzidisMarkMacroArgTokensVisitorDelegate(CXCursor cursor, CXCursor parent,
4902a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                  CXClientData client_data) {
4903a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  return static_cast<MarkMacroArgTokensVisitor*>(client_data)->visit(cursor,
4904a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                                                     parent);
4905a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4906a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4907a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisnamespace {
49086628a614c504263ae539462f049d523dd07ac1baTed Kremenek  struct clang_annotateTokens_Data {
49096628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXTranslationUnit TU;
49106628a614c504263ae539462f049d523dd07ac1baTed Kremenek    ASTUnit *CXXUnit;
49116628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXToken *Tokens;
49126628a614c504263ae539462f049d523dd07ac1baTed Kremenek    unsigned NumTokens;
49136628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXCursor *Cursors;
49146628a614c504263ae539462f049d523dd07ac1baTed Kremenek  };
4915ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek}
4916ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek
4917ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisstatic void annotatePreprocessorTokens(CXTranslationUnit TU,
4918ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       SourceRange RegionOfInterest,
4919ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       AnnotateTokensData &Annotated) {
4920ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4921ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4922ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SourceManager &SourceMgr = CXXUnit->getSourceManager();
4923ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  std::pair<FileID, unsigned> BeginLocInfo
4924ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(RegionOfInterest.getBegin());
4925ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  std::pair<FileID, unsigned> EndLocInfo
4926ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(RegionOfInterest.getEnd());
4927ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4928ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (BeginLocInfo.first != EndLocInfo.first)
4929ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4930ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4931ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  StringRef Buffer;
4932ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  bool Invalid = false;
4933ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Buffer = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid);
4934ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (Buffer.empty() || Invalid)
4935ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4936ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4937ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
4938ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis            CXXUnit->getASTContext().getLangOptions(),
4939ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis            Buffer.begin(), Buffer.data() + BeginLocInfo.second,
4940ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis            Buffer.end());
4941ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Lex.SetCommentRetentionState(true);
4942ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4943ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  // Lex tokens in raw mode until we hit the end of the range, to avoid
4944ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  // entering #includes or expanding macros.
4945ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  while (true) {
4946ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    Token Tok;
4947ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    Lex.LexFromRawLexer(Tok);
4948ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4949ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  reprocess:
4950ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    if (Tok.is(tok::hash) && Tok.isAtStartOfLine()) {
4951ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // We have found a preprocessing directive. Gobble it up so that we
4952ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // don't see it while preprocessing these tokens later, but keep track
4953ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // of all of the token locations inside this preprocessing directive so
4954ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // that we can annotate them appropriately.
4955ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      //
4956ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // FIXME: Some simple tests here could identify macro definitions and
4957ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // #undefs, to provide specific cursor kinds for those.
4958ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      SmallVector<SourceLocation, 32> Locations;
4959ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      do {
4960ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Locations.push_back(Tok.getLocation());
4961ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Lex.LexFromRawLexer(Tok);
4962ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      } while (!Tok.isAtStartOfLine() && !Tok.is(tok::eof));
4963ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4964ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      using namespace cxcursor;
4965ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      CXCursor Cursor
4966ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      = MakePreprocessingDirectiveCursor(SourceRange(Locations.front(),
4967ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                                     Locations.back()),
4968ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                         TU);
4969ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      for (unsigned I = 0, N = Locations.size(); I != N; ++I) {
4970ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Annotated[Locations[I].getRawEncoding()] = Cursor;
4971ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
4972ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4973ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      if (Tok.isAtStartOfLine())
4974ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto reprocess;
4975ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4976ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      continue;
4977ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
4978ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4979ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    if (Tok.is(tok::eof))
4980ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      break;
4981ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
4982ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis}
4983ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
49846628a614c504263ae539462f049d523dd07ac1baTed Kremenek// This gets run a separate thread to avoid stack blowout.
49856628a614c504263ae539462f049d523dd07ac1baTed Kremenekstatic void clang_annotateTokensImpl(void *UserData) {
49866628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXTranslationUnit TU = ((clang_annotateTokens_Data*)UserData)->TU;
49876628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = ((clang_annotateTokens_Data*)UserData)->CXXUnit;
49886628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXToken *Tokens = ((clang_annotateTokens_Data*)UserData)->Tokens;
49896628a614c504263ae539462f049d523dd07ac1baTed Kremenek  const unsigned NumTokens = ((clang_annotateTokens_Data*)UserData)->NumTokens;
49906628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor *Cursors = ((clang_annotateTokens_Data*)UserData)->Cursors;
4991fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
49920396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Determine the region of interest, which contains all of the tokens.
49930045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  SourceRange RegionOfInterest;
49946628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setBegin(
49956628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU, Tokens[0])));
49966628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setEnd(
49976628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU,
49986628a614c504263ae539462f049d523dd07ac1baTed Kremenek                                                         Tokens[NumTokens-1])));
4999fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
50000396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // A mapping from the source locations found when re-lexing or traversing the
50010396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // region of interest to the corresponding cursors.
50020045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  AnnotateTokensData Annotated;
5003ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5004fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Relex the tokens within the source range to look for preprocessing
50050396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // directives.
5006ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  annotatePreprocessorTokens(TU, RegionOfInterest, Annotated);
50076628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5008a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (CXXUnit->getPreprocessor().getPreprocessingRecord()) {
5009a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // Search and mark tokens that are macro argument expansions.
5010a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MarkMacroArgTokensVisitor Visitor(CXXUnit->getSourceManager(),
5011a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                      Tokens, NumTokens);
5012a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    CursorVisitor MacroArgMarker(TU,
5013a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                 MarkMacroArgTokensVisitorDelegate, &Visitor,
5014f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                 /*VisitPreprocessorLast=*/true,
5015e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                                 /*VisitIncludedEntities=*/false,
5016f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                 RegionOfInterest);
5017a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MacroArgMarker.visitPreprocessedEntitiesInRegion();
5018a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
5019a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
50200396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Annotate all of the source locations in the region of interest that map to
5021fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // a specific cursor.
5022fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  AnnotateTokensWorker W(Annotated, Tokens, Cursors, NumTokens,
5023a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                         TU, RegionOfInterest);
50246628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50256c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // FIXME: We use a ridiculous stack size here because the data-recursion
50266c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm uses a large stack frame than the non-data recursive version,
50276c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // and AnnotationTokensWorker currently transforms the data-recursion
50286c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm back into a traditional recursion by explicitly calling
50296c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // VisitChildren().  We will need to remove this explicit recursive call.
50306628a614c504263ae539462f049d523dd07ac1baTed Kremenek  W.AnnotateTokens();
50316628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5032f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // If we ran into any entities that involve context-sensitive keywords,
5033f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // take another pass through the tokens to mark them as such.
5034f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (W.hasContextSensitiveKeywords()) {
5035f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    for (unsigned I = 0; I != NumTokens; ++I) {
5036f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (clang_getTokenKind(Tokens[I]) != CXToken_Identifier)
5037f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5038f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5039f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCPropertyDecl) {
5040f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5041f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (ObjCPropertyDecl *Property
50426628a614c504263ae539462f049d523dd07ac1baTed Kremenek            = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(Cursors[I]))) {
5043f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          if (Property->getPropertyAttributesAsWritten() != 0 &&
5044f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              llvm::StringSwitch<bool>(II->getName())
50456628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readonly", true)
50466628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("assign", true)
5047f85e193739c953358c865005855253af4f68a497John McCall              .Case("unsafe_unretained", true)
50486628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readwrite", true)
50496628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("retain", true)
50506628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("copy", true)
50516628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("nonatomic", true)
50526628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("atomic", true)
50536628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("getter", true)
50546628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("setter", true)
5055f85e193739c953358c865005855253af4f68a497John McCall              .Case("strong", true)
5056f85e193739c953358c865005855253af4f68a497John McCall              .Case("weak", true)
50576628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Default(false))
5058f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            Tokens[I].int_data[0] = CXToken_Keyword;
5059f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
5060f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5061f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5062f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5063f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCInstanceMethodDecl ||
5064f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Cursors[I].kind == CXCursor_ObjCClassMethodDecl) {
5065f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5066f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (llvm::StringSwitch<bool>(II->getName())
50676628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("in", true)
50686628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("out", true)
50696628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("inout", true)
50706628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("oneway", true)
50716628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("bycopy", true)
50726628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("byref", true)
50736628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Default(false))
5074f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Tokens[I].int_data[0] = CXToken_Keyword;
5075f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5076f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
50776639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
50786639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      if (Cursors[I].kind == CXCursor_CXXFinalAttr ||
50796639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis          Cursors[I].kind == CXCursor_CXXOverrideAttr) {
50806639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis        Tokens[I].int_data[0] = CXToken_Keyword;
50816628a614c504263ae539462f049d523dd07ac1baTed Kremenek        continue;
5082f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5083f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
5084f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
5085fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
50866628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50876628a614c504263ae539462f049d523dd07ac1baTed Kremenekextern "C" {
50886628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50896628a614c504263ae539462f049d523dd07ac1baTed Kremenekvoid clang_annotateTokens(CXTranslationUnit TU,
50906628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXToken *Tokens, unsigned NumTokens,
50916628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXCursor *Cursors) {
50926628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50936628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (NumTokens == 0 || !Tokens || !Cursors)
50946628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
50956628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50966628a614c504263ae539462f049d523dd07ac1baTed Kremenek  // Any token we don't specifically annotate will have a NULL cursor.
50976628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor C = clang_getNullCursor();
50986628a614c504263ae539462f049d523dd07ac1baTed Kremenek  for (unsigned I = 0; I != NumTokens; ++I)
50996628a614c504263ae539462f049d523dd07ac1baTed Kremenek    Cursors[I] = C;
51006628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51016628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
51026628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!CXXUnit)
51036628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
51046628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51056628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
51066628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51076628a614c504263ae539462f049d523dd07ac1baTed Kremenek  clang_annotateTokens_Data data = { TU, CXXUnit, Tokens, NumTokens, Cursors };
51086628a614c504263ae539462f049d523dd07ac1baTed Kremenek  llvm::CrashRecoveryContext CRC;
51096628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!RunSafely(CRC, clang_annotateTokensImpl, &data,
51106628a614c504263ae539462f049d523dd07ac1baTed Kremenek                 GetSafetyThreadStackSize() * 2)) {
51116628a614c504263ae539462f049d523dd07ac1baTed Kremenek    fprintf(stderr, "libclang: crash detected while annotating tokens\n");
51126628a614c504263ae539462f049d523dd07ac1baTed Kremenek  }
51136628a614c504263ae539462f049d523dd07ac1baTed Kremenek}
51146628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5115fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor} // end: extern "C"
5116fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
5117fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
511816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek// Operations for querying linkage of a cursor.
511916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
512016b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
512116b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenekextern "C" {
512216b4259aecaa22b642d35d36fd89965ed700c1e0Ted KremenekCXLinkageKind clang_getCursorLinkage(CXCursor cursor) {
51230396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  if (!clang_isDeclaration(cursor.kind))
51240396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    return CXLinkage_Invalid;
51250396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor
512616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  Decl *D = cxcursor::getCursorDecl(cursor);
512716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  if (NamedDecl *ND = dyn_cast_or_null<NamedDecl>(D))
512816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    switch (ND->getLinkage()) {
512916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case NoLinkage: return CXLinkage_NoLinkage;
513016b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case InternalLinkage: return CXLinkage_Internal;
513116b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case UniqueExternalLinkage: return CXLinkage_UniqueExternal;
513216b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case ExternalLinkage: return CXLinkage_External;
513316b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    };
513416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
513516b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  return CXLinkage_Invalid;
513616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek}
513716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek} // end: extern "C"
513816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
513916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
514045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek// Operations for querying language of a cursor.
514145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
514245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
514345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekstatic CXLanguageKind getDeclLanguage(const Decl *D) {
514445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  switch (D->getKind()) {
514545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    default:
514645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      break;
514745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ImplicitParam:
514845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCAtDefsField:
514945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategory:
515045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategoryImpl:
515145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCClass:
515245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCompatibleAlias:
515345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCForwardProtocol:
515445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCImplementation:
515545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCInterface:
515645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCIvar:
515745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCMethod:
515845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProperty:
515945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCPropertyImpl:
516045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProtocol:
516145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_ObjC;
516245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConstructor:
516345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConversion:
516445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXDestructor:
516545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXMethod:
516645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXRecord:
516745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplate:
516845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplatePartialSpecialization:
516945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplateSpecialization:
517045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Friend:
517145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FriendTemplate:
517245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FunctionTemplate:
517345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::LinkageSpec:
517445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Namespace:
517545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NamespaceAlias:
517645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NonTypeTemplateParm:
517745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::StaticAssert:
517845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTemplateParm:
517945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTypeParm:
518045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingTypename:
518145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingValue:
518245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Using:
518345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingDirective:
518445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingShadow:
518545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_CPlusPlus;
518645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  }
518745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
518845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_C;
518945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
519045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
519145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekextern "C" {
519258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
519358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregorenum CXAvailabilityKind clang_getCursorAvailability(CXCursor cursor) {
519458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  if (clang_isDeclaration(cursor.kind))
519558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    if (Decl *D = cxcursor::getCursorDecl(cursor)) {
51960a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      if (isa<FunctionDecl>(D) && cast<FunctionDecl>(D)->isDeleted())
519758ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Available;
519858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
51990a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      switch (D->getAvailability()) {
52000a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Available:
52010a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_NotYetIntroduced:
52020a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_Available;
52030a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52040a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Deprecated:
520558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Deprecated;
52060a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52070a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Unavailable:
52080a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_NotAvailable;
52090a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      }
521058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
52110a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
521258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  return CXAvailability_Available;
521358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor}
521458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
521545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted KremenekCXLanguageKind clang_getCursorLanguage(CXCursor cursor) {
521645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  if (clang_isDeclaration(cursor.kind))
521745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    return getDeclLanguage(cxcursor::getCursorDecl(cursor));
521845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
521945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_Invalid;
522045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
52213910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52223910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// \brief If the given cursor is the "templated" declaration
52233910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// descibing a class or function template, return the class or
52243910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// function template.
52253910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregorstatic Decl *maybeGetTemplateCursor(Decl *D) {
52263910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (!D)
52273910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    return 0;
52283910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52293910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
52303910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (FunctionTemplateDecl *FunTmpl = FD->getDescribedFunctionTemplate())
52313910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return FunTmpl;
52323910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52333910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D))
52343910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (ClassTemplateDecl *ClassTmpl = RD->getDescribedClassTemplate())
52353910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return ClassTmpl;
52363910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52373910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  return D;
52383910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor}
52393910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52402be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorSemanticParent(CXCursor cursor) {
52412be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
52422be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
52432be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getDeclContext();
52443910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
52453910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
52463910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52473910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
52483910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
52492be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
52502be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52512be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52522be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isStatement(cursor.kind) || clang_isExpression(cursor.kind)) {
52532be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor))
5254a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(D, getCursorTU(cursor));
52552be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52562be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52572be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
52582be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
52592be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52602be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorLexicalParent(CXCursor cursor) {
52612be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
52622be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
52632be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getLexicalDeclContext();
52643910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
52653910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
52663910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52673910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
52683910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
52692be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
52702be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52712be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52722be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // FIXME: Note that we can't easily compute the lexical context of a
52732be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // statement or expression, so we return nothing.
52742be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
52752be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
52762be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52779f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_getOverriddenCursors(CXCursor cursor,
52789f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                CXCursor **overridden,
52799f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                unsigned *num_overridden) {
52809f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (overridden)
52819f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *overridden = 0;
52829f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (num_overridden)
52839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *num_overridden = 0;
52849f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!overridden || !num_overridden)
52859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
52869f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
5287b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  SmallVector<CXCursor, 8> Overridden;
5288b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  cxcursor::getOverriddenCursors(cursor, Overridden);
52899f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
52902407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek  // Don't allocate memory if we have no overriden cursors.
52912407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek  if (Overridden.size() == 0)
52922407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek    return;
52932407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek
5294b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  *num_overridden = Overridden.size();
5295b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  *overridden = new CXCursor [Overridden.size()];
5296b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  std::copy(Overridden.begin(), Overridden.end(), *overridden);
52979f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
52989f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
52999f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_disposeOverriddenCursors(CXCursor *overridden) {
53009f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  delete [] overridden;
53019f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53029f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
5303ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas GregorCXFile clang_getIncludedFile(CXCursor cursor) {
5304ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (cursor.kind != CXCursor_InclusionDirective)
5305ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return 0;
5306ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
5307ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  InclusionDirective *ID = getCursorInclusionDirective(cursor);
5308ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  return (void *)ID->getFile();
5309ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor}
5310ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
531145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek} // end: extern "C"
531245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
53139ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
53149ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
53159ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek// C++ AST instrospection.
53169ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
53179ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
53189ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekextern "C" {
53199ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekunsigned clang_CXXMethod_isStatic(CXCursor C) {
53209ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek  if (!clang_isDeclaration(C.kind))
53219ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek    return 0;
532249f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor
532349f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  CXXMethodDecl *Method = 0;
532449f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
532549f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
532649f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
532749f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  else
532849f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
532949f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  return (Method && Method->isStatic()) ? 1 : 0;
533040b492a43bac3ed0c465772aa6921d011cfc273fTed Kremenek}
5331b12903e1a4b8d1b611b8c7e4f910665d628e68cdTed Kremenek
5332211924b563aa31421836cee7655be729ad02733fDouglas Gregorunsigned clang_CXXMethod_isVirtual(CXCursor C) {
5333211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (!clang_isDeclaration(C.kind))
5334211924b563aa31421836cee7655be729ad02733fDouglas Gregor    return 0;
5335211924b563aa31421836cee7655be729ad02733fDouglas Gregor
5336211924b563aa31421836cee7655be729ad02733fDouglas Gregor  CXXMethodDecl *Method = 0;
5337211924b563aa31421836cee7655be729ad02733fDouglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
5338211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
5339211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
5340211924b563aa31421836cee7655be729ad02733fDouglas Gregor  else
5341211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
5342211924b563aa31421836cee7655be729ad02733fDouglas Gregor  return (Method && Method->isVirtual()) ? 1 : 0;
5343211924b563aa31421836cee7655be729ad02733fDouglas Gregor}
53449ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek} // end: extern "C"
53459ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
534645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
534795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek// Attribute introspection.
534895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
534995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
535095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenekextern "C" {
535195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted KremenekCXType clang_getIBOutletCollectionType(CXCursor C) {
535295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  if (C.kind != CXCursor_IBOutletCollectionAttr)
5353a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return cxtype::MakeCXType(QualType(), cxcursor::getCursorTU(C));
535495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
535595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  IBOutletCollectionAttr *A =
535695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(C));
535795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
535818aa2ff4641847d7f8866e8c5912d4d0ddb858ceArgyrios Kyrtzidis  return cxtype::MakeCXType(A->getInterface(), cxcursor::getCursorTU(C));
535995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek}
536095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek} // end: extern "C"
536195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
536295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
536359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek// Inspecting memory usage.
536459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
536559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5366f787002478f09af1741fb0f82a562002e6799c49Ted Kremenektypedef std::vector<CXTUResourceUsageEntry> MemUsageEntries;
536759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5368f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekstatic inline void createCXTUResourceUsageEntry(MemUsageEntries &entries,
5369f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek                                              enum CXTUResourceUsageKind k,
5370ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek                                              unsigned long amount) {
5371f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsageEntry entry = { k, amount };
537259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.push_back(entry);
537359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
537459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
537559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekextern "C" {
537659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5377f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekconst char *clang_getTUResourceUsageName(CXTUResourceUsageKind kind) {
537859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  const char *str = "";
537959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  switch (kind) {
5380f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_AST:
538159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: expressions, declarations, and types";
538259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5383f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Identifiers:
538459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: identifiers";
538559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5386f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Selectors:
538759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: selectors";
5388e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5389f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_GlobalCompletionResults:
53904e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      str = "Code completion: cached global results";
5391e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5392457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek    case CXTUResourceUsage_SourceManagerContentCache:
5393457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      str = "SourceManager: content cache allocator";
5394457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      break;
5395ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    case CXTUResourceUsage_AST_SideTables:
5396ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      str = "ASTContext: side tables";
5397ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      break;
5398f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_Malloc:
5399f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: malloc'ed memory buffers";
5400f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5401f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_MMap:
5402f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: mmap'ed memory buffers";
5403f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5404e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc:
5405e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: malloc'ed memory buffers";
5406e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
5407e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_MMap:
5408e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: mmap'ed memory buffers";
5409e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
54105e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_Preprocessor:
54115e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: malloc'ed memory";
54125e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
54135e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_PreprocessingRecord:
54145e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: PreprocessingRecord";
54155e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
5416ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek    case CXTUResourceUsage_SourceManager_DataStructures:
5417ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      str = "SourceManager: data structures and tables";
5418ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      break;
5419d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek    case CXTUResourceUsage_Preprocessor_HeaderSearch:
5420d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      str = "Preprocessor: header search tables";
5421d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      break;
542259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
542359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return str;
542459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
542559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5426f787002478f09af1741fb0f82a562002e6799c49Ted KremenekCXTUResourceUsage clang_getCXTUResourceUsage(CXTranslationUnit TU) {
542759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (!TU) {
5428f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    CXTUResourceUsage usage = { (void*) 0, 0, 0 };
542959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return usage;
543059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
543159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
543259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTUnit *astUnit = static_cast<ASTUnit*>(TU->TUData);
543359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  llvm::OwningPtr<MemUsageEntries> entries(new MemUsageEntries());
543459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTContext &astContext = astUnit->getASTContext();
543559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
543659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by AST nodes and types?
5437f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST,
5438ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getASTAllocatedMemory());
543959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
544059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by identifiers?
5441f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Identifiers,
544259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Idents.getAllocator().getTotalMemory());
544359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
544459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used for selectors?
5445f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Selectors,
544659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Selectors.getTotalMemory());
544759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5448ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  // How much memory is used by ASTContext's side tables?
5449ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST_SideTables,
5450ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getSideTableAllocatedMemory());
5451ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek
54524e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  // How much memory is used for caching global code completion results?
54534e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned long completionBytes = 0;
54544e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  if (GlobalCodeCompletionAllocator *completionAllocator =
54554e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      astUnit->getCachedCompletionAllocator().getPtr()) {
54565e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    completionBytes = completionAllocator->getTotalMemory();
54574e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  }
5458457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5459457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               CXTUResourceUsage_GlobalCompletionResults,
5460457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               completionBytes);
5461457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek
5462457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  // How much memory is being used by SourceManager's content cache?
5463457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5464457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          CXTUResourceUsage_SourceManagerContentCache,
5465457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          (unsigned long) astContext.getSourceManager().getContentCacheSize());
5466f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5467f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  // How much memory is being used by the MemoryBuffer's in SourceManager?
5468f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  const SourceManager::MemoryBufferSizes &srcBufs =
5469f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    astUnit->getSourceManager().getMemoryBufferSizes();
5470f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5471f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5472f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_Malloc,
5473f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.malloc_bytes);
5474ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5475f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_MMap,
5476f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.mmap_bytes);
5477ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5478ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               CXTUResourceUsage_SourceManager_DataStructures,
5479ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               (unsigned long) astContext.getSourceManager()
5480ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                                .getDataStructureSizes());
5481e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5482e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  // How much memory is being used by the ExternalASTSource?
5483e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  if (ExternalASTSource *esrc = astContext.getExternalSource()) {
5484e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    const ExternalASTSource::MemoryBufferSizes &sizes =
5485e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      esrc->getMemoryBufferSizes();
5486e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5487e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5488e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc,
5489e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.malloc_bytes);
5490e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5491e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_MMap,
5492e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.mmap_bytes);
5493e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  }
54945e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
54955e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  // How much memory is being used by the Preprocessor?
54965e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  Preprocessor &pp = astUnit->getPreprocessor();
54975e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  createCXTUResourceUsageEntry(*entries,
54985e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                               CXTUResourceUsage_Preprocessor,
5499c5c5e92ec53f7e6ac7ebbbf77c6d8e4b7d88daecArgyrios Kyrtzidis                               pp.getTotalMemory());
55005e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
55015e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  if (PreprocessingRecord *pRec = pp.getPreprocessingRecord()) {
55025e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    createCXTUResourceUsageEntry(*entries,
55035e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 CXTUResourceUsage_PreprocessingRecord,
55045e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 pRec->getTotalMemory());
55055e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  }
55065e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5507d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5508d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               CXTUResourceUsage_Preprocessor_HeaderSearch,
5509d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               pp.getHeaderSearchInfo().getTotalMemory());
55105e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5511f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = { (void*) entries.get(),
551259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            (unsigned) entries->size(),
551359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            entries->size() ? &(*entries)[0] : 0 };
551459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.take();
551559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return usage;
551659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
551759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5518f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekvoid clang_disposeCXTUResourceUsage(CXTUResourceUsage usage) {
551959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (usage.data)
552059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    delete (MemUsageEntries*) usage.data;
552159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
552259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
552359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek} // end extern "C"
552459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
55256df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregorvoid clang::PrintLibclangResourceUsage(CXTranslationUnit TU) {
55266df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  CXTUResourceUsage Usage = clang_getCXTUResourceUsage(TU);
55276df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  for (unsigned I = 0; I != Usage.numEntries; ++I)
55286df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    fprintf(stderr, "  %s: %lu\n",
55296df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            clang_getTUResourceUsageName(Usage.entries[I].kind),
55306df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            Usage.entries[I].amount);
55316df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
55326df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  clang_disposeCXTUResourceUsage(Usage);
55336df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor}
55346df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
553559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
553604bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek// Misc. utility functions.
553704bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek//===----------------------------------------------------------------------===//
5538f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
5539abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbar/// Default to using an 8 MB stack size on "safety" threads.
5540abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbarstatic unsigned SafetyStackThreadSize = 8 << 20;
5541bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5542bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarnamespace clang {
5543bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5544bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarbool RunSafely(llvm::CrashRecoveryContext &CRC,
55456c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               void (*Fn)(void*), void *UserData,
55466c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               unsigned Size) {
55476c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (!Size)
55486c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek    Size = GetSafetyThreadStackSize();
55496c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (Size)
5550bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar    return CRC.RunSafelyOnThread(Fn, UserData, Size);
5551bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return CRC.RunSafely(Fn, UserData);
5552bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5553bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5554bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarunsigned GetSafetyThreadStackSize() {
5555bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return SafetyStackThreadSize;
5556bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5557bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5558bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarvoid SetSafetyThreadStackSize(unsigned Value) {
5559bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  SafetyStackThreadSize = Value;
5560bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5561bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5562bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5563bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
556404bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenekextern "C" {
556504bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
5566a2a9d6e4e5b6001b86b7dfc5db1ea296ce29a3d3Ted KremenekCXString clang_getClangVersion() {
5567ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(getClangFullVersion());
556804bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek}
556904bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
557004bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek} // end: extern "C"
557159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5572