CIndex.cpp revision 4e4d08403ca5cfd4d558fa2936215d3a4e5a528d
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"
38b2c60b04a597cc5ba4154837cf8e0a155a376fd7Argyrios Kyrtzidis#include "llvm/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);
15416ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (!D) {
15516ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      assert(0 && "Invalid declaration cursor");
15616ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      return true; // abort.
15716ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    }
15816ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
15965ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // Ignore implicit declarations, unless it's an objc method because
16065ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // currently we should report implicit methods for properties when indexing.
16165ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    if (D->isImplicit() && !isa<ObjCMethodDecl>(D))
162b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return false;
163b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
1640d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
16533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // If we have a range of interest, and this cursor doesn't intersect with it,
16633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // we're done.
16733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (RegionOfInterest.isValid() && !CheckedRegionOfInterest) {
168a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    SourceRange Range = getRawCursorExtent(Cursor);
169f408f32aa9ae3d97bc656267dc5d78fa7d03499bDaniel Dunbar    if (Range.isInvalid() || CompareRegionOfInterest(Range))
17033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor      return false;
17133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  }
172f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
173b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  switch (Visitor(Cursor, Parent, ClientData)) {
174b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Break:
175b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
1760d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
177b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Continue:
178b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return false;
1792e331b938b38057e333fab0ba841130ea8467794Douglas Gregor
180b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Recurse:
181b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return VisitChildren(Cursor);
182b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
1830d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
1847530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie  llvm_unreachable("Invalid CXChildVisitResult!");
185b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
1860d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
187f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidisstatic bool visitPreprocessedEntitiesInRange(SourceRange R,
188f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                             PreprocessingRecord &PPRec,
189f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                             CursorVisitor &Visitor) {
190f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  SourceManager &SM = Visitor.getASTUnit()->getSourceManager();
191f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  FileID FID;
192f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
193e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis  if (!Visitor.shouldVisitIncludedEntities()) {
194f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    // If the begin/end of the range lie in the same FileID, do the optimization
195f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    // where we skip preprocessed entities that do not come from the same FileID.
196acca41167ce78bb032906f6b1d2ced62efbe059aArgyrios Kyrtzidis    FID = SM.getFileID(SM.getFileLoc(R.getBegin()));
197acca41167ce78bb032906f6b1d2ced62efbe059aArgyrios Kyrtzidis    if (FID != SM.getFileID(SM.getFileLoc(R.getEnd())))
198f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      FID = FileID();
199f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  }
200f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
201f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  std::pair<PreprocessingRecord::iterator, PreprocessingRecord::iterator>
202f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    Entities = PPRec.getPreprocessedEntitiesInRange(R);
203f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  return Visitor.visitPreprocessedEntities(Entities.first, Entities.second,
204f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                           PPRec, FID);
205f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis}
206f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
207dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidisvoid CursorVisitor::visitFileRegion() {
208dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (RegionOfInterest.isInvalid())
209dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
210dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
211dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  ASTUnit *Unit = static_cast<ASTUnit *>(TU->TUData);
212dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceManager &SM = Unit->getSourceManager();
213dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
214dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  std::pair<FileID, unsigned>
215dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Begin = SM.getDecomposedLoc(SM.getFileLoc(RegionOfInterest.getBegin())),
216dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End = SM.getDecomposedLoc(SM.getFileLoc(RegionOfInterest.getEnd()));
217dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
218dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (End.first != Begin.first) {
219dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    // If the end does not reside in the same file, try to recover by
220dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    // picking the end of the file of begin location.
221dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End.first = Begin.first;
222dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    End.second = SM.getFileIDSize(Begin.first);
223dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
224dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
225dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  assert(Begin.first == End.first);
226dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (Begin.second > End.second)
227dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
228dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
229dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  FileID File = Begin.first;
230dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  unsigned Offset = Begin.second;
231dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  unsigned Length = End.second - Begin.second;
232dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
233b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!VisitDeclsOnly && !VisitPreprocessorLast)
234b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis    if (visitPreprocessedEntitiesInRegion())
235b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis      return; // visitation break.
236dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
237dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  visitDeclsFromFileRegion(File, Offset, Length);
238dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
239b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!VisitDeclsOnly && VisitPreprocessorLast)
240dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    visitPreprocessedEntitiesInRegion();
241dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis}
242dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
243e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidisstatic bool isInLexicalContext(Decl *D, DeclContext *DC) {
244e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  if (!DC)
245e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    return false;
246e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
247e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  for (DeclContext *DeclDC = D->getLexicalDeclContext();
248e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis         DeclDC; DeclDC = DeclDC->getLexicalParent()) {
249e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (DeclDC == DC)
250e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      return true;
251e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  }
252e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  return false;
253e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis}
254e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
255dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidisvoid CursorVisitor::visitDeclsFromFileRegion(FileID File,
256dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis                                             unsigned Offset, unsigned Length) {
257dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  ASTUnit *Unit = static_cast<ASTUnit *>(TU->TUData);
258dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceManager &SM = Unit->getSourceManager();
259dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SourceRange Range = RegionOfInterest;
260dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
261dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SmallVector<Decl *, 16> Decls;
262dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  Unit->findFileRegionDecls(File, Offset, Length, Decls);
263dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
264dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // If we didn't find any file level decls for the file, try looking at the
265dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // file that it was included from.
266c14a03dffff69b5e1c55cc118fc52d8fd9f3a28dArgyrios Kyrtzidis  while (Decls.empty() || Decls.front()->isTopLevelDeclInObjCContainer()) {
267dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    bool Invalid = false;
268dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    const SrcMgr::SLocEntry &SLEntry = SM.getSLocEntry(File, &Invalid);
269dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (Invalid)
270dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      return;
271dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
272dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    SourceLocation Outer;
273dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (SLEntry.isFile())
274dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      Outer = SLEntry.getFile().getIncludeLoc();
275dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    else
276dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      Outer = SLEntry.getExpansion().getExpansionLocStart();
277dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (Outer.isInvalid())
278dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      return;
279dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
280dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    llvm::tie(File, Offset) = SM.getDecomposedExpansionLoc(Outer);
281dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Length = 0;
282dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Unit->findFileRegionDecls(File, Offset, Length, Decls);
283dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
284dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
285dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  assert(!Decls.empty());
286dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
287dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  bool VisitedAtLeastOnce = false;
288e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis  DeclContext *CurDC = 0;
289dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  SmallVector<Decl *, 16>::iterator DIt = Decls.begin();
290dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  for (SmallVector<Decl *, 16>::iterator DE = Decls.end(); DIt != DE; ++DIt) {
291dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Decl *D = *DIt;
292ed8bef44c0545fd55a78715606f8d733f6498b21Argyrios Kyrtzidis    if (D->getSourceRange().isInvalid())
293ed8bef44c0545fd55a78715606f8d733f6498b21Argyrios Kyrtzidis      continue;
294dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
295e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (isInLexicalContext(D, CurDC))
296e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      continue;
297e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
298e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    CurDC = dyn_cast<DeclContext>(D);
299e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
300e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis    if (TagDecl *TD = dyn_cast<TagDecl>(D))
301e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis      if (!TD->isFreeStanding())
302e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis        continue;
303e2079cf54ded1eda9e35d215aef6628373368276Argyrios Kyrtzidis
304dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    RangeComparisonResult CompRes = RangeCompare(SM, D->getSourceRange(),Range);
305dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CompRes == RangeBefore)
306dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      continue;
307dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CompRes == RangeAfter)
308dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
309dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
310dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    assert(CompRes == RangeOverlap);
311dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    VisitedAtLeastOnce = true;
31203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
31303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (isa<ObjCContainerDecl>(D)) {
31403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDI_current = &DIt;
31503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDE_current = DE;
31603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    } else {
31703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      FileDI_current = 0;
31803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    }
31903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
320ba98617b994864b7554ff75445983ad02a962f45Argyrios Kyrtzidis    if (Visit(MakeCXCursor(D, TU, Range), /*CheckedRegionOfInterest=*/true))
321dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
322dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
323dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
324dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  if (VisitedAtLeastOnce)
325dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    return;
326dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
327dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // No Decls overlapped with the range. Move up the lexical context until there
328dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // is a context that contains the range or we reach the translation unit
329dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  // level.
330dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  DeclContext *DC = DIt == Decls.begin() ? (*DIt)->getLexicalDeclContext()
331dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis                                         : (*(DIt-1))->getLexicalDeclContext();
332dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
333dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  while (DC && !DC->isTranslationUnit()) {
334dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    Decl *D = cast<Decl>(DC);
335dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    SourceRange CurDeclRange = D->getSourceRange();
336dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (CurDeclRange.isInvalid())
337dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
338dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
339dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    if (RangeCompare(SM, CurDeclRange, Range) == RangeOverlap) {
340ba98617b994864b7554ff75445983ad02a962f45Argyrios Kyrtzidis      Visit(MakeCXCursor(D, TU, Range), /*CheckedRegionOfInterest=*/true);
341dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis      break;
342dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    }
343dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
344dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    DC = D->getLexicalDeclContext();
345dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis  }
346dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis}
347dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis
3484c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntitiesInRegion() {
349b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis  if (!AU->getPreprocessor().getPreprocessingRecord())
350b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis    return false;
351b49e728a4d1a84b72f3aebf60ff494684f9cb004Argyrios Kyrtzidis
352788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  PreprocessingRecord &PPRec
353a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    = *AU->getPreprocessor().getPreprocessingRecord();
354f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  SourceManager &SM = AU->getSourceManager();
355788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
35692ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (RegionOfInterest.isValid()) {
357ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange MappedRange = AU->mapRangeToPreamble(RegionOfInterest);
358f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    SourceLocation B = MappedRange.getBegin();
359f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    SourceLocation E = MappedRange.getEnd();
360f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
361f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (AU->isInPreambleFileID(B)) {
362f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      if (SM.isLoadedSourceLocation(E))
363f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis        return visitPreprocessedEntitiesInRange(SourceRange(B, E),
364f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                                 PPRec, *this);
365f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
366f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // Beginning of range lies in the preamble but it also extends beyond
367f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // it into the main file. Split the range into 2 parts, one covering
368f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // the preamble and another covering the main file. This allows subsequent
369f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // calls to visitPreprocessedEntitiesInRange to accept a source range that
370f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // lies in the same FileID, allowing it to skip preprocessed entities that
371f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      // do not come from the same FileID.
372f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      bool breaked =
373f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis        visitPreprocessedEntitiesInRange(
374f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                   SourceRange(B, AU->getEndOfPreambleFileID()),
375f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                          PPRec, *this);
376f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      if (breaked) return true;
377f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      return visitPreprocessedEntitiesInRange(
378f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                    SourceRange(AU->getStartOfMainFileID(), E),
379f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                        PPRec, *this);
380f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    }
381f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
382f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    return visitPreprocessedEntitiesInRange(SourceRange(B, E), PPRec, *this);
38392ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  }
38492ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis
385788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  bool OnlyLocalDecls
38632038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor    = !AU->isMainFileAST() && AU->getOnlyLocalDecls();
38732038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor
38892ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (OnlyLocalDecls)
389f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    return visitPreprocessedEntities(PPRec.local_begin(), PPRec.local_end(),
390f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                     PPRec);
3914c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
392f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  return visitPreprocessedEntities(PPRec.begin(), PPRec.end(), PPRec);
3934c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor}
3944c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
3954c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregortemplate<typename InputIterator>
3964c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntities(InputIterator First,
397f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              InputIterator Last,
398f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              PreprocessingRecord &PPRec,
399f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                              FileID FID) {
4004c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  for (; First != Last; ++First) {
401f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (!FID.isInvalid() && !PPRec.isEntityInFileID(First, FID))
402f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis      continue;
403f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis
404f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    PreprocessedEntity *PPE = *First;
405f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (MacroExpansion *ME = dyn_cast<MacroExpansion>(PPE)) {
4064c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroExpansionCursor(ME, TU)))
4074c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4084c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4094c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4104c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4114c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
412f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (MacroDefinition *MD = dyn_cast<MacroDefinition>(PPE)) {
4134c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroDefinitionCursor(MD, TU)))
4144c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
41589d9980bbc2e4a4ac86673e6ec16fb9f5babb63bDouglas Gregor
4164c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4174c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4184c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
419f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis    if (InclusionDirective *ID = dyn_cast<InclusionDirective>(PPE)) {
4204c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeInclusionDirectiveCursor(ID, TU)))
4214c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4224c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4234c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
424788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor    }
425788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  }
426788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
4274c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  return false;
428788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor}
429788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
430b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \brief Visit the children of the given cursor.
431a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek///
432b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \returns true if the visitation should be aborted, false if it
433b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// should continue.
434f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekbool CursorVisitor::VisitChildren(CXCursor Cursor) {
435c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (clang_isReference(Cursor.kind) &&
436c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      Cursor.kind != CXCursor_CXXBaseSpecifier) {
437a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    // By definition, references have no children.
438a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return false;
439a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  }
440f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
441f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Set the Parent field to Cursor, then back to its old value once we're
442b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // done.
4430f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  SetParentRAII SetParent(Parent, StmtParent, Cursor);
444f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
445b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isDeclaration(Cursor.kind)) {
446b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Decl *D = getCursorDecl(Cursor);
44706d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (!D)
44806d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return false;
44906d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
450539311e0221df256c70c1c3080c8af847cd29dffTed Kremenek    return VisitAttributes(D) || Visit(D);
451b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
452f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
45306d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isStatement(Cursor.kind)) {
45406d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Stmt *S = getCursorStmt(Cursor))
45506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(S);
45606d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
45706d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
45806d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
45906d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
46006d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isExpression(Cursor.kind)) {
46106d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Expr *E = getCursorExpr(Cursor))
46206d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(E);
46306d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
46406d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
46506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
466f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
467b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isTranslationUnit(Cursor.kind)) {
468a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    CXTranslationUnit tu = getCursorTU(Cursor);
469a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    ASTUnit *CXXUnit = static_cast<ASTUnit*>(tu->TUData);
47004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
47104a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    int VisitOrder[2] = { VisitPreprocessorLast, !VisitPreprocessorLast };
47204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    for (unsigned I = 0; I != 2; ++I) {
47304a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      if (VisitOrder[I]) {
47404a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        if (!CXXUnit->isMainFileAST() && CXXUnit->getOnlyLocalDecls() &&
47504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor            RegionOfInterest.isInvalid()) {
47604a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          for (ASTUnit::top_level_iterator TL = CXXUnit->top_level_begin(),
47704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                        TLEnd = CXXUnit->top_level_end();
47804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor               TL != TLEnd; ++TL) {
479aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis            if (Visit(MakeCXCursor(*TL, tu, RegionOfInterest), true))
48004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor              return true;
48104a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          }
48204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        } else if (VisitDeclContext(
48304a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                CXXUnit->getASTContext().getTranslationUnitDecl()))
4847b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor          return true;
48504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        continue;
4867b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor      }
4873178cb674ac8c3b59e1791e14d38d48619a1b621Bob Wilson
48804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      // Walk the preprocessing record.
4894c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (CXXUnit->getPreprocessor().getPreprocessingRecord())
4904c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        visitPreprocessedEntitiesInRegion();
4910396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    }
49204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
4937b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor    return false;
494b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
495f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
496c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
497c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    if (CXXBaseSpecifier *Base = getCursorCXXBaseSpecifier(Cursor)) {
498c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      if (TypeSourceInfo *BaseTSInfo = Base->getTypeSourceInfo()) {
499c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor        return Visit(BaseTSInfo->getTypeLoc());
500c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      }
501c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    }
502c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  }
503221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
504221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
505221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    IBOutletCollectionAttr *A =
506221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(Cursor));
507221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    if (const ObjCInterfaceType *InterT = A->getInterface()->getAs<ObjCInterfaceType>())
508221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      return Visit(cxcursor::MakeCursorObjCClassRef(InterT->getInterface(),
509221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis                                                    A->getInterfaceLoc(), TU));
510221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  }
511221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
512b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // Nothing to visit at the moment.
513b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
514dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
515dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5161ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenekbool CursorVisitor::VisitBlockDecl(BlockDecl *B) {
51713c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor  if (TypeSourceInfo *TSInfo = B->getSignatureAsWritten())
51813c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor    if (Visit(TSInfo->getTypeLoc()))
51913c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor        return true;
5201ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
521664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  if (Stmt *Body = B->getBody())
522aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Body, StmtParent, TU, RegionOfInterest));
523664cffd330611d78fc0286f539589920a37ca328Ted Kremenek
524664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  return false;
5251ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek}
5261ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
527d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekllvm::Optional<bool> CursorVisitor::shouldVisitCursor(CXCursor Cursor) {
528d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (RegionOfInterest.isValid()) {
5296653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange Range = getFullCursorExtent(Cursor, AU->getSourceManager());
530d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Range.isInvalid())
531d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
5326653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
533d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    switch (CompareRegionOfInterest(Range)) {
534d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeBefore:
535d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes before the region of interest; skip it.
536d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
53723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
538d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeAfter:
539d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes after the region of interest; we're done.
540d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
541d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar
542d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeOverlap:
543d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration overlaps the region of interest; visit it.
544d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      break;
545d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
546d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
547d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return true;
548d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
549f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
550d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekbool CursorVisitor::VisitDeclContext(DeclContext *DC) {
551d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  DeclContext::decl_iterator I = DC->decls_begin(), E = DC->decls_end();
552f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
553d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually remove.  This part of a hack to support proper
554d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // iteration over all Decls contained lexically within an ObjC container.
555d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator*> DI_saved(DI_current, &I);
556d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator> DE_saved(DE_current, E);
557f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
558d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for ( ; I != E; ++I) {
559d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *D = *I;
560d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (D->getLexicalDeclContext() != DC)
561d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
562aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    CXCursor Cursor = MakeCXCursor(D, TU, RegionOfInterest);
5631836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis
5641836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis    // FIXME: ObjCClassRef/ObjCProtocolRef for forward class/protocol
5651836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis    // declarations is a mismatch with the compiler semantics.
5661836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis    if (Cursor.kind == CXCursor_ObjCInterfaceDecl) {
5671836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis      ObjCInterfaceDecl *ID = cast<ObjCInterfaceDecl>(D);
5681836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis      if (!ID->isThisDeclarationADefinition())
5691836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis        Cursor = MakeCursorObjCClassRef(ID, ID->getLocation(), TU);
5701836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis
5711836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis    } else if (Cursor.kind == CXCursor_ObjCProtocolDecl) {
5721836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis      ObjCProtocolDecl *PD = cast<ObjCProtocolDecl>(D);
5731836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis      if (!PD->isThisDeclarationADefinition())
5741836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis        Cursor = MakeCursorObjCProtocolRef(PD, PD->getLocation(), TU);
5751836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis    }
5761836db0f2c7527ac11af0044ab89150f9aaf22e1Argyrios Kyrtzidis
577d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
578d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
579d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
580d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
581d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
582d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar    if (Visit(Cursor, true))
583b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
584b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
585b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
586dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
587dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5881ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTranslationUnitDecl(TranslationUnitDecl *D) {
5891ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  llvm_unreachable("Translation units are visited directly by Visit()");
5901ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
5911ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
592162e1c1b487352434552147967c3dd296ebee2f7Richard Smithbool CursorVisitor::VisitTypeAliasDecl(TypeAliasDecl *D) {
593162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
594162e1c1b487352434552147967c3dd296ebee2f7Richard Smith    return Visit(TSInfo->getTypeLoc());
595162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
596162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return false;
597162e1c1b487352434552147967c3dd296ebee2f7Richard Smith}
598162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
5991ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTypedefDecl(TypedefDecl *D) {
6001ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
6011ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(TSInfo->getTypeLoc());
602f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
6031ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
6041ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6051ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6061ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTagDecl(TagDecl *D) {
6071ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitDeclContext(D);
6081ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6091ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6100ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregorbool CursorVisitor::VisitClassTemplateSpecializationDecl(
6110ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor                                          ClassTemplateSpecializationDecl *D) {
6120ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  bool ShouldVisitBody = false;
6130ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  switch (D->getSpecializationKind()) {
6140ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_Undeclared:
6150ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ImplicitInstantiation:
6160ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    // Nothing to visit
6170ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return false;
6180ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6190ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDeclaration:
6200ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDefinition:
6210ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6220ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6230ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitSpecialization:
6240ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    ShouldVisitBody = true;
6250ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6260ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6270ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6280ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  // Visit the template arguments used in the specialization.
6290ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (TypeSourceInfo *SpecType = D->getTypeAsWritten()) {
6300ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    TypeLoc TL = SpecType->getTypeLoc();
6310ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    if (TemplateSpecializationTypeLoc *TSTLoc
6320ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          = dyn_cast<TemplateSpecializationTypeLoc>(&TL)) {
6330ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor      for (unsigned I = 0, N = TSTLoc->getNumArgs(); I != N; ++I)
6340ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor        if (VisitTemplateArgumentLoc(TSTLoc->getArgLoc(I)))
6350ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          return true;
6360ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    }
6370ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6380ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6390ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (ShouldVisitBody && VisitCXXRecordDecl(D))
6400ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return true;
6410ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6420ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  return false;
6430ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor}
6440ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
64574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregorbool CursorVisitor::VisitClassTemplatePartialSpecializationDecl(
64674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor                                   ClassTemplatePartialSpecializationDecl *D) {
64774dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
64874dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // before visiting these template parameters.
64974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
65074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return true;
65174dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
65274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // Visit the partial specialization arguments.
65374dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  const TemplateArgumentLoc *TemplateArgs = D->getTemplateArgsAsWritten();
65474dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  for (unsigned I = 0, N = D->getNumTemplateArgsAsWritten(); I != N; ++I)
65574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    if (VisitTemplateArgumentLoc(TemplateArgs[I]))
65674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor      return true;
65774dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
65874dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  return VisitCXXRecordDecl(D);
65974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor}
66074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
661fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmDecl(TemplateTypeParmDecl *D) {
66284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  // Visit the default argument.
66384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
66484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (TypeSourceInfo *DefArg = D->getDefaultArgumentInfo())
66584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      if (Visit(DefArg->getTypeLoc()))
66684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor        return true;
66784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
668fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
669fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
670fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
6711ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitEnumConstantDecl(EnumConstantDecl *D) {
6721ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInitExpr())
673aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Init, StmtParent, TU, RegionOfInterest));
6741ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
6751ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6761ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6777d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitDeclaratorDecl(DeclaratorDecl *DD) {
6787d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  if (TypeSourceInfo *TSInfo = DD->getTypeSourceInfo())
6797d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
6807d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      return true;
6817d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
682c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
683c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = DD->getQualifierLoc())
684c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
685c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
686c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
6877d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  return false;
6887d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
6897d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
690a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor/// \brief Compare two base or member initializers based on their source order.
691cbb67480094b3bcb5b715acd827cbad55e2a204cSean Huntstatic int CompareCXXCtorInitializers(const void* Xp, const void *Yp) {
692cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *X
693cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Xp);
694cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *Y
695cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Yp);
696a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
697a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  if ((*X)->getSourceOrder() < (*Y)->getSourceOrder())
698a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return -1;
699a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else if ((*X)->getSourceOrder() > (*Y)->getSourceOrder())
700a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 1;
701a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else
702a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 0;
703a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor}
704a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
705b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitFunctionDecl(FunctionDecl *ND) {
70601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (TypeSourceInfo *TSInfo = ND->getTypeSourceInfo()) {
70701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function declaration's syntactic components in the order
70801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // written. This requires a bit of work.
709723df245307a530da5433dfb43accf187dc3e243Abramo Bagnara    TypeLoc TL = TSInfo->getTypeLoc().IgnoreParens();
71001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    FunctionTypeLoc *FTL = dyn_cast<FunctionTypeLoc>(&TL);
71101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
71201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // If we have a function declared directly (without the use of a typedef),
71301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // visit just the return type. Otherwise, just visit the function's type
71401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // now.
71501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if ((FTL && !isa<CXXConversionDecl>(ND) && Visit(FTL->getResultLoc())) ||
71601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor        (!FTL && Visit(TL)))
71701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
71801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
719c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // Visit the nested-name-specifier, if present.
720c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (NestedNameSpecifierLoc QualifierLoc = ND->getQualifierLoc())
721c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      if (VisitNestedNameSpecifierLoc(QualifierLoc))
722c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor        return true;
72301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the declaration name.
72501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (VisitDeclarationNameInfo(ND->getNameInfo()))
72601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
72701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Visit explicitly-specified template arguments!
72901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function parameters, if we have a function type.
73101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (FTL && VisitFunctionTypeLoc(*FTL, true))
73201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
73301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Attributes?
73501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
73601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73710620eb5164e31208fcbf0437cd79ae535ed0559Sean Hunt  if (ND->doesThisDeclarationHaveABody() && !ND->isLateTemplateParsed()) {
738a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(ND)) {
739a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Find the initializers that were written in the source.
7405f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner      SmallVector<CXXCtorInitializer *, 4> WrittenInits;
741a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (CXXConstructorDecl::init_iterator I = Constructor->init_begin(),
742a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                          IEnd = Constructor->init_end();
743a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor           I != IEnd; ++I) {
744a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (!(*I)->isWritten())
745a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor          continue;
746a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
747a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        WrittenInits.push_back(*I);
748a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
749a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
750a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Sort the initializers in source order
751a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      llvm::array_pod_sort(WrittenInits.begin(), WrittenInits.end(),
752cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt                           &CompareCXXCtorInitializers);
753a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
754a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Visit the initializers in source order
755a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (unsigned I = 0, N = WrittenInits.size(); I != N; ++I) {
756cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt        CXXCtorInitializer *Init = WrittenInits[I];
75700eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet        if (Init->isAnyMemberInitializer()) {
75800eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet          if (Visit(MakeCursorMemberRef(Init->getAnyMember(),
759a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                        Init->getMemberLocation(), TU)))
760a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
76176852c218a207ef43583515cb835b6e855353a0fDouglas Gregor        } else if (TypeSourceInfo *TInfo = Init->getTypeSourceInfo()) {
76276852c218a207ef43583515cb835b6e855353a0fDouglas Gregor          if (Visit(TInfo->getTypeLoc()))
763a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
764a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        }
765a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
766a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        // Visit the initializer value.
767a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (Expr *Initializer = Init->getInit())
768aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis          if (Visit(MakeCXCursor(Initializer, ND, TU, RegionOfInterest)))
769a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
770a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
771a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
772a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
773aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(ND->getBody(), StmtParent, TU, RegionOfInterest)))
774a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return true;
775a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  }
776f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
777b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
778b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
779dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
7801ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitFieldDecl(FieldDecl *D) {
7811ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7821ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
783f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7841ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *BitWidth = D->getBitWidth())
785aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(BitWidth, StmtParent, TU, RegionOfInterest));
786f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7871ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7881ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7891ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
7901ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitVarDecl(VarDecl *D) {
7911ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7921ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
793f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7941ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInit())
795aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Init, StmtParent, TU, RegionOfInterest));
796f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7971ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7981ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7991ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
80084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitNonTypeTemplateParmDecl(NonTypeTemplateParmDecl *D) {
80184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitDeclaratorDecl(D))
80284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
80384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
80484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
80584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (Expr *DefArg = D->getDefaultArgument())
806aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(DefArg, StmtParent, TU, RegionOfInterest));
80784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
80884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
80984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
81084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
811fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitFunctionTemplateDecl(FunctionTemplateDecl *D) {
812fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the FunctionDecl
813fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // before visiting these template parameters.
814fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
815fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return true;
816fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
817fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return VisitFunctionDecl(D->getTemplatedDecl());
818fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
819fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
82039d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregorbool CursorVisitor::VisitClassTemplateDecl(ClassTemplateDecl *D) {
82139d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
82239d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // before visiting these template parameters.
82339d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
82439d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return true;
82539d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82639d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  return VisitCXXRecordDecl(D->getTemplatedDecl());
82739d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor}
82839d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitTemplateTemplateParmDecl(TemplateTemplateParmDecl *D) {
83084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
83184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
83284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
83384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited() &&
83484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      VisitTemplateArgumentLoc(D->getDefaultArgument()))
83584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
83684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
83784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
83884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
83984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
8401ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCMethodDecl(ObjCMethodDecl *ND) {
8414bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  if (TypeSourceInfo *TSInfo = ND->getResultTypeSourceInfo())
8424bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
8434bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor      return true;
8444bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor
845f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  for (ObjCMethodDecl::param_iterator P = ND->param_begin(),
8461ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       PEnd = ND->param_end();
8471ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       P != PEnd; ++P) {
848aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(*P, TU, RegionOfInterest)))
8491ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
8501ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  }
851f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (ND->isThisDeclarationADefinition() &&
853aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      Visit(MakeCXCursor(ND->getBody(), StmtParent, TU, RegionOfInterest)))
8541ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
855f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8561ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
8571ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
8581ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
85903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidistemplate <typename DeclIt>
86003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidisstatic void addRangedDeclsInContainer(DeclIt *DI_current, DeclIt DE_current,
86103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                      SourceManager &SM, SourceLocation EndLoc,
86203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                      SmallVectorImpl<Decl *> &Decls) {
86303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  DeclIt next = *DI_current;
86403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  while (++next != DE_current) {
86503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    Decl *D_next = *next;
86603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (!D_next)
86703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      break;
86803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    SourceLocation L = D_next->getLocStart();
86903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (!L.isValid())
87003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      break;
87103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (SM.isBeforeInTranslationUnit(L, EndLoc)) {
87203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      *DI_current = next;
87303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      Decls.push_back(D_next);
87403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      continue;
87503ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    }
87603ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    break;
87703ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  }
87803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis}
87903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
880d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremeneknamespace {
881d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  struct ContainerDeclsSort {
882d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    SourceManager &SM;
883d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    ContainerDeclsSort(SourceManager &sm) : SM(sm) {}
884d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    bool operator()(Decl *A, Decl *B) {
885d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_A = A->getLocStart();
886d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_B = B->getLocStart();
887d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      assert(L_A.isValid() && L_B.isValid());
888d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return SM.isBeforeInTranslationUnit(L_A, L_B);
889d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
890d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  };
891d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
892d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
893a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregorbool CursorVisitor::VisitObjCContainerDecl(ObjCContainerDecl *D) {
894d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually convert back to just 'VisitDeclContext()'.  Essentially
895d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // an @implementation can lexically contain Decls that are not properly
896d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // nested in the AST.  When we identify such cases, we need to retrofit
897d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // this nesting here.
89803ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  if (!DI_current && !FileDI_current)
899d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
900d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
901d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Scan the Decls that immediately come after the container
902d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // in the current DeclContext.  If any fall within the
903d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // container's lexical region, stash them into a vector
904d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // for later processing.
9055f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<Decl *, 24> DeclsInContainer;
906d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SourceLocation EndLoc = D->getSourceRange().getEnd();
907a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  SourceManager &SM = AU->getSourceManager();
908d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (EndLoc.isValid()) {
90903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (DI_current) {
91003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      addRangedDeclsInContainer(DI_current, DE_current, SM, EndLoc,
91103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                DeclsInContainer);
91203ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    } else {
91303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      addRangedDeclsInContainer(FileDI_current, FileDE_current, SM, EndLoc,
91403ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis                                DeclsInContainer);
915d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
916d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
917d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
918d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // The common case.
919d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (DeclsInContainer.empty())
920d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
921d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
922d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Get all the Decls in the DeclContext, and sort them with the
923d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // additional ones we've collected.  Then visit them.
924d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for (DeclContext::decl_iterator I = D->decls_begin(), E = D->decls_end();
925d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek       I!=E; ++I) {
926d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *subDecl = *I;
9270582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek    if (!subDecl || subDecl->getLexicalDeclContext() != D ||
9280582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek        subDecl->getLocStart().isInvalid())
929d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
930d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    DeclsInContainer.push_back(subDecl);
931d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
932d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
933d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now sort the Decls so that they appear in lexical order.
934d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  std::sort(DeclsInContainer.begin(), DeclsInContainer.end(),
935d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek            ContainerDeclsSort(SM));
936d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
937d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now visit the decls.
9385f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  for (SmallVectorImpl<Decl*>::iterator I = DeclsInContainer.begin(),
939d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek         E = DeclsInContainer.end(); I != E; ++I) {
940aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    CXCursor Cursor = MakeCXCursor(*I, TU, RegionOfInterest);
941d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
942d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
943d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
944d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
945d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
946d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Visit(Cursor, true))
947d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return true;
948d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
949d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return false;
950a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor}
951a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor
952b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCCategoryDecl(ObjCCategoryDecl *ND) {
953b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (Visit(MakeCursorObjCClassRef(ND->getClassInterface(), ND->getLocation(),
954b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                   TU)))
955b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
956f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
95778db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCCategoryDecl::protocol_loc_iterator PL = ND->protocol_loc_begin();
95878db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCCategoryDecl::protocol_iterator I = ND->protocol_begin(),
95978db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = ND->protocol_end(); I != E; ++I, ++PL)
960b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
961b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
962f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
963a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(ND);
964dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
965dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
9661ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCProtocolDecl(ObjCProtocolDecl *PID) {
967bd9482d859a74bf2c45ef8b8aedec61c0e1c8374Douglas Gregor  if (!PID->isThisDeclarationADefinition())
968bd9482d859a74bf2c45ef8b8aedec61c0e1c8374Douglas Gregor    return Visit(MakeCursorObjCProtocolRef(PID, PID->getLocation(), TU));
969bd9482d859a74bf2c45ef8b8aedec61c0e1c8374Douglas Gregor
9701ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  ObjCProtocolDecl::protocol_loc_iterator PL = PID->protocol_loc_begin();
9711ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  for (ObjCProtocolDecl::protocol_iterator I = PID->protocol_begin(),
9721ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       E = PID->protocol_end(); I != E; ++I, ++PL)
9731ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
9741ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
975f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
9761ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(PID);
9771ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
9781ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
97923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenekbool CursorVisitor::VisitObjCPropertyDecl(ObjCPropertyDecl *PD) {
98083cb94269015bf2770ade71e616c5322ea7e76e1Douglas Gregor  if (PD->getTypeSourceInfo() && Visit(PD->getTypeSourceInfo()->getTypeLoc()))
981fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall    return true;
982fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall
98323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // FIXME: This implements a workaround with @property declarations also being
98423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // installed in the DeclContext for the @interface.  Eventually this code
98523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // should be removed.
98623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCCategoryDecl *CDecl = dyn_cast<ObjCCategoryDecl>(PD->getDeclContext());
98723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!CDecl || !CDecl->IsClassExtension())
98823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
98923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCInterfaceDecl *ID = CDecl->getClassInterface();
99123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!ID)
99223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
99323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  IdentifierInfo *PropertyId = PD->getIdentifier();
99523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCPropertyDecl *prevDecl =
99623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    ObjCPropertyDecl::findPropertyDecl(cast<DeclContext>(ID), PropertyId);
99723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!prevDecl)
99923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
100023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
100123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // Visit synthesized methods since they will be skipped when visiting
100223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // the @interface.
100323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getGetterMethodDecl())
1004a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
1005aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(MD, TU, RegionOfInterest)))
100623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
100723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
100823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getSetterMethodDecl())
1009a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
1010aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(MD, TU, RegionOfInterest)))
101123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
101223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
101323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  return false;
101423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek}
101523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
1016b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
1017375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor  if (!D->isThisDeclarationADefinition()) {
1018375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor    // Forward declaration is treated like a reference.
1019375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor    return Visit(MakeCursorObjCClassRef(D, D->getLocation(), TU));
1020375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor  }
1021375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor
1022dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek  // Issue callbacks for super class.
1023b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (D->getSuperClass() &&
1024b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1025f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
1026b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                        TU)))
1027b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
1028f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
102978db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCInterfaceDecl::protocol_loc_iterator PL = D->protocol_loc_begin();
103078db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCInterfaceDecl::protocol_iterator I = D->protocol_begin(),
103178db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = D->protocol_end(); I != E; ++I, ++PL)
1032b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
1033b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
1034f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
1035a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(D);
1036dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1037dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10381ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplDecl(ObjCImplDecl *D) {
10391ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(D);
10401ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10411ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10421ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) {
1043ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  // 'ID' could be null when dealing with invalid code.
1044ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  if (ObjCInterfaceDecl *ID = D->getClassInterface())
1045ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek    if (Visit(MakeCursorObjCClassRef(ID, D->getLocation(), TU)))
1046ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek      return true;
1047f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10481ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
10491ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10501ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10511ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplementationDecl(ObjCImplementationDecl *D) {
10521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#if 0
10531ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // Issue callbacks for super class.
10541ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // FIXME: No source location information!
10551ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (D->getSuperClass() &&
10561ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1057f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
10581ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor                                        TU)))
1059a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return true;
10601ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#endif
1061f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10621ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
1063dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1064dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
1065a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregorbool CursorVisitor::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *PD) {
1066a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  if (ObjCIvarDecl *Ivar = PD->getPropertyIvarDecl())
1067a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor    return Visit(MakeCursorMemberRef(Ivar, PD->getPropertyIvarDeclLoc(), TU));
1068a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
1069a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  return false;
1070a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor}
1071a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
10728f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenekbool CursorVisitor::VisitNamespaceDecl(NamespaceDecl *D) {
10738f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  return VisitDeclContext(D);
10748f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek}
10758f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek
10766931900f43cea558c6974075256c07728dbfecc6Douglas Gregorbool CursorVisitor::VisitNamespaceAliasDecl(NamespaceAliasDecl *D) {
1077c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
10780cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
10790cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1080c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
10816931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10826931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  return Visit(MakeCursorNamespaceRef(D->getAliasedNamespace(),
10836931900f43cea558c6974075256c07728dbfecc6Douglas Gregor                                      D->getTargetNameLoc(), TU));
10846931900f43cea558c6974075256c07728dbfecc6Douglas Gregor}
10856931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10867e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUsingDecl(UsingDecl *D) {
1087c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1088dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1089dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1090c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1091dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
10927e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
10931f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (Visit(MakeCursorOverloadedDeclRef(D, D->getLocation(), TU)))
10941f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return true;
10951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
10967e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
10977e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
10987e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
10990a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregorbool CursorVisitor::VisitUsingDirectiveDecl(UsingDirectiveDecl *D) {
1100c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1101db9924191092b4d426cc066637d81698211846aaDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1102db9924191092b4d426cc066637d81698211846aaDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1103c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
11040a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11050a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  return Visit(MakeCursorNamespaceRef(D->getNominatedNamespaceAsWritten(),
11060a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor                                      D->getIdentLocation(), TU));
11070a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor}
11080a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11097e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingValueDecl(UnresolvedUsingValueDecl *D) {
1110c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1111dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1112dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1113c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1114dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1115c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11167e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
11177e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11187e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
11197e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypenameDecl(
11207e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor                                               UnresolvedUsingTypenameDecl *D) {
1121c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1122dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1123dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1124c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1125c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11267e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return false;
11277e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11287e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
112901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitDeclarationNameInfo(DeclarationNameInfo Name) {
113001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  switch (Name.getName().getNameKind()) {
113101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::Identifier:
113201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXLiteralOperatorName:
113301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXOperatorName:
113401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXUsingDirective:
113501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
113601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
113701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConstructorName:
113801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXDestructorName:
113901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConversionFunctionName:
114001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (TypeSourceInfo *TSInfo = Name.getNamedTypeInfo())
114101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return Visit(TSInfo->getTypeLoc());
114201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
114301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
114401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCZeroArgSelector:
114501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCOneArgSelector:
114601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCMultiArgSelector:
114701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Per-identifier location info?
114801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
114901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
11507530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie
11517530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie  llvm_unreachable("Invalid DeclarationName::Kind!");
115201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
115301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1154c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregorbool CursorVisitor::VisitNestedNameSpecifier(NestedNameSpecifier *NNS,
1155c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                             SourceRange Range) {
1156c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // FIXME: This whole routine is a hack to work around the lack of proper
1157c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // source information in nested-name-specifiers (PR5791). Since we do have
1158c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // a beginning source location, we can visit the first component of the
1159c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // nested-name-specifier, if it's a single-token component.
1160c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  if (!NNS)
1161c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return false;
1162c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1163c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Get the first component in the nested-name-specifier.
1164c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  while (NestedNameSpecifier *Prefix = NNS->getPrefix())
1165c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    NNS = Prefix;
1166c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1167c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  switch (NNS->getKind()) {
1168c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Namespace:
1169c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(), Range.getBegin(),
1170c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                        TU));
1171c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
117214aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor  case NestedNameSpecifier::NamespaceAlias:
117314aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
117414aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor                                        Range.getBegin(), TU));
117514aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor
1176c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpec: {
1177c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // If the type has a form where we know that the beginning of the source
1178c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // range matches up with a reference cursor. Visit the appropriate reference
1179c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // cursor.
1180f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall    const Type *T = NNS->getAsType();
1181c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TypedefType *Typedef = dyn_cast<TypedefType>(T))
1182c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Typedef->getDecl(), Range.getBegin(), TU));
1183c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TagType *Tag = dyn_cast<TagType>(T))
1184c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Tag->getDecl(), Range.getBegin(), TU));
1185c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TemplateSpecializationType *TST
1186c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                      = dyn_cast<TemplateSpecializationType>(T))
1187c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return VisitTemplateName(TST->getTemplateName(), Range.getBegin());
1188c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1189c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1190c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1191c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpecWithTemplate:
1192c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Global:
1193c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Identifier:
1194c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1195c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1196c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1197c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  return false;
1198c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor}
1199c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1200dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregorbool
1201dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas GregorCursorVisitor::VisitNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier) {
12025f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<NestedNameSpecifierLoc, 4> Qualifiers;
1203dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  for (; Qualifier; Qualifier = Qualifier.getPrefix())
1204dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    Qualifiers.push_back(Qualifier);
1205dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1206dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  while (!Qualifiers.empty()) {
1207dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifierLoc Q = Qualifiers.pop_back_val();
1208dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifier *NNS = Q.getNestedNameSpecifier();
1209dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    switch (NNS->getKind()) {
1210dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Namespace:
1211dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(),
1212c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1213dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1214dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1215dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1216dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1217dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1218dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::NamespaceAlias:
1219dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
1220c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1221dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1222dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1223dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1224dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1225dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1226dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpec:
1227dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpecWithTemplate:
1228dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(Q.getTypeLoc()))
1229dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1230dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1231dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1232dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1233dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Global:
1234dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Identifier:
1235dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1236dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    }
1237dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1238dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1239dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  return false;
1240dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor}
1241dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1242fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateParameters(
1243fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          const TemplateParameterList *Params) {
1244fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (!Params)
1245fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1246fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1247fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (TemplateParameterList::const_iterator P = Params->begin(),
1248fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          PEnd = Params->end();
1249fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor       P != PEnd; ++P) {
1250aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (Visit(MakeCXCursor(*P, TU, RegionOfInterest)))
1251fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1252fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
1253fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1254fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1255fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1256fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
12570b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregorbool CursorVisitor::VisitTemplateName(TemplateName Name, SourceLocation Loc) {
12580b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  switch (Name.getKind()) {
12590b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::Template:
12600b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(Name.getAsTemplateDecl(), Loc, TU));
12610b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12620b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::OverloadedTemplate:
12631f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // Visit the overloaded template set.
12641f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (Visit(MakeCursorOverloadedDeclRef(Name, Loc, TU)))
12651f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return true;
12661f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
12670b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12680b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12690b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::DependentTemplate:
12700b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12710b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12720b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12730b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::QualifiedTemplate:
12740b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12750b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(
12760b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                  Name.getAsQualifiedTemplateName()->getDecl(),
12770b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                       Loc, TU));
1278146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall
1279146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall  case TemplateName::SubstTemplateTemplateParm:
1280146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall    return Visit(MakeCursorTemplateRef(
1281146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                         Name.getAsSubstTemplateTemplateParm()->getParameter(),
1282146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                                       Loc, TU));
12831aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor
12841aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor  case TemplateName::SubstTemplateTemplateParmPack:
12851aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor    return Visit(MakeCursorTemplateRef(
12861aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                  Name.getAsSubstTemplateTemplateParmPack()->getParameterPack(),
12871aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                                       Loc, TU));
12880b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  }
12897530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie
12907530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie  llvm_unreachable("Invalid TemplateName::Kind!");
12910b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor}
12920b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
1293fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateArgumentLoc(const TemplateArgumentLoc &TAL) {
1294fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  switch (TAL.getArgument().getKind()) {
1295fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Null:
1296fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Integral:
1297fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Pack:
1298fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
129987dd697dcc8ecb64df73ae64d61b8c80ff0c157cDouglas Gregor
1300fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Type:
1301fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (TypeSourceInfo *TSInfo = TAL.getTypeSourceInfo())
1302fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return Visit(TSInfo->getTypeLoc());
1303fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1304fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1305fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Declaration:
1306fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceDeclExpression())
1307aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(E, StmtParent, TU, RegionOfInterest));
1308fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1309fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1310fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Expression:
1311fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceExpression())
1312aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return Visit(MakeCXCursor(E, StmtParent, TU, RegionOfInterest));
1313fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1314fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1315fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Template:
1316a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor  case TemplateArgument::TemplateExpansion:
1317b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor    if (VisitNestedNameSpecifierLoc(TAL.getTemplateQualifierLoc()))
1318b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor      return true;
1319b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor
1320a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor    return VisitTemplateName(TAL.getArgument().getAsTemplateOrTemplatePattern(),
13210b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                             TAL.getTemplateNameLoc());
1322fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
13237530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie
13247530c034c0c71a64c5a9173206d9742ae847af8bDavid Blaikie  llvm_unreachable("Invalid TemplateArgument::Kind!");
1325fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1326fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1327a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenekbool CursorVisitor::VisitLinkageSpecDecl(LinkageSpecDecl *D) {
1328a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  return VisitDeclContext(D);
1329a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek}
1330a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek
133101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitQualifiedTypeLoc(QualifiedTypeLoc TL) {
133201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  return Visit(TL.getUnqualifiedLoc());
133301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
133401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1335f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBuiltinTypeLoc(BuiltinTypeLoc TL) {
1336a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTContext &Context = AU->getASTContext();
1337f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1338f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // Some builtin types (such as Objective-C's "id", "sel", and
1339f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // "Class") have associated declarations. Create cursors for those.
1340f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  QualType VisitType;
1341e0a22d06888c13989b3f72db319f1d498bf69153John McCall  switch (TL.getTypePtr()->getKind()) {
13422dde35bc626153492f5f58202506c88a27fbff5bJohn McCall
13436b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Void:
1344f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::NullPtr:
13456b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Dependent:
13462dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define BUILTIN_TYPE(Id, SingletonId)
13472dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define SIGNED_TYPE(Id, SingletonId) case BuiltinType::Id:
13482dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define UNSIGNED_TYPE(Id, SingletonId) case BuiltinType::Id:
13492dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define FLOATING_TYPE(Id, SingletonId) case BuiltinType::Id:
13502dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#define PLACEHOLDER_TYPE(Id, SingletonId) case BuiltinType::Id:
13512dde35bc626153492f5f58202506c88a27fbff5bJohn McCall#include "clang/AST/BuiltinTypes.def"
1352f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13536b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1354f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCId:
1355f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCIdType();
1356f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13576b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
13586b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::ObjCClass:
13596b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    VisitType = Context.getObjCClassType();
13606b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    break;
13616b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1362f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCSel:
1363f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCSelType();
1364f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
1365f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1366f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1367f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (!VisitType.isNull()) {
1368f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (const TypedefType *Typedef = VisitType->getAs<TypedefType>())
1369f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek      return Visit(MakeCursorTypeRef(Typedef->getDecl(), TL.getBuiltinLoc(),
1370f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                     TU));
1371f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1372f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1373f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1374f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1375f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
13767d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitTypedefTypeLoc(TypedefTypeLoc TL) {
1377162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return Visit(MakeCursorTypeRef(TL.getTypedefNameDecl(), TL.getNameLoc(), TU));
13787d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
13797d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
1380f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypeLoc(UnresolvedUsingTypeLoc TL) {
1381f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1382f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1383f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1384f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitTagTypeLoc(TagTypeLoc TL) {
13856f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis  if (TL.isDefinition())
1386aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(TL.getDecl(), TU, RegionOfInterest));
13876f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis
1388f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1389f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1390f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1391fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc TL) {
1392960d13dde337a59dacc9dc3936c26d4aa8478986Chandler Carruth  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1393fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1394fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1395f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc TL) {
1396f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(MakeCursorObjCClassRef(TL.getIFaceDecl(), TL.getNameLoc(), TU)))
1397f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1398f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1399c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return false;
1400c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall}
1401c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1402c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCallbool CursorVisitor::VisitObjCObjectTypeLoc(ObjCObjectTypeLoc TL) {
1403c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  if (TL.hasBaseTypeAsWritten() && Visit(TL.getBaseLoc()))
1404c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall    return true;
1405c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1406f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumProtocols(); I != N; ++I) {
1407f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (Visit(MakeCursorObjCProtocolRef(TL.getProtocol(I), TL.getProtocolLoc(I),
1408f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                        TU)))
1409f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor      return true;
1410f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1411f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1412f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1413f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1414f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1415f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCObjectPointerTypeLoc(ObjCObjectPointerTypeLoc TL) {
1416c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return Visit(TL.getPointeeLoc());
1417f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1418f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1419075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnarabool CursorVisitor::VisitParenTypeLoc(ParenTypeLoc TL) {
1420075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara  return Visit(TL.getInnerLoc());
1421075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara}
1422075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara
1423f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitPointerTypeLoc(PointerTypeLoc TL) {
1424f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1425f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1426f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1427f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBlockPointerTypeLoc(BlockPointerTypeLoc TL) {
1428f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1429f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1430f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1431f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitMemberPointerTypeLoc(MemberPointerTypeLoc TL) {
1432f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1433f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1434f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1435f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitLValueReferenceTypeLoc(LValueReferenceTypeLoc TL) {
1436f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1437f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1438f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1439f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitRValueReferenceTypeLoc(RValueReferenceTypeLoc TL) {
1440f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1441f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1442f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
14433422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidisbool CursorVisitor::VisitAttributedTypeLoc(AttributedTypeLoc TL) {
14443422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis  return Visit(TL.getModifiedLoc());
14453422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis}
14463422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis
144701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitFunctionTypeLoc(FunctionTypeLoc TL,
144801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor                                         bool SkipResultType) {
144901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (!SkipResultType && Visit(TL.getResultLoc()))
1450f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1451f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1452f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
14535dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek    if (Decl *D = TL.getArg(I))
1454aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      if (Visit(MakeCXCursor(D, TU, RegionOfInterest)))
14555dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek        return true;
1456f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1457f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1458f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1459f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1460f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitArrayTypeLoc(ArrayTypeLoc TL) {
1461f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(TL.getElementLoc()))
1462f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1463f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1464f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Expr *Size = TL.getSizeExpr())
1465aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    return Visit(MakeCXCursor(Size, StmtParent, TU, RegionOfInterest));
1466f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1467f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1468f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1469f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1470fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateSpecializationTypeLoc(
1471fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                             TemplateSpecializationTypeLoc TL) {
14720b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  // Visit the template name.
14730b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  if (VisitTemplateName(TL.getTypePtr()->getTemplateName(),
14740b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                        TL.getTemplateNameLoc()))
14750b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return true;
1476fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1477fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // Visit the template arguments.
1478fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
1479fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
1480fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1481fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1482fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1483fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1484fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
14852332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfExprTypeLoc(TypeOfExprTypeLoc TL) {
14862332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return Visit(MakeCXCursor(TL.getUnderlyingExpr(), StmtParent, TU));
14872332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
14882332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
14892332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfTypeLoc(TypeOfTypeLoc TL) {
14902332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
1491ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt    return Visit(TSInfo->getTypeLoc());
1492ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1493ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  return false;
1494ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt}
1495ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1496ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Huntbool CursorVisitor::VisitUnaryTransformTypeLoc(UnaryTransformTypeLoc TL) {
1497ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
14982332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor    return Visit(TSInfo->getTypeLoc());
14992332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15002332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return false;
15012332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
15022332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15032494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregorbool CursorVisitor::VisitDependentNameTypeLoc(DependentNameTypeLoc TL) {
15042494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15052494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    return true;
15062494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
15072494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  return false;
15082494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor}
15092494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
151094fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregorbool CursorVisitor::VisitDependentTemplateSpecializationTypeLoc(
151194fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor                                    DependentTemplateSpecializationTypeLoc TL) {
151294fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the nested-name-specifier, if there is one.
151394fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  if (TL.getQualifierLoc() &&
151494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
151594fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    return true;
151694fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
151794fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the template arguments.
151894fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
151994fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
152094fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      return true;
152194fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
152294fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  return false;
152394fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor}
152494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
15259e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregorbool CursorVisitor::VisitElaboratedTypeLoc(ElaboratedTypeLoc TL) {
15269e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15279e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor    return true;
15289e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15299e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  return Visit(TL.getNamedTypeLoc());
15309e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor}
15319e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15327536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregorbool CursorVisitor::VisitPackExpansionTypeLoc(PackExpansionTypeLoc TL) {
15337536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor  return Visit(TL.getPatternLoc());
15347536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor}
15357536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor
1536427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitDecltypeTypeLoc(DecltypeTypeLoc TL) {
1537427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  if (Expr *E = TL.getUnderlyingExpr())
1538427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis    return Visit(MakeCXCursor(E, StmtParent, TU));
1539427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1540427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return false;
1541427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1542427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1543427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
1544427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1545427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1546427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1547b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedmanbool CursorVisitor::VisitAtomicTypeLoc(AtomicTypeLoc TL) {
1548b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman  return Visit(TL.getValueLoc());
1549b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman}
1550b001de7458d17c17e6d8b8034c7cfcefd3b70c00Eli Friedman
1551427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#define DEFAULT_TYPELOC_IMPL(CLASS, PARENT) \
1552427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::Visit##CLASS##TypeLoc(CLASS##TypeLoc TL) { \
1553427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit##PARENT##Loc(TL); \
1554427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1555427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1556427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Complex, Type)
1557427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ConstantArray, ArrayType)
1558427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(IncompleteArray, ArrayType)
1559427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(VariableArray, ArrayType)
1560427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedArray, ArrayType)
1561427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedExtVector, Type)
1562427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Vector, Type)
1563427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ExtVector, VectorType)
1564427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionProto, FunctionType)
1565427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionNoProto, FunctionType)
1566427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Record, TagType)
1567427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Enum, TagType)
1568427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParm, Type)
1569427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParmPack, Type)
1570427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Auto, Type)
1571427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
15723064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenekbool CursorVisitor::VisitCXXRecordDecl(CXXRecordDecl *D) {
1573c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
1574c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1575c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1576c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
1577c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
15785e1cdac63c3d9c9b32fa41fa0b2d242a58a20d49John McCall  if (D->isCompleteDefinition()) {
15793064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    for (CXXRecordDecl::base_class_iterator I = D->bases_begin(),
15803064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek         E = D->bases_end(); I != E; ++I) {
15813064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      if (Visit(cxcursor::MakeCursorCXXBaseSpecifier(I, TU)))
15823064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        return true;
15833064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
15843064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  }
15853064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
15863064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  return VisitTagDecl(D);
15873064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek}
15883064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
158909dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenekbool CursorVisitor::VisitAttributes(Decl *D) {
1590cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt  for (AttrVec::const_iterator i = D->attr_begin(), e = D->attr_end();
1591cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt       i != e; ++i)
1592cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt    if (Visit(MakeCXCursor(*i, D, TU)))
159309dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek        return true;
159409dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
159509dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek  return false;
159609dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek}
159709dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
1598c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1599c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Data-recursive visitor methods.
1600c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1601c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
160228a719433411ef782b582946823bc648ddcc4533Ted Kremeneknamespace {
1603035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#define DEF_JOB(NAME, DATA, KIND)\
1604035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass NAME : public VisitorJob {\
1605035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:\
1606035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  NAME(DATA *d, CXCursor parent) : VisitorJob(parent, VisitorJob::KIND, d) {} \
1607035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) { return VJ->getKind() == KIND; }\
1608f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DATA *get() const { return static_cast<DATA*>(data[0]); }\
1609035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1610035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1611035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(StmtVisit, Stmt, StmtVisitKind)
1612035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(MemberExprParts, MemberExpr, MemberExprPartsKind)
1613e4979ccb5960608edce73f3b274eb7c2de15dac5Ted KremenekDEF_JOB(DeclRefExprParts, DeclRefExpr, DeclRefExprPartsKind)
1614035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(OverloadExprParts, OverloadExpr, OverloadExprPartsKind)
1615b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios KyrtzidisDEF_JOB(ExplicitTemplateArgsVisit, ASTTemplateArgumentListInfo,
161660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        ExplicitTemplateArgsVisitKind)
161794d96291cd041adc5731a2294828a9c20e450b74Douglas GregorDEF_JOB(SizeOfPackExprParts, SizeOfPackExpr, SizeOfPackExprPartsKind)
1618011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas GregorDEF_JOB(LambdaExprParts, LambdaExpr, LambdaExprPartsKind)
1619035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#undef DEF_JOB
1620035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1621035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass DeclVisit : public VisitorJob {
1622035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1623035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  DeclVisit(Decl *d, CXCursor parent, bool isFirst) :
1624035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::DeclVisitKind,
1625035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               d, isFirst ? (void*) 1 : (void*) 0) {}
1626035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
162782f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    return VJ->getKind() == DeclVisitKind;
1628035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1629f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  Decl *get() const { return static_cast<Decl*>(data[0]); }
1630f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  bool isFirst() const { return data[1] ? true : false; }
1631035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1632035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass TypeLocVisit : public VisitorJob {
1633035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1634035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  TypeLocVisit(TypeLoc tl, CXCursor parent) :
1635035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::TypeLocVisitKind,
1636035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               tl.getType().getAsOpaquePtr(), tl.getOpaqueData()) {}
1637035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1638035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1639035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return VJ->getKind() == TypeLocVisitKind;
1640035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1641035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
164282f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek  TypeLoc get() const {
1643f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    QualType T = QualType::getFromOpaquePtr(data[0]);
1644f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return TypeLoc(T, data[1]);
1645035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1646035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1647035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1648ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekclass LabelRefVisit : public VisitorJob {
1649ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekpublic:
1650ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelRefVisit(LabelDecl *LD, SourceLocation labelLoc, CXCursor parent)
1651ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner    : VisitorJob(parent, VisitorJob::LabelRefVisitKind, LD,
1652dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 labelLoc.getPtrEncoding()) {}
1653ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek
1654ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1655ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek    return VJ->getKind() == VisitorJob::LabelRefVisitKind;
1656ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  }
1657ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelDecl *get() const { return static_cast<LabelDecl*>(data[0]); }
1658ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  SourceLocation getLoc() const {
1659dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin    return SourceLocation::getFromPtrEncoding(data[1]); }
1660f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek};
1661f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1662f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorclass NestedNameSpecifierLocVisit : public VisitorJob {
1663f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorpublic:
1664f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLocVisit(NestedNameSpecifierLoc Qualifier, CXCursor parent)
1665f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    : VisitorJob(parent, VisitorJob::NestedNameSpecifierLocVisitKind,
1666f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getNestedNameSpecifier(),
1667f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getOpaqueData()) { }
1668f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1669f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  static bool classof(const VisitorJob *VJ) {
1670f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return VJ->getKind() == VisitorJob::NestedNameSpecifierLocVisitKind;
1671f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1672f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1673f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLoc get() const {
1674f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return NestedNameSpecifierLoc(static_cast<NestedNameSpecifier*>(data[0]),
1675f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                                  data[1]);
1676f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1677f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor};
1678f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1679f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekclass DeclarationNameInfoVisit : public VisitorJob {
1680f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekpublic:
1681f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfoVisit(Stmt *S, CXCursor parent)
1682f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    : VisitorJob(parent, VisitorJob::DeclarationNameInfoVisitKind, S) {}
1683f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1684f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return VJ->getKind() == VisitorJob::DeclarationNameInfoVisitKind;
1685f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1686f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfo get() const {
1687f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    Stmt *S = static_cast<Stmt*>(data[0]);
1688f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    switch (S->getStmtClass()) {
1689f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    default:
1690f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      llvm_unreachable("Unhandled Stmt");
1691ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor    case clang::Stmt::MSDependentExistsStmtClass:
1692ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor      return cast<MSDependentExistsStmt>(S)->getNameInfo();
1693f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::CXXDependentScopeMemberExprClass:
1694f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<CXXDependentScopeMemberExpr>(S)->getMemberNameInfo();
1695f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::DependentScopeDeclRefExprClass:
1696f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<DependentScopeDeclRefExpr>(S)->getNameInfo();
1697f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    }
1698f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1699ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek};
1700cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekclass MemberRefVisit : public VisitorJob {
1701cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekpublic:
1702cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  MemberRefVisit(FieldDecl *D, SourceLocation L, CXCursor parent)
1703cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    : VisitorJob(parent, VisitorJob::MemberRefVisitKind, D,
1704dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 L.getPtrEncoding()) {}
1705cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  static bool classof(const VisitorJob *VJ) {
1706cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return VJ->getKind() == VisitorJob::MemberRefVisitKind;
1707cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1708cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  FieldDecl *get() const {
1709cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return static_cast<FieldDecl*>(data[0]);
1710cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1711cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  SourceLocation getLoc() const {
1712cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return SourceLocation::getFromRawEncoding((unsigned)(uintptr_t) data[1]);
1713cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1714cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek};
171528a719433411ef782b582946823bc648ddcc4533Ted Kremenekclass EnqueueVisitor : public StmtVisitor<EnqueueVisitor, void> {
171628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitorWorkList &WL;
171728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  CXCursor Parent;
171828a719433411ef782b582946823bc648ddcc4533Ted Kremenekpublic:
171928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueVisitor(VisitorWorkList &wl, CXCursor parent)
172028a719433411ef782b582946823bc648ddcc4533Ted Kremenek    : WL(wl), Parent(parent) {}
172128a719433411ef782b582946823bc648ddcc4533Ted Kremenek
1722ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitAddrLabelExpr(AddrLabelExpr *E);
172373d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitBlockExpr(BlockExpr *B);
172428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCompoundLiteralExpr(CompoundLiteralExpr *E);
1725083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  void VisitCompoundStmt(CompoundStmt *S);
172611b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXDefaultArgExpr(CXXDefaultArgExpr *E) { /* Do nothing. */ }
1727ba0513de93d2fab6db5ab30b6927209fcc883078Douglas Gregor  void VisitMSDependentExistsStmt(MSDependentExistsStmt *S);
1728f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitCXXDependentScopeMemberExpr(CXXDependentScopeMemberExpr *E);
172911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXNewExpr(CXXNewExpr *E);
17306d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  void VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E);
173128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E);
1732cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E);
173373d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E);
1734b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  void VisitCXXTypeidExpr(CXXTypeidExpr *E);
173555b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  void VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr *E);
17361e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  void VisitCXXUuidofExpr(CXXUuidofExpr *E);
1737dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis  void VisitCXXCatchStmt(CXXCatchStmt *S);
1738e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  void VisitDeclRefExpr(DeclRefExpr *D);
1739035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  void VisitDeclStmt(DeclStmt *S);
1740f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E);
1741cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitDesignatedInitExpr(DesignatedInitExpr *E);
174228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitExplicitCastExpr(ExplicitCastExpr *E);
174328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitForStmt(ForStmt *FS);
1744ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitGotoStmt(GotoStmt *GS);
174528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitIfStmt(IfStmt *If);
174628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitInitListExpr(InitListExpr *IE);
174728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitMemberExpr(MemberExpr *M);
1748cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitOffsetOfExpr(OffsetOfExpr *E);
174973d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitObjCEncodeExpr(ObjCEncodeExpr *E);
175028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitObjCMessageExpr(ObjCMessageExpr *M);
175128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitOverloadExpr(OverloadExpr *E);
1752f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne  void VisitUnaryExprOrTypeTraitExpr(UnaryExprOrTypeTraitExpr *E);
175328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitStmt(Stmt *S);
175428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitSwitchStmt(SwitchStmt *S);
175528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitWhileStmt(WhileStmt *W);
17562939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  void VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E);
17576ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  void VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E);
17584ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregor  void VisitTypeTraitExpr(TypeTraitExpr *E);
175921ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  void VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E);
1760552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  void VisitExpressionTraitExpr(ExpressionTraitExpr *E);
176128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U);
17629d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  void VisitVAArgExpr(VAArgExpr *E);
176394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  void VisitSizeOfPackExpr(SizeOfPackExpr *E);
17644b9c2d235fb9449e249d74f48ecfec601650de93John McCall  void VisitPseudoObjectExpr(PseudoObjectExpr *E);
17654b9c2d235fb9449e249d74f48ecfec601650de93John McCall  void VisitOpaqueValueExpr(OpaqueValueExpr *E);
1766011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor  void VisitLambdaExpr(LambdaExpr *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) {
18602aed8b88613863f3c439cdfb205bdf8b608fb205Sebastian Redl  // Enqueue the initializer , if any.
18612aed8b88613863f3c439cdfb205bdf8b608fb205Sebastian Redl  AddStmt(E->getInitializer());
186211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the array size, if any.
186311b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddStmt(E->getArraySize());
186411b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the allocated type.
186511b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddTypeLoc(E->getAllocatedTypeSourceInfo());
186611b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the placement arguments.
186711b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  for (unsigned I = E->getNumPlacementArgs(); I > 0; --I)
186811b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek    AddStmt(E->getPlacementArg(I-1));
186911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek}
187028a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *CE) {
18718b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek  for (unsigned I = CE->getNumArgs(); I > 1 /* Yes, this is 1 */; --I)
18728b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek    AddStmt(CE->getArg(I-1));
187328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getCallee());
187428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getArg(0));
187528a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1876cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E) {
1877cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the name of the type being destroyed.
1878cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getDestroyedTypeInfo());
1879cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the scope type that looks disturbingly like the nested-name-specifier
1880cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // but isn't.
1881cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getScopeTypeInfo());
1882cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the nested-name-specifier.
1883f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = E->getQualifierLoc())
1884f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1885cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit base expression.
1886cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getBase());
1887cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
18886d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenekvoid EnqueueVisitor::VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E) {
18896d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
18906d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
189173d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E) {
189273d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  EnqueueChildren(E);
189373d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
189473d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
1895b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenekvoid EnqueueVisitor::VisitCXXTypeidExpr(CXXTypeidExpr *E) {
1896b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  EnqueueChildren(E);
1897b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  if (E->isTypeOperand())
1898b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
1899b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek}
190055b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek
190155b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenekvoid EnqueueVisitor::VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr
190255b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek                                                     *E) {
190355b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  EnqueueChildren(E);
190455b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
190555b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek}
19061e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenekvoid EnqueueVisitor::VisitCXXUuidofExpr(CXXUuidofExpr *E) {
19071e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  EnqueueChildren(E);
19081e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  if (E->isTypeOperand())
19091e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
19101e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek}
1911dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis
1912dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidisvoid EnqueueVisitor::VisitCXXCatchStmt(CXXCatchStmt *S) {
1913dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis  EnqueueChildren(S);
1914dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis  AddDecl(S->getExceptionDecl());
1915dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis}
1916dcbb2fb8710459fdc8073b76a4ef73fbbcbeac9fArgyrios Kyrtzidis
1917e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenekvoid EnqueueVisitor::VisitDeclRefExpr(DeclRefExpr *DR) {
191860608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  if (DR->hasExplicitTemplateArgs()) {
191960608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek    AddExplicitTemplateArgs(&DR->getExplicitTemplateArgs());
192060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  }
1921e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  WL.push_back(DeclRefExprParts(DR, Parent));
1922e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek}
1923f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E) {
1924f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
1925f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddDeclarationNameInfo(E);
192600cf3cc2718671aa48e8da264a523b0058a8591eDouglas Gregor  AddNestedNameSpecifierLoc(E->getQualifierLoc());
1927f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
1928035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekvoid EnqueueVisitor::VisitDeclStmt(DeclStmt *S) {
1929035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  unsigned size = WL.size();
1930035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  bool isFirst = true;
1931035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  for (DeclStmt::decl_iterator D = S->decl_begin(), DEnd = S->decl_end();
1932035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek       D != DEnd; ++D) {
1933035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    AddDecl(*D, isFirst);
1934035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    isFirst = false;
1935035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1936035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  if (size == WL.size())
1937035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return;
1938035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // Now reverse the entries we just added.  This will match the DFS
1939035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // ordering performed by the worklist.
1940035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  VisitorWorkList::iterator I = WL.begin() + size, E = WL.end();
1941035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  std::reverse(I, E);
1942035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek}
1943cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitDesignatedInitExpr(DesignatedInitExpr *E) {
1944cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getInit());
1945cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  typedef DesignatedInitExpr::Designator Designator;
1946cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (DesignatedInitExpr::reverse_designators_iterator
1947cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D = E->designators_rbegin(), DEnd = E->designators_rend();
1948cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D != DEnd; ++D) {
1949cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isFieldDesignator()) {
1950cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      if (FieldDecl *Field = D->getField())
1951cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        AddMemberRef(Field, D->getFieldLoc());
1952cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1953cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1954cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isArrayDesignator()) {
1955cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getArrayIndex(*D));
1956cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1957cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1958cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    assert(D->isArrayRangeDesignator() && "Unknown designator kind");
1959cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeEnd(*D));
1960cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeStart(*D));
1961cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1962cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
196328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitExplicitCastExpr(ExplicitCastExpr *E) {
196428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(E);
196528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(E->getTypeInfoAsWritten());
196628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
196728a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitForStmt(ForStmt *FS) {
196828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getBody());
196928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInc());
197028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getCond());
197128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(FS->getConditionVariable());
197228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInit());
197328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1974ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekvoid EnqueueVisitor::VisitGotoStmt(GotoStmt *GS) {
1975ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  WL.push_back(LabelRefVisit(GS->getLabel(), GS->getLabelLoc(), Parent));
1976ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek}
197728a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitIfStmt(IfStmt *If) {
197828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getElse());
197928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getThen());
198028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getCond());
198128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(If->getConditionVariable());
198228a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
198328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitInitListExpr(InitListExpr *IE) {
198428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  // We care about the syntactic form of the initializer list, only.
198528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (InitListExpr *Syntactic = IE->getSyntacticForm())
198628a719433411ef782b582946823bc648ddcc4533Ted Kremenek    IE = Syntactic;
198728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(IE);
198828a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
198928a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitMemberExpr(MemberExpr *M) {
199089629a746019a42797495b091711a1d68467e88aDouglas Gregor  WL.push_back(MemberExprParts(M, Parent));
199189629a746019a42797495b091711a1d68467e88aDouglas Gregor
199289629a746019a42797495b091711a1d68467e88aDouglas Gregor  // If the base of the member access expression is an implicit 'this', don't
199389629a746019a42797495b091711a1d68467e88aDouglas Gregor  // visit it.
199489629a746019a42797495b091711a1d68467e88aDouglas Gregor  // FIXME: If we ever want to show these implicit accesses, this will be
199589629a746019a42797495b091711a1d68467e88aDouglas Gregor  // unfortunate. However, clang_getCursor() relies on this behavior.
199675e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor  if (!M->isImplicitAccess())
199775e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor    AddStmt(M->getBase());
199828a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
199973d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitObjCEncodeExpr(ObjCEncodeExpr *E) {
200073d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getEncodedTypeSourceInfo());
200173d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
200228a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitObjCMessageExpr(ObjCMessageExpr *M) {
200328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(M);
200428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(M->getClassReceiverTypeInfo());
200528a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
2006cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitOffsetOfExpr(OffsetOfExpr *E) {
2007cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the components of the offsetof expression.
2008cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (unsigned N = E->getNumComponents(), I = N; I > 0; --I) {
2009cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    typedef OffsetOfExpr::OffsetOfNode OffsetOfNode;
2010cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    const OffsetOfNode &Node = E->getComponent(I-1);
2011cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    switch (Node.getKind()) {
2012cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Array:
2013cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getIndexExpr(Node.getArrayExprIndex()));
2014cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2015cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Field:
201606dec892b5300b43263d25c5476b506c9d6cfbadAbramo Bagnara      AddMemberRef(Node.getField(), Node.getSourceRange().getEnd());
2017cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2018cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Identifier:
2019cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Base:
2020cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
2021cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
2022cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
2023cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the type into which we're computing the offset.
2024cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
2025cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
202628a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitOverloadExpr(OverloadExpr *E) {
202760608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
20286045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek  WL.push_back(OverloadExprParts(E, Parent));
20296045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek}
2030f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbournevoid EnqueueVisitor::VisitUnaryExprOrTypeTraitExpr(
2031f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne                                              UnaryExprOrTypeTraitExpr *E) {
20326d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  EnqueueChildren(E);
20336d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  if (E->isArgumentType())
20346d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek    AddTypeLoc(E->getArgumentTypeInfo());
20356d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
203628a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitStmt(Stmt *S) {
203728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(S);
203828a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
203928a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitSwitchStmt(SwitchStmt *S) {
204028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getBody());
204128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getCond());
204228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(S->getConditionVariable());
204328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
2044fafa75aebadef8d6b44a920e3f40529f150a5574Ted Kremenek
204528a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitWhileStmt(WhileStmt *W) {
204628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getBody());
204728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getCond());
204828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(W->getConditionVariable());
204928a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
205021ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
20512939b6f356161f572712d4d6310b65f9599e3675Ted Kremenekvoid EnqueueVisitor::VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E) {
20522939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  AddTypeLoc(E->getQueriedTypeSourceInfo());
20532939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek}
20546ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
20556ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichetvoid EnqueueVisitor::VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E) {
20566ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  AddTypeLoc(E->getRhsTypeSourceInfo());
20570a03a3f98b14006a54bcac9e8908a7c9f50e519fFrancois Pichet  AddTypeLoc(E->getLhsTypeSourceInfo());
20586ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet}
20596ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
20604ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregorvoid EnqueueVisitor::VisitTypeTraitExpr(TypeTraitExpr *E) {
20614ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregor  for (unsigned I = E->getNumArgs(); I > 0; --I)
20624ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregor    AddTypeLoc(E->getArg(I-1));
20634ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregor}
20644ca8ac2e61c37ddadf37024af86f3e1019af8532Douglas Gregor
206521ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegleyvoid EnqueueVisitor::VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E) {
206621ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  AddTypeLoc(E->getQueriedTypeSourceInfo());
206721ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley}
206821ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
2069552622067dc45013d240f73952fece703f5e63bdJohn Wiegleyvoid EnqueueVisitor::VisitExpressionTraitExpr(ExpressionTraitExpr *E) {
2070552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  EnqueueChildren(E);
2071552622067dc45013d240f73952fece703f5e63bdJohn Wiegley}
2072552622067dc45013d240f73952fece703f5e63bdJohn Wiegley
207328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U) {
207428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitOverloadExpr(U);
207528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (!U->isImplicitAccess())
207628a719433411ef782b582946823bc648ddcc4533Ted Kremenek    AddStmt(U->getBase());
207728a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
20789d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenekvoid EnqueueVisitor::VisitVAArgExpr(VAArgExpr *E) {
20799d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddStmt(E->getSubExpr());
20809d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddTypeLoc(E->getWrittenTypeInfo());
20819d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek}
208294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregorvoid EnqueueVisitor::VisitSizeOfPackExpr(SizeOfPackExpr *E) {
208394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  WL.push_back(SizeOfPackExprParts(E, Parent));
208494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor}
20854b9c2d235fb9449e249d74f48ecfec601650de93John McCallvoid EnqueueVisitor::VisitOpaqueValueExpr(OpaqueValueExpr *E) {
20864b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // If the opaque value has a source expression, just transparently
20874b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // visit that.  This is useful for (e.g.) pseudo-object expressions.
20884b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (Expr *SourceExpr = E->getSourceExpr())
20894b9c2d235fb9449e249d74f48ecfec601650de93John McCall    return Visit(SourceExpr);
20904b9c2d235fb9449e249d74f48ecfec601650de93John McCall}
2091011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregorvoid EnqueueVisitor::VisitLambdaExpr(LambdaExpr *E) {
2092011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor  AddStmt(E->getBody());
2093011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor  WL.push_back(LambdaExprParts(E, Parent));
2094011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor}
20954b9c2d235fb9449e249d74f48ecfec601650de93John McCallvoid EnqueueVisitor::VisitPseudoObjectExpr(PseudoObjectExpr *E) {
20964b9c2d235fb9449e249d74f48ecfec601650de93John McCall  // Treat the expression like its syntactic form.
20974b9c2d235fb9449e249d74f48ecfec601650de93John McCall  Visit(E->getSyntacticForm());
20984b9c2d235fb9449e249d74f48ecfec601650de93John McCall}
20996045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek
2100c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekvoid CursorVisitor::EnqueueWorkList(VisitorWorkList &WL, Stmt *S) {
2101aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  EnqueueVisitor(WL, MakeCXCursor(S, StmtParent, TU,RegionOfInterest)).Visit(S);
2102c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2103c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2104c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::IsInRegionOfInterest(CXCursor C) {
2105c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  if (RegionOfInterest.isValid()) {
2106c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SourceRange Range = getRawCursorExtent(C);
2107c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    if (Range.isInvalid() || CompareRegionOfInterest(Range))
2108c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      return false;
2109c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2110c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return true;
2111c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2112c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2113c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::RunVisitorWorkList(VisitorWorkList &WL) {
2114c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  while (!WL.empty()) {
2115c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Dequeue the worklist item.
211682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    VisitorJob LI = WL.back();
211782f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    WL.pop_back();
211882f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek
2119c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Set the Parent field, then back to its old value once we're done.
2120c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SetParentRAII SetParent(Parent, StmtParent, LI.getParent());
2121c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2122c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    switch (LI.getKind()) {
2123f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      case VisitorJob::DeclVisitKind: {
212482f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Decl *D = cast<DeclVisit>(&LI)->get();
2125f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        if (!D)
2126f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek          continue;
2127f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2128f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // For now, perform default visitation for Decls.
2129aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        if (Visit(MakeCXCursor(D, TU, RegionOfInterest,
2130aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                               cast<DeclVisit>(&LI)->isFirst())))
2131f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek            return true;
2132f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2133f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        continue;
2134f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      }
213560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      case VisitorJob::ExplicitTemplateArgsVisitKind: {
2136b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis        const ASTTemplateArgumentListInfo *ArgList =
213760608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          cast<ExplicitTemplateArgsVisit>(&LI)->get();
213860608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        for (const TemplateArgumentLoc *Arg = ArgList->getTemplateArgs(),
213960608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               *ArgEnd = Arg + ArgList->NumTemplateArgs;
214060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               Arg != ArgEnd; ++Arg) {
214160608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          if (VisitTemplateArgumentLoc(*Arg))
214260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek            return true;
214360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        }
214460608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        continue;
214560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      }
2146cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      case VisitorJob::TypeLocVisitKind: {
2147cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        // Perform default visitation for TypeLocs.
214882f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        if (Visit(cast<TypeLocVisit>(&LI)->get()))
2149cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek          return true;
2150cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        continue;
2151cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      }
2152ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      case VisitorJob::LabelRefVisitKind: {
2153ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner        LabelDecl *LS = cast<LabelRefVisit>(&LI)->get();
2154e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        if (LabelStmt *stmt = LS->getStmt()) {
2155e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          if (Visit(MakeCursorLabelRef(stmt, cast<LabelRefVisit>(&LI)->getLoc(),
2156e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek                                       TU))) {
2157e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek            return true;
2158e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          }
2159e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        }
2160ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek        continue;
2161ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      }
216247695c8ad8424851f62e0d4a983b45b15daee1c5Ted Kremenek
2163f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      case VisitorJob::NestedNameSpecifierLocVisitKind: {
2164f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        NestedNameSpecifierLocVisit *V = cast<NestedNameSpecifierLocVisit>(&LI);
2165f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        if (VisitNestedNameSpecifierLoc(V->get()))
2166f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor          return true;
2167f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        continue;
2168f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      }
2169f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
2170f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      case VisitorJob::DeclarationNameInfoVisitKind: {
2171f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        if (VisitDeclarationNameInfo(cast<DeclarationNameInfoVisit>(&LI)
2172f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek                                     ->get()))
2173f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek          return true;
2174f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        continue;
2175f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      }
2176cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      case VisitorJob::MemberRefVisitKind: {
2177cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        MemberRefVisit *V = cast<MemberRefVisit>(&LI);
2178cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (Visit(MakeCursorMemberRef(V->get(), V->getLoc(), TU)))
2179cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          return true;
2180cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        continue;
2181cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      }
2182c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::StmtVisitKind: {
218382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Stmt *S = cast<StmtVisit>(&LI)->get();
21848c269ac75569454a049385b1246140db5f2b6faaTed Kremenek        if (!S)
21858c269ac75569454a049385b1246140db5f2b6faaTed Kremenek          continue;
21868c269ac75569454a049385b1246140db5f2b6faaTed Kremenek
2187f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // Update the current cursor.
2188aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis        CXCursor Cursor = MakeCXCursor(S, StmtParent, TU, RegionOfInterest);
2189cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (!IsInRegionOfInterest(Cursor))
2190cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          continue;
2191cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        switch (Visitor(Cursor, Parent, ClientData)) {
2192cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Break: return true;
2193cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Continue: break;
2194cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Recurse:
2195cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek            EnqueueWorkList(WL, S);
219682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek            break;
2197c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
219882f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        continue;
2199c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2200c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::MemberExprPartsKind: {
2201c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Handle the other pieces in the MemberExpr besides the base.
220282f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        MemberExpr *M = cast<MemberExprParts>(&LI)->get();
2203c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2204c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the nested-name-specifier
220540d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = M->getQualifierLoc())
220640d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2207c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            return true;
2208c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2209c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the declaration name.
2210c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (VisitDeclarationNameInfo(M->getMemberNameInfo()))
2211c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          return true;
2212c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2213c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the explicitly-specified template arguments, if any.
2214c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (M->hasExplicitTemplateArgs()) {
2215c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          for (const TemplateArgumentLoc *Arg = M->getTemplateArgs(),
2216c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               *ArgEnd = Arg + M->getNumTemplateArgs();
2217c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               Arg != ArgEnd; ++Arg) {
2218c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            if (VisitTemplateArgumentLoc(*Arg))
2219c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek              return true;
2220c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          }
2221c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
2222c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        continue;
2223c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2224e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      case VisitorJob::DeclRefExprPartsKind: {
222582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        DeclRefExpr *DR = cast<DeclRefExprParts>(&LI)->get();
2226e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit nested-name-specifier, if present.
222740d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = DR->getQualifierLoc())
222840d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2229e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek            return true;
2230e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit declaration name.
2231e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        if (VisitDeclarationNameInfo(DR->getNameInfo()))
2232e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek          return true;
2233e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        continue;
2234e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      }
22356045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      case VisitorJob::OverloadExprPartsKind: {
223682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        OverloadExpr *O = cast<OverloadExprParts>(&LI)->get();
22376045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the nested-name-specifier.
22384c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = O->getQualifierLoc())
22394c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
22406045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek            return true;
22416045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the declaration name.
22426045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (VisitDeclarationNameInfo(O->getNameInfo()))
22436045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22446045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the overloaded declaration reference.
22456045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (Visit(MakeCursorOverloadedDeclRef(O, TU)))
22466045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22476045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        continue;
22486045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      }
224994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      case VisitorJob::SizeOfPackExprPartsKind: {
225094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        SizeOfPackExpr *E = cast<SizeOfPackExprParts>(&LI)->get();
225194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        NamedDecl *Pack = E->getPack();
225294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTypeParmDecl>(Pack)) {
225394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTypeRef(cast<TemplateTypeParmDecl>(Pack),
225494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                      E->getPackLoc(), TU)))
225594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
225694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
225794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
225894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
225994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
226094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTemplateParmDecl>(Pack)) {
226194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTemplateRef(cast<TemplateTemplateParmDecl>(Pack),
226294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                          E->getPackLoc(), TU)))
226394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
226494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
226594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
226694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
226794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
226894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // Non-type template parameter packs and function parameter packs are
226994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // treated like DeclRefExpr cursors.
227094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        continue;
227194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      }
2272011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
2273011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      case VisitorJob::LambdaExprPartsKind: {
2274011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        // Visit captures.
2275011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        LambdaExpr *E = cast<LambdaExprParts>(&LI)->get();
2276011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        for (LambdaExpr::capture_iterator C = E->explicit_capture_begin(),
2277011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                                       CEnd = E->explicit_capture_end();
2278011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor             C != CEnd; ++C) {
2279011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          if (C->capturesThis())
2280011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            continue;
2281011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
2282011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          if (Visit(MakeCursorVariableRef(C->getCapturedVar(),
2283011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                                          C->getLocation(),
2284011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                                          TU)))
2285011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            return true;
2286011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        }
2287011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
2288011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        // Visit parameters and return type, if present.
2289011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        if (E->hasExplicitParameters() || E->hasExplicitResultType()) {
2290011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          TypeLoc TL = E->getCallOperator()->getTypeSourceInfo()->getTypeLoc();
2291011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          if (E->hasExplicitParameters() && E->hasExplicitResultType()) {
2292011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            // Visit the whole type.
2293011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            if (Visit(TL))
2294011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor              return true;
2295011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          } else if (isa<FunctionProtoTypeLoc>(TL)) {
2296011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            FunctionProtoTypeLoc Proto = cast<FunctionProtoTypeLoc>(TL);
2297011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            if (E->hasExplicitParameters()) {
2298011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor              // Visit parameters.
2299011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor              for (unsigned I = 0, N = Proto.getNumArgs(); I != N; ++I)
2300011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                if (Visit(MakeCXCursor(Proto.getArg(I), TU)))
2301011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                  return true;
2302011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            } else {
2303011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor              // Visit result type.
2304011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor              if (Visit(Proto.getResultLoc()))
2305011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor                return true;
2306011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor            }
2307011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor          }
2308011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        }
2309011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor        break;
2310011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      }
2311c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    }
2312c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2313c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return false;
2314c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2315c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2316cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekbool CursorVisitor::Visit(Stmt *S) {
2317d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  VisitorWorkList *WL = 0;
2318d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  if (!WorkListFreeList.empty()) {
2319d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = WorkListFreeList.back();
2320d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL->clear();
2321d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListFreeList.pop_back();
2322d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2323d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  else {
2324d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = new VisitorWorkList();
2325d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListCache.push_back(WL);
2326d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2327d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  EnqueueWorkList(*WL, S);
2328d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  bool result = RunVisitorWorkList(*WL);
2329d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  WorkListFreeList.push_back(WL);
2330d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  return result;
2331c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2332c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
233348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichetnamespace {
233448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichettypedef llvm::SmallVector<SourceRange, 4> RefNamePieces;
233548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois PichetRefNamePieces buildPieces(unsigned NameFlags, bool IsMemberRefExpr,
233648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const DeclarationNameInfo &NI,
233748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const SourceRange &QLoc,
2338b0c3e0909bb04af0bfb82ad01ab6909649d68ccaArgyrios Kyrtzidis                          const ASTTemplateArgumentListInfo *TemplateArgs = 0){
233948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantQualifier = NameFlags & CXNameRange_WantQualifier;
234048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantTemplateArgs = NameFlags & CXNameRange_WantTemplateArgs;
234148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantSinglePiece = NameFlags & CXNameRange_WantSinglePiece;
234248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
234348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const DeclarationName::NameKind Kind = NI.getName().getNameKind();
234448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
234548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  RefNamePieces Pieces;
234648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
234748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantQualifier && QLoc.isValid())
234848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(QLoc);
234948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
235048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind != DeclarationName::CXXOperatorName || IsMemberRefExpr)
235148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(NI.getLoc());
235248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
235348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantTemplateArgs && TemplateArgs)
235448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceRange(TemplateArgs->LAngleLoc,
235548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                                 TemplateArgs->RAngleLoc));
235648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
235748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind == DeclarationName::CXXOperatorName) {
235848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
235948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.BeginOpNameLoc));
236048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
236148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.EndOpNameLoc));
236248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
236348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
236448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantSinglePiece) {
236548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    SourceRange R(Pieces.front().getBegin(), Pieces.back().getEnd());
236648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.clear();
236748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(R);
236848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
236948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
237048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  return Pieces;
237148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
237248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
237348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
2374c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2375c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Misc. API hooks.
2376c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2377c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
23788c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic llvm::sys::Mutex EnableMultithreadingMutex;
23798c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic bool EnabledMultithreading;
23808c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
2381fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidisstatic void fatal_error_handler(void *user_data, const std::string& reason) {
2382fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis  // Write the result out to stderr avoiding errs() because raw_ostreams can
2383fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis  // call report_fatal_error.
2384db7a800e0b76036d0faa7123f2e05e45ee3294e5Argyrios Kyrtzidis  fprintf(stderr, "LIBCLANG FATAL ERROR: %s\n", reason.c_str());
2385fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis  ::abort();
2386fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis}
2387fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis
23885e4bc590b0ea010e38372d0b4a0aab578a746fe6Benjamin Kramerextern "C" {
23890a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas GregorCXIndex clang_createIndex(int excludeDeclarationsFromPCH,
23900a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          int displayDiagnostics) {
239148615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // Disable pretty stack trace functionality, which will otherwise be a very
239248615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // poor citizen of the world and set up all sorts of signal handlers.
239348615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  llvm::DisablePrettyStackTrace = true;
239448615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar
2395c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // We use crash recovery to make some of our APIs more reliable, implicitly
2396c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // enable it.
2397c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  llvm::CrashRecoveryContext::Enable();
2398c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar
23998c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  // Enable support for multithreading in LLVM.
24008c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  {
24018c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    llvm::sys::ScopedLock L(EnableMultithreadingMutex);
24028c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    if (!EnabledMultithreading) {
2403fa39f5b76bafdf536c5e305f821eb1b7f11079bdArgyrios Kyrtzidis      llvm::install_fatal_error_handler(fatal_error_handler, 0);
24048c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      llvm::llvm_start_multithreaded();
24058c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      EnabledMultithreading = true;
24068c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    }
24078c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  }
24088c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
2409a030b7cf5e6aad5889b1b662b6979840bc75f87fDouglas Gregor  CIndexer *CIdxr = new CIndexer();
2410e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  if (excludeDeclarationsFromPCH)
2411e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    CIdxr->setOnlyLocalDecls();
24120a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (displayDiagnostics)
24130a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    CIdxr->setDisplayDiagnostics();
2414e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  return CIdxr;
2415600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2416600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
24179ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeIndex(CXIndex CIdx) {
24182b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (CIdx)
24192b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    delete static_cast<CIndexer *>(CIdx);
24202bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
24212bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff
2422d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenekvoid clang_toggleCrashRecovery(unsigned isEnabled) {
2423d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  if (isEnabled)
2424d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Enable();
2425d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  else
2426d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Disable();
2427d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek}
2428d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek
24299ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit clang_createTranslationUnit(CXIndex CIdx,
2430a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                              const char *ast_filename) {
24312b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
24322b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    return 0;
2433f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
24347d1d49d2971b20a97b3c2a301470b9eaaa130137Douglas Gregor  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2435389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOptions FileSystemOpts;
2436389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOpts.WorkingDir = CXXIdx->getWorkingDirectory();
24370d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
2438c93dc7889644293e318e19d82830ea2acc45b678Dylan Noblesmith  IntrusiveRefCntPtr<DiagnosticsEngine> Diags;
2439a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *TU = ASTUnit::LoadFromASTFile(ast_filename, Diags, FileSystemOpts,
2440a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                  CXXIdx->getOnlyLocalDecls(),
2441bef35c91b594f66216f4aab303b71a6c5ab7abcfArgyrios Kyrtzidis                                  0, 0,
2442bef35c91b594f66216f4aab303b71a6c5ab7abcfArgyrios Kyrtzidis                                  /*CaptureDiagnostics=*/true,
2443bef35c91b594f66216f4aab303b71a6c5ab7abcfArgyrios Kyrtzidis                                  /*AllowPCHWithCompilerErrors=*/true);
2444a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return MakeCXTranslationUnit(TU);
2445600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2446600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2447b1c031be513705d924038f497279b9b599868ba1Douglas Gregorunsigned clang_defaultEditingTranslationUnitOptions() {
24482a2c50b330e7754499f42173616a36865b5f313bDouglas Gregor  return CXTranslationUnit_PrecompiledPreamble |
2449b5af843a20e237ad1a13ad66a867e200695b8c8eDouglas Gregor         CXTranslationUnit_CacheCompletionResults;
2450b1c031be513705d924038f497279b9b599868ba1Douglas Gregor}
2451b1c031be513705d924038f497279b9b599868ba1Douglas Gregor
24529ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit
24539ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarclang_createTranslationUnitFromSourceFile(CXIndex CIdx,
24549ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          const char *source_filename,
24559ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          int num_command_line_args,
24562ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                          const char * const *command_line_args,
24574db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                                          unsigned num_unsaved_files,
2458a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                          struct CXUnsavedFile *unsaved_files) {
2459e1d4330adaaa7faf093e725c9c993207eb2d778aArgyrios Kyrtzidis  unsigned Options = CXTranslationUnit_DetailedPreprocessingRecord;
24605a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor  return clang_parseTranslationUnit(CIdx, source_filename,
24615a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    command_line_args, num_command_line_args,
24625a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    unsaved_files, num_unsaved_files,
2463dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                    Options);
24645a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor}
246519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
246619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbarstruct ParseTranslationUnitInfo {
246719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx;
246819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename;
24692ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char *const *command_line_args;
247019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args;
247119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
247219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files;
247319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options;
247419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXTranslationUnit result;
247519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar};
2476b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_parseTranslationUnit_Impl(void *UserData) {
247719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo *PTUI =
247819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    static_cast<ParseTranslationUnitInfo*>(UserData);
247919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx = PTUI->CIdx;
248019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename = PTUI->source_filename;
24812ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char * const *command_line_args = PTUI->command_line_args;
248219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args = PTUI->num_command_line_args;
248319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files = PTUI->unsaved_files;
248419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files = PTUI->num_unsaved_files;
248519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options = PTUI->options;
248619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  PTUI->result = 0;
24875a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor
24882b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
248919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return;
2490f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2491e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2492e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff
249344c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  bool PrecompilePreamble = options & CXTranslationUnit_PrecompiledPreamble;
2494467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  // FIXME: Add a flag for modules.
2495467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  TranslationUnitKind TUKind
2496467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor    = (options & CXTranslationUnit_Incomplete)? TU_Prefix : TU_Complete;
249787c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  bool CacheCodeCompetionResults
249887c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    = options & CXTranslationUnit_CacheCompletionResults;
249987c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor
25005352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  // Configure the diagnostics.
25015352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  DiagnosticOptions DiagOpts;
2502c93dc7889644293e318e19d82830ea2acc45b678Dylan Noblesmith  IntrusiveRefCntPtr<DiagnosticsEngine>
250325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Diags(CompilerInstance::createDiagnostics(DiagOpts, num_command_line_args,
250425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                                command_line_args));
250525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
250625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
2507d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie  llvm::CrashRecoveryContextCleanupRegistrar<DiagnosticsEngine,
2508d6471f7c1921c7802804ce3ff6fe9768310f72b9David Blaikie    llvm::CrashRecoveryContextReleaseRefCleanup<DiagnosticsEngine> >
250925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    DiagCleanup(Diags.getPtr());
251025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
25111e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<std::vector<ASTUnit::RemappedFile> >
251225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
251325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
251425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
251525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
251625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
2517f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
25184db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
25195f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2520f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    const llvm::MemoryBuffer *Buffer
2521a0a270c0f1c0a4e3482438bdc5f4a7bd3d25f0a6Chris Lattner      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
252225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
252325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
25244db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
2525f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
25261e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<std::vector<const char *> >
252725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args(new std::vector<const char*>());
252825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
252925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this method.
253025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<std::vector<const char*> >
253125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    ArgsCleanup(Args.get());
253225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
253352ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // Since the Clang C library is primarily used by batch tools dealing with
253452ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // (often very broken) source code, where spell-checking can have a
253552ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // significant negative impact on performance (particularly when
253652ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // precompiled headers are involved), we disable it by default.
2537b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  // Only do this if we haven't found a spell-checking-related argument.
2538b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  bool FoundSpellCheckingArgument = false;
2539b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  for (int I = 0; I != num_command_line_args; ++I) {
2540b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (strcmp(command_line_args[I], "-fno-spell-checking") == 0 ||
2541b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        strcmp(command_line_args[I], "-fspell-checking") == 0) {
2542b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      FoundSpellCheckingArgument = true;
2543b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      break;
2544e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    }
2545b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  }
2546b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  if (!FoundSpellCheckingArgument)
254725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-fno-spell-checking");
2548b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
254925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  Args->insert(Args->end(), command_line_args,
255025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek               command_line_args + num_command_line_args);
2551d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2552c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // The 'source_filename' argument is optional.  If the caller does not
2553c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // specify it then it is assumed that the source file is specified
2554c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // in the actual argument list.
2555c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // Put the source file after command_line_args otherwise if '-x' flag is
2556c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // present it will be unused.
2557c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  if (source_filename)
255825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back(source_filename);
2559c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis
256044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  // Do we need the detailed preprocessing record?
256144c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (options & CXTranslationUnit_DetailedPreprocessingRecord) {
256225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-Xclang");
256325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-detailed-preprocessing-record");
256444c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  }
256544c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
2566026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  unsigned NumErrors = Diags->getClient()->getNumErrors();
25671e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<ASTUnit> Unit(
25684ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek    ASTUnit::LoadFromCommandLine(Args->size() ? &(*Args)[0] : 0
25694ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 /* vector::data() not portable */,
25704ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 Args->size() ? (&(*Args)[0] + Args->size()) :0,
2571b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 Diags,
2572b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getClangResourcesPath(),
2573b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getOnlyLocalDecls(),
2574e47be3e9682e82da15059006f43c7f3c021e4fffDouglas Gregor                                 /*CaptureDiagnostics=*/true,
25754ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 RemappedFiles->size() ? &(*RemappedFiles)[0]:0,
257625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                 RemappedFiles->size(),
2577299a4a967b02c9f0d0d94ad8560e3ced893f9116Argyrios Kyrtzidis                                 /*RemappedFilesKeepOriginalName=*/true,
2578b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 PrecompilePreamble,
2579467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor                                 TUKind,
2580bef35c91b594f66216f4aab303b71a6c5ab7abcfArgyrios Kyrtzidis                                 CacheCodeCompetionResults,
2581bef35c91b594f66216f4aab303b71a6c5ab7abcfArgyrios Kyrtzidis                                 /*AllowPCHWithCompilerErrors=*/true));
2582b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
2583026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  if (NumErrors != Diags->getClient()->getNumErrors()) {
2584b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    // Make sure to check that 'Unit' is non-NULL.
2585b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (CXXIdx->getDisplayDiagnostics() && Unit.get()) {
2586b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      for (ASTUnit::stored_diag_iterator D = Unit->stored_diag_begin(),
2587b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                      DEnd = Unit->stored_diag_end();
2588b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor           D != DEnd; ++D) {
25894e4d08403ca5cfd4d558fa2936215d3a4e5a528dDavid Blaikie        CXStoredDiagnostic Diag(*D, Unit->getASTContext().getLangOpts());
2590b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        CXString Msg = clang_formatDiagnostic(&Diag,
2591b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                    clang_defaultDiagnosticDisplayOptions());
2592b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        fprintf(stderr, "%s\n", clang_getCString(Msg));
2593b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        clang_disposeString(Msg);
2594b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      }
2595274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor#ifdef LLVM_ON_WIN32
2596b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // On Windows, force a flush, since there may be multiple copies of
2597b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // stderr and stdout in the file system, all with different buffers
2598b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // but writing to the same device.
2599b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      fflush(stderr);
2600b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor#endif
2601b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    }
2602a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  }
2603d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2604a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  PTUI->result = MakeCXTranslationUnit(Unit.take());
260519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar}
260619ffd492a31a25fb691098bf79f317e5f3edf177Daniel DunbarCXTranslationUnit clang_parseTranslationUnit(CXIndex CIdx,
260719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             const char *source_filename,
26082ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                         const char * const *command_line_args,
260919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             int num_command_line_args,
26109e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                            struct CXUnsavedFile *unsaved_files,
261119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned num_unsaved_files,
261219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned options) {
261319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo PTUI = { CIdx, source_filename, command_line_args,
26149e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_command_line_args, unsaved_files,
26159e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_unsaved_files, options, 0 };
261619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  llvm::CrashRecoveryContext CRC;
261719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
2618bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_parseTranslationUnit_Impl, &PTUI)) {
261960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "libclang: crash detected during parsing: {\n");
262060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'source_filename' : '%s'\n", source_filename);
262160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'command_line_args' : [");
262260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (int i = 0; i != num_command_line_args; ++i) {
262360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
262460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
262560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "'%s'", command_line_args[i]);
262660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
262760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
262860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'unsaved_files' : [");
262960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (unsigned i = 0; i != num_unsaved_files; ++i) {
263060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
263160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
263260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "('%s', '...', %ld)", unsaved_files[i].Filename,
263360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar              unsaved_files[i].Length);
263460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
263560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
263660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'options' : %d,\n", options);
263760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "}\n");
263860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar
263919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return 0;
26406df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE")) {
26416df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(PTUI.result);
264219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  }
26436df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
264419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  return PTUI.result;
26455b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff}
26465b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff
26471999844e7a18786e61e619e1dc6c789827541863Douglas Gregorunsigned clang_defaultSaveOptions(CXTranslationUnit TU) {
26481999844e7a18786e61e619e1dc6c789827541863Douglas Gregor  return CXSaveTranslationUnit_None;
26491999844e7a18786e61e619e1dc6c789827541863Douglas Gregor}
26501999844e7a18786e61e619e1dc6c789827541863Douglas Gregor
26511999844e7a18786e61e619e1dc6c789827541863Douglas Gregorint clang_saveTranslationUnit(CXTranslationUnit TU, const char *FileName,
26521999844e7a18786e61e619e1dc6c789827541863Douglas Gregor                              unsigned options) {
26537ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU)
265439c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    return CXSaveError_InvalidTU;
26557ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
265639c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  CXSaveError result = static_cast<ASTUnit *>(TU->TUData)->Save(FileName);
26576df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  if (getenv("LIBCLANG_RESOURCE_USAGE"))
26586df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
26596df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  return result;
26607ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
266119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
26629ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeTranslationUnit(CXTranslationUnit CTUnit) {
2663ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  if (CTUnit) {
2664ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // If the translation unit has been marked as unsafe to free, just discard
2665ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // it.
2666a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    if (static_cast<ASTUnit *>(CTUnit->TUData)->isUnsafeToFree())
2667ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar      return;
2668ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2669a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete static_cast<ASTUnit *>(CTUnit->TUData);
2670a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    disposeCXStringPool(CTUnit->StringPool);
2671153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek    delete static_cast<CXDiagnosticSetImpl *>(CTUnit->Diagnostics);
2672a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete CTUnit;
2673ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  }
26742bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
26750d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
2676e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregorunsigned clang_defaultReparseOptions(CXTranslationUnit TU) {
2677e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor  return CXReparse_None;
2678e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor}
2679e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor
2680ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarstruct ReparseTranslationUnitInfo {
2681ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU;
2682ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files;
2683ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
2684ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options;
2685ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  int result;
2686ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar};
2687593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2688b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_reparseTranslationUnit_Impl(void *UserData) {
2689ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo *RTUI =
2690ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    static_cast<ReparseTranslationUnitInfo*>(UserData);
2691ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU = RTUI->TU;
2692153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2693153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  // Reset the associated diagnostics.
2694153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  delete static_cast<CXDiagnosticSetImpl*>(TU->Diagnostics);
2695153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek  TU->Diagnostics = 0;
2696153221717e39ce41323d5bc6b8b8bf130923c1bdTed Kremenek
2697ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files = RTUI->num_unsaved_files;
2698ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files = RTUI->unsaved_files;
2699ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options = RTUI->options;
2700ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  (void) options;
2701ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  RTUI->result = 1;
2702ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2703abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU)
2704ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return;
2705593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2706a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
2707593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
2708abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
27091e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<std::vector<ASTUnit::RemappedFile> >
271025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
271125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
271225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
271325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
271425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
271525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
2716abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
27175f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2718abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    const llvm::MemoryBuffer *Buffer
27191abc6bca3d7fb0e7b1e40fbcad6cfb5e10594548Douglas Gregor      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
272025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
272125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
2722abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
2723abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
27244ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek  if (!CXXUnit->Reparse(RemappedFiles->size() ? &(*RemappedFiles)[0] : 0,
27254ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                        RemappedFiles->size()))
2726593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor    RTUI->result = 0;
2727abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
2728593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2729ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarint clang_reparseTranslationUnit(CXTranslationUnit TU,
2730ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned num_unsaved_files,
2731ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 struct CXUnsavedFile *unsaved_files,
2732ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned options) {
2733ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo RTUI = { TU, num_unsaved_files, unsaved_files,
2734ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                      options, 0 };
27358c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis
2736e7de9b4a1f4a15620ab15bc8159018df7d54080aArgyrios Kyrtzidis  if (getenv("LIBCLANG_NOTHREADS")) {
27378c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis    clang_reparseTranslationUnit_Impl(&RTUI);
27388c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis    return RTUI.result;
27398c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis  }
27408c4b47eff0848dfd80854d70cb84b23b185828d4Argyrios Kyrtzidis
2741ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  llvm::CrashRecoveryContext CRC;
2742ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2743bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_reparseTranslationUnit_Impl, &RTUI)) {
2744b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbar    fprintf(stderr, "libclang: crash detected during reparsing\n");
2745a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    static_cast<ASTUnit *>(TU->TUData)->setUnsafeToFree(true);
2746ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return 1;
27476df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE"))
27486df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
27491dfb26af4d6aa4f7818e256659a79f1ec2cba784Ted Kremenek
2750ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  return RTUI.result;
2751ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar}
2752ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2753df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
27549ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getTranslationUnitSpelling(CXTranslationUnit CTUnit) {
27552b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CTUnit)
2756ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
2757f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2758a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(CTUnit->TUData);
2759ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(CXXUnit->getOriginalSourceFileName(), true);
2760af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
27611eb79b58e56b99cf557d5d353586a10c5360364dDaniel Dunbar
27627eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas GregorCXCursor clang_getTranslationUnitCursor(CXTranslationUnit TU) {
2763aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  CXCursor Result = { CXCursor_TranslationUnit, 0, { 0, 0, TU } };
27647eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return Result;
27657eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
27667eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
2767fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2768600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2769fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2770fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXFile Operations.
2771fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2772fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2773fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
277474844072411bae91d5dbb89955d200cbe1e0a1c8Ted KremenekCXString clang_getFileName(CXFile SFile) {
277598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
2776a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return createCXString((const char*)NULL);
2777f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
277888145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
277974844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  return createCXString(FEnt->getName());
278088145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
278188145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff
278288145034694ed5267fa6fa5febc54fadc02bd479Steve Narofftime_t clang_getFileTime(CXFile SFile) {
278398258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
278498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    return 0;
2785f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
278688145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
278788145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  return FEnt->getModificationTime();
2788ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff}
2789f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2790b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXFile clang_getFile(CXTranslationUnit tu, const char *file_name) {
2791b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!tu)
2792b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor    return 0;
2793f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2794a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2795f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2796b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  FileManager &FMgr = CXXUnit->getFileManager();
279739b49bcaaddb1049234fca9500c0ac02c088e23dChris Lattner  return const_cast<FileEntry *>(FMgr.getFile(file_name));
2798b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2799f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2800dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorunsigned clang_isFileMultipleIncludeGuarded(CXTranslationUnit tu, CXFile file) {
2801dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  if (!tu || !file)
2802dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor    return 0;
2803dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2804dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2805dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  FileEntry *FEnt = static_cast<FileEntry *>(file);
2806dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  return CXXUnit->getPreprocessor().getHeaderSearchInfo()
2807dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor                                          .isFileMultipleIncludeGuarded(FEnt);
2808dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor}
2809dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2810fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2811fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2812fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2813fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXCursor Operations.
2814fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2815fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2816fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekstatic Decl *getDeclFromExpr(Stmt *E) {
2817c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2818db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return getDeclFromExpr(CE->getSubExpr());
2819db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2820fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (DeclRefExpr *RefExpr = dyn_cast<DeclRefExpr>(E))
2821fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RefExpr->getDecl();
2822fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (MemberExpr *ME = dyn_cast<MemberExpr>(E))
2823fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return ME->getMemberDecl();
2824fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCIvarRefExpr *RE = dyn_cast<ObjCIvarRefExpr>(E))
2825fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RE->getDecl();
2826db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCPropertyRefExpr *PRE = dyn_cast<ObjCPropertyRefExpr>(E))
282712f78a6741a4cb3d904340f8d3d2714568b50e7aJohn McCall    return PRE->isExplicitProperty() ? PRE->getExplicitProperty() : 0;
28284b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (PseudoObjectExpr *POE = dyn_cast<PseudoObjectExpr>(E))
28294b9c2d235fb9449e249d74f48ecfec601650de93John McCall    return getDeclFromExpr(POE->getSyntacticForm());
28304b9c2d235fb9449e249d74f48ecfec601650de93John McCall  if (OpaqueValueExpr *OVE = dyn_cast<OpaqueValueExpr>(E))
28314b9c2d235fb9449e249d74f48ecfec601650de93John McCall    if (Expr *Src = OVE->getSourceExpr())
28324b9c2d235fb9449e249d74f48ecfec601650de93John McCall      return getDeclFromExpr(Src);
2833db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2834fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (CallExpr *CE = dyn_cast<CallExpr>(E))
2835fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return getDeclFromExpr(CE->getCallee());
28365f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  if (CXXConstructExpr *CE = dyn_cast<CXXConstructExpr>(E))
283793798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    if (!CE->isElidable())
283893798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CE->getConstructor();
2839fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCMessageExpr *OME = dyn_cast<ObjCMessageExpr>(E))
2840fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return OME->getMethodDecl();
2841f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2842db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCProtocolExpr *PE = dyn_cast<ObjCProtocolExpr>(E))
2843db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return PE->getProtocol();
2844c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor  if (SubstNonTypeTemplateParmPackExpr *NTTP
2845c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor                              = dyn_cast<SubstNonTypeTemplateParmPackExpr>(E))
2846c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor    return NTTP->getParameterPack();
284794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
284894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    if (isa<NonTypeTemplateParmDecl>(SizeOfPack->getPack()) ||
284994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        isa<ParmVarDecl>(SizeOfPack->getPack()))
285094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      return SizeOfPack->getPack();
2851db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2852fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  return 0;
2853fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek}
2854ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff
2855c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbarstatic SourceLocation getLocationFromExpr(Expr *E) {
2856c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2857c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis    return getLocationFromExpr(CE->getSubExpr());
2858c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis
2859c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCMessageExpr *Msg = dyn_cast<ObjCMessageExpr>(E))
2860c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return /*FIXME:*/Msg->getLeftLoc();
2861c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E))
2862c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return DRE->getLocation();
2863c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (MemberExpr *Member = dyn_cast<MemberExpr>(E))
2864c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Member->getMemberLoc();
2865c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCIvarRefExpr *Ivar = dyn_cast<ObjCIvarRefExpr>(E))
2866c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Ivar->getLocation();
286794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
286894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    return SizeOfPack->getPackLoc();
286994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
2870c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  return E->getLocStart();
2871c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar}
2872c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar
2873fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
2874f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2875f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekunsigned clang_visitChildren(CXCursor parent,
2876b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXCursorVisitor visitor,
2877b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXClientData client_data) {
2878f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis  CursorVisitor CursorVis(getCursorTU(parent), visitor, client_data,
2879f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                          /*VisitPreprocessorLast=*/false);
2880b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return CursorVis.VisitChildren(parent);
2881b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
2882b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor
28833387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#ifndef __has_feature
28843387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#define __has_feature(x) 0
28853387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
28863387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#if __has_feature(blocks)
28873387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef enum CXChildVisitResult
28883387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall     (^CXCursorVisitorBlock)(CXCursor cursor, CXCursor parent);
28893387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
28903387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
28913387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
28923387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
28933387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block(cursor, parent);
28943387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
28953387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#else
28963387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// If we are compiled with a compiler that doesn't have native blocks support,
28973387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// define and call the block manually, so the
28983387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef struct _CXChildVisitResult
28993387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall{
29003387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	void *isa;
29013387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int flags;
29023387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int reserved;
29039e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar	enum CXChildVisitResult(*invoke)(struct _CXChildVisitResult*, CXCursor,
29049e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                         CXCursor);
29053387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall} *CXCursorVisitorBlock;
29063387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
29073387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
29083387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
29093387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
29103387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block->invoke(block, cursor, parent);
29113387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
29123387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
29133387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
29143387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
29159e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbarunsigned clang_visitChildrenWithBlock(CXCursor parent,
29169e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                      CXCursorVisitorBlock block) {
29173387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return clang_visitChildren(parent, visitWithBlock, block);
29183387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
29193387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
292078205d4bada39d95097e766af9eb30cdd0159461Douglas Gregorstatic CXString getDeclSpelling(Decl *D) {
292116ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis  if (!D)
292216ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    return createCXString("");
292316ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
292416ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis  NamedDecl *ND = dyn_cast<NamedDecl>(D);
2925e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  if (!ND) {
29265f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
2927e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
2928e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return createCXString(Property->getIdentifier()->getName());
2929e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
2930ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
2931e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  }
2932e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
293378205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCMethodDecl *OMD = dyn_cast<ObjCMethodDecl>(ND))
2934ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(OMD->getSelector().getAsString());
2935f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
293678205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCCategoryImplDecl *CIMP = dyn_cast<ObjCCategoryImplDecl>(ND))
293778205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // No, this isn't the same as the code below. getIdentifier() is non-virtual
293878205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // and returns different names. NamedDecl returns the class name and
293978205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // ObjCCategoryImplDecl returns the category name.
2940ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(CIMP->getIdentifier()->getNameStart());
2941f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
29420a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  if (isa<UsingDirectiveDecl>(D))
29430a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("");
29440a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
294536d592718ff342f762e32cbde73d1113f88cb275Dylan Noblesmith  SmallString<1024> S;
294650aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  llvm::raw_svector_ostream os(S);
294750aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  ND->printName(os);
294850aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek
294950aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  return createCXString(os.str());
295078205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor}
2951f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
29529ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getCursorSpelling(CXCursor C) {
29537eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  if (clang_isTranslationUnit(C.kind))
2954a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return clang_getTranslationUnitSpelling(
2955a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                            static_cast<CXTranslationUnit>(C.data[2]));
29567eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
2957f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  if (clang_isReference(C.kind)) {
2958f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    switch (C.kind) {
2959acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCSuperClassRef: {
29602e331b938b38057e333fab0ba841130ea8467794Douglas Gregor      ObjCInterfaceDecl *Super = getCursorObjCSuperClassRef(C).first;
2961ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Super->getIdentifier()->getNameStart());
2962acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
2963acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCClassRef: {
29641adb082a709f7b588f03672999294e061234b2cfDouglas Gregor      ObjCInterfaceDecl *Class = getCursorObjCClassRef(C).first;
2965ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Class->getIdentifier()->getNameStart());
2966acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
2967acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCProtocolRef: {
296878db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor      ObjCProtocolDecl *OID = getCursorObjCProtocolRef(C).first;
2969f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      assert(OID && "getCursorSpelling(): Missing protocol decl");
2970ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(OID->getIdentifier()->getNameStart());
2971acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
29723064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
29733064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = getCursorCXXBaseSpecifier(C);
29743064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return createCXString(B->getType().getAsString());
29753064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
29767d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    case CXCursor_TypeRef: {
29777d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      TypeDecl *Type = getCursorTypeRef(C).first;
29787d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      assert(Type && "Missing type decl");
29797d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
2980ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(getCursorContext(C).getTypeDeclType(Type).
2981ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                              getAsString());
29827d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
29830b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
29840b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      TemplateDecl *Template = getCursorTemplateRef(C).first;
29856931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(Template && "Missing template decl");
29860b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
29870b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString(Template->getNameAsString());
29880b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
29896931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
29906931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
29916931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      NamedDecl *NS = getCursorNamespaceRef(C).first;
29926931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(NS && "Missing namespace decl");
29936931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
29946931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return createCXString(NS->getNameAsString());
29956931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
29967d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
2997a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
2998a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      FieldDecl *Field = getCursorMemberRef(C).first;
2999a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      assert(Field && "Missing member decl");
3000a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3001a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return createCXString(Field->getNameAsString());
3002a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
3003a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
300436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
300536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      LabelStmt *Label = getCursorLabelRef(C).first;
300636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      assert(Label && "Missing label");
300736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
3008ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
300936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
301036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
30111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef: {
30121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
30131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Decl *D = Storage.dyn_cast<Decl *>()) {
30141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        if (NamedDecl *ND = dyn_cast<NamedDecl>(D))
30151f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          return createCXString(ND->getNameAsString());
30161f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
30171f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
30181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
30191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString(E->getName().getAsString());
30201f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedTemplateStorage *Ovl
30211f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        = Storage.get<OverloadedTemplateStorage*>();
30221f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Ovl->size() == 0)
30231f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
30241f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return createCXString((*Ovl->begin())->getNameAsString());
30251f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    }
30261f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3027011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    case CXCursor_VariableRef: {
3028011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      VarDecl *Var = getCursorVariableRef(C).first;
3029011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      assert(Var && "Missing variable decl");
3030011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
3031011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      return createCXString(Var->getNameAsString());
3032011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    }
3033011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
3034acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    default:
3035ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString("<not implemented>");
3036f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    }
3037f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  }
303897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
303997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
304097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    Decl *D = getDeclFromExpr(getCursorExpr(C));
304197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    if (D)
304278205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor      return getDeclSpelling(D);
3043ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
304497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
304597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
304636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
304736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
304836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (LabelStmt *Label = dyn_cast_or_null<LabelStmt>(S))
3049ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
305036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
305136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("");
305236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
305336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
30549b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
30559e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    return createCXString(getCursorMacroExpansion(C)->getName()
30564ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor                                                           ->getNameStart());
30574ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor
3058572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition)
3059572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString(getCursorMacroDefinition(C)->getName()
3060572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor                                                           ->getNameStart());
3061572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3062ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective)
3063ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString(getCursorInclusionDirective(C)->getFileName());
3064ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
306560cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor  if (clang_isDeclaration(C.kind))
306660cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor    return getDeclSpelling(getCursorDecl(C));
3067e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
30685f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  if (C.kind == CXCursor_AnnotateAttr) {
30695f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    AnnotateAttr *AA = cast<AnnotateAttr>(cxcursor::getCursorAttr(C));
30705f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    return createCXString(AA->getAnnotation());
30715f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  }
30725f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen
307384b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis  if (C.kind == CXCursor_AsmLabelAttr) {
307484b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis    AsmLabelAttr *AA = cast<AsmLabelAttr>(cxcursor::getCursorAttr(C));
307584b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis    return createCXString(AA->getLabel());
307684b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis  }
307784b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis
3078ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString("");
3079f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3080f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
3081358559d8d7b458c5f64941842383a16e61f0828dDouglas GregorCXString clang_getCursorDisplayName(CXCursor C) {
3082358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!clang_isDeclaration(C.kind))
3083358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return clang_getCursorSpelling(C);
3084358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3085358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  Decl *D = getCursorDecl(C);
3086358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!D)
3087358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString("");
3088358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
308930c42404202d2e2512e51efc6066bd614cfdb5a4Douglas Gregor  PrintingPolicy Policy = getCursorContext(C).getPrintingPolicy();
3090358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast<FunctionTemplateDecl>(D))
3091358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    D = FunTmpl->getTemplatedDecl();
3092358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3093358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionDecl *Function = dyn_cast<FunctionDecl>(D)) {
309436d592718ff342f762e32cbde73d1113f88cb275Dylan Noblesmith    SmallString<64> Str;
3095358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3096a59d20b135bfde058a5a69045bab5ec4e2553f74Benjamin Kramer    OS << *Function;
3097358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->getPrimaryTemplate())
3098358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "<>";
3099358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "(";
3100358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Function->getNumParams(); I != N; ++I) {
3101358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3102358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3103358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << Function->getParamDecl(I)->getType().getAsString(Policy);
3104358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3105358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3106358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->isVariadic()) {
3107358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Function->getNumParams())
3108358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3109358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "...";
3110358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3111358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ")";
3112358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3113358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3114358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3115358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateDecl *ClassTemplate = dyn_cast<ClassTemplateDecl>(D)) {
311636d592718ff342f762e32cbde73d1113f88cb275Dylan Noblesmith    SmallString<64> Str;
3117358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3118a59d20b135bfde058a5a69045bab5ec4e2553f74Benjamin Kramer    OS << *ClassTemplate;
3119358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "<";
3120358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    TemplateParameterList *Params = ClassTemplate->getTemplateParameters();
3121358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Params->size(); I != N; ++I) {
3122358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3123358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3124358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3125358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      NamedDecl *Param = Params->getParam(I);
3126358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Param->getIdentifier()) {
3127358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << Param->getIdentifier()->getName();
3128358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        continue;
3129358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      }
3130358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3131358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // There is no parameter name, which makes this tricky. Try to come up
3132358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // with something useful that isn't too long.
3133358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(Param))
3134358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << (TTP->wasDeclaredWithTypename()? "typename" : "class");
3135358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else if (NonTypeTemplateParmDecl *NTTP
3136358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                    = dyn_cast<NonTypeTemplateParmDecl>(Param))
3137358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << NTTP->getType().getAsString(Policy);
3138358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else
3139358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << "template<...> class";
3140358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3141358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3142358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ">";
3143358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3144358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3145358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3146358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateSpecializationDecl *ClassSpec
3147358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
3148358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    // If the type was explicitly written, use that.
3149358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (TypeSourceInfo *TSInfo = ClassSpec->getTypeAsWritten())
3150358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      return createCXString(TSInfo->getType().getAsString(Policy));
3151358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
315236d592718ff342f762e32cbde73d1113f88cb275Dylan Noblesmith    SmallString<64> Str;
3153358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3154a59d20b135bfde058a5a69045bab5ec4e2553f74Benjamin Kramer    OS << *ClassSpec;
3155358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << TemplateSpecializationType::PrintTemplateArgumentList(
3156910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().data(),
3157910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().size(),
3158358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                                                Policy);
3159358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3160358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3161358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3162358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  return clang_getCursorSpelling(C);
3163358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor}
3164358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3165e68fff6fc083c6270d835216a3de0b82c6ef0310Ted KremenekCXString clang_getCursorKindSpelling(enum CXCursorKind Kind) {
316689922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  switch (Kind) {
3167e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FunctionDecl:
3168e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FunctionDecl");
3169e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypedefDecl:
3170e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypedefDecl");
3171e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumDecl:
3172e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumDecl");
3173e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumConstantDecl:
3174e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumConstantDecl");
3175e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_StructDecl:
3176e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("StructDecl");
3177e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnionDecl:
3178e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnionDecl");
3179e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ClassDecl:
3180e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ClassDecl");
3181e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FieldDecl:
3182e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FieldDecl");
3183e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_VarDecl:
3184e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("VarDecl");
3185e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ParmDecl:
3186e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ParmDecl");
3187e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInterfaceDecl:
3188e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInterfaceDecl");
3189e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryDecl:
3190e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryDecl");
3191e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolDecl:
3192e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolDecl");
3193e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCPropertyDecl:
3194e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCPropertyDecl");
3195e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCIvarDecl:
3196e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCIvarDecl");
3197e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInstanceMethodDecl:
3198e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInstanceMethodDecl");
3199e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassMethodDecl:
3200e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassMethodDecl");
3201e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCImplementationDecl:
3202e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCImplementationDecl");
3203e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryImplDecl:
3204e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryImplDecl");
32058bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek  case CXCursor_CXXMethod:
32068bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek      return createCXString("CXXMethod");
3207e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedDecl:
3208e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedDecl");
3209e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCSuperClassRef:
3210e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCSuperClassRef");
3211e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolRef:
3212e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolRef");
3213e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassRef:
3214e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassRef");
3215e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypeRef:
3216e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypeRef");
32170b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case CXCursor_TemplateRef:
32180b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString("TemplateRef");
32196931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceRef:
32206931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceRef");
3221a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  case CXCursor_MemberRef:
3222a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return createCXString("MemberRef");
322336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelRef:
322436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("LabelRef");
32251f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case CXCursor_OverloadedDeclRef:
32261f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return createCXString("OverloadedDeclRef");
3227011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor  case CXCursor_VariableRef:
3228011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    return createCXString("VariableRef");
322942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IntegerLiteral:
323042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IntegerLiteral");
323142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_FloatingLiteral:
323242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("FloatingLiteral");
323342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ImaginaryLiteral:
323442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ImaginaryLiteral");
323542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_StringLiteral:
323642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("StringLiteral");
323742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CharacterLiteral:
323842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CharacterLiteral");
323942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ParenExpr:
324042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ParenExpr");
324142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnaryOperator:
324242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnaryOperator");
324342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ArraySubscriptExpr:
324442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ArraySubscriptExpr");
324542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_BinaryOperator:
324642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("BinaryOperator");
324742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundAssignOperator:
324842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundAssignOperator");
324942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ConditionalOperator:
325042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ConditionalOperator");
325142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CStyleCastExpr:
325242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CStyleCastExpr");
325342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundLiteralExpr:
325442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundLiteralExpr");
325542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_InitListExpr:
325642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("InitListExpr");
325742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_AddrLabelExpr:
325842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("AddrLabelExpr");
325942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_StmtExpr:
326042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("StmtExpr");
326142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GenericSelectionExpr:
326242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GenericSelectionExpr");
326342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GNUNullExpr:
326442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GNUNullExpr");
326542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXStaticCastExpr:
326642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXStaticCastExpr");
326742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXDynamicCastExpr:
326842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXDynamicCastExpr");
326942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXReinterpretCastExpr:
327042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXReinterpretCastExpr");
327142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXConstCastExpr:
327242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXConstCastExpr");
327342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXFunctionalCastExpr:
327442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXFunctionalCastExpr");
327542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXTypeidExpr:
327642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXTypeidExpr");
327742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXBoolLiteralExpr:
327842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXBoolLiteralExpr");
327942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXNullPtrLiteralExpr:
328042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXNullPtrLiteralExpr");
328142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXThisExpr:
328242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXThisExpr");
328342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXThrowExpr:
328442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXThrowExpr");
328542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXNewExpr:
328642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXNewExpr");
328742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXDeleteExpr:
328842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXDeleteExpr");
328942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnaryExpr:
329042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnaryExpr");
329142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCStringLiteral:
329242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCStringLiteral");
3293b3f7542a950ac0d585a7783e825cfe670e05c553Ted Kremenek  case CXCursor_ObjCBoolLiteralExpr:
3294b3f7542a950ac0d585a7783e825cfe670e05c553Ted Kremenek      return createCXString("ObjCBoolLiteralExpr");
329542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCEncodeExpr:
329642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCEncodeExpr");
329742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCSelectorExpr:
329842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCSelectorExpr");
329942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCProtocolExpr:
330042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCProtocolExpr");
330142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCBridgedCastExpr:
330242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCBridgedCastExpr");
33031ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek  case CXCursor_BlockExpr:
33041ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek      return createCXString("BlockExpr");
330542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_PackExpansionExpr:
330642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("PackExpansionExpr");
330742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SizeOfPackExpr:
330842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SizeOfPackExpr");
3309011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor  case CXCursor_LambdaExpr:
3310011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    return createCXString("LambdaExpr");
331142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_UnexposedExpr:
331242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("UnexposedExpr");
3313e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_DeclRefExpr:
3314e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("DeclRefExpr");
3315e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_MemberRefExpr:
3316e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("MemberRefExpr");
3317e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_CallExpr:
3318e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("CallExpr");
3319e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCMessageExpr:
3320e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCMessageExpr");
3321e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedStmt:
3322e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedStmt");
332342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DeclStmt:
332442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DeclStmt");
332536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelStmt:
332636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return createCXString("LabelStmt");
332742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CompoundStmt:
332842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CompoundStmt");
332942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CaseStmt:
333042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CaseStmt");
333142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DefaultStmt:
333242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DefaultStmt");
333342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IfStmt:
333442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IfStmt");
333542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SwitchStmt:
333642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SwitchStmt");
333742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_WhileStmt:
333842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("WhileStmt");
333942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_DoStmt:
334042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("DoStmt");
334142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ForStmt:
334242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ForStmt");
334342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_GotoStmt:
334442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("GotoStmt");
334542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_IndirectGotoStmt:
334642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("IndirectGotoStmt");
334742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ContinueStmt:
334842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ContinueStmt");
334942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_BreakStmt:
335042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("BreakStmt");
335142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ReturnStmt:
335242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ReturnStmt");
335342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_AsmStmt:
335442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("AsmStmt");
335542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtTryStmt:
335642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtTryStmt");
335742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtCatchStmt:
335842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtCatchStmt");
335942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtFinallyStmt:
336042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtFinallyStmt");
336142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtThrowStmt:
336242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtThrowStmt");
336342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAtSynchronizedStmt:
336442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAtSynchronizedStmt");
336542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCAutoreleasePoolStmt:
336642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCAutoreleasePoolStmt");
336742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_ObjCForCollectionStmt:
336842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("ObjCForCollectionStmt");
336942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXCatchStmt:
337042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXCatchStmt");
337142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXTryStmt:
337242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXTryStmt");
337342b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_CXXForRangeStmt:
337442b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("CXXForRangeStmt");
337542b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHTryStmt:
337642b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHTryStmt");
337742b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHExceptStmt:
337842b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHExceptStmt");
337942b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_SEHFinallyStmt:
338042b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("SEHFinallyStmt");
338142b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor  case CXCursor_NullStmt:
338242b2984771a7fd1b17c78bbb2c59fed3db2f1960Douglas Gregor      return createCXString("NullStmt");
3383e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_InvalidFile:
3384e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("InvalidFile");
3385292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek  case CXCursor_InvalidCode:
3386292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek    return createCXString("InvalidCode");
3387e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NoDeclFound:
3388e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NoDeclFound");
3389e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NotImplemented:
3390e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NotImplemented");
3391e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TranslationUnit:
3392e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TranslationUnit");
3393e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_UnexposedAttr:
3394e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("UnexposedAttr");
3395e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_IBActionAttr:
3396e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("attribute(ibaction)");
33979f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_IBOutletAttr:
33989f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor     return createCXString("attribute(iboutlet)");
3399857e918a8a40deb128840308a318bf623d68295fTed Kremenek  case CXCursor_IBOutletCollectionAttr:
3400857e918a8a40deb128840308a318bf623d68295fTed Kremenek      return createCXString("attribute(iboutletcollection)");
34016639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXFinalAttr:
34026639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(final)");
34036639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXOverrideAttr:
34046639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(override)");
34055f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen  case CXCursor_AnnotateAttr:
34065f1c822def3efffe1d8f7299fbbbc3b1cdd4833dErik Verbruggen    return createCXString("attribute(annotate)");
340784b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis  case CXCursor_AsmLabelAttr:
340884b796492de8a708150dd3f86ae191041d42eef9Argyrios Kyrtzidis    return createCXString("asm label");
34099f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_PreprocessingDirective:
34109f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return createCXString("preprocessing directive");
3411572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  case CXCursor_MacroDefinition:
3412572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString("macro definition");
34139b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  case CXCursor_MacroExpansion:
34149b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    return createCXString("macro expansion");
3415ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  case CXCursor_InclusionDirective:
3416ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString("inclusion directive");
34178f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  case CXCursor_Namespace:
34188f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek    return createCXString("Namespace");
3419a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  case CXCursor_LinkageSpec:
3420a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek    return createCXString("LinkageSpec");
34213064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  case CXCursor_CXXBaseSpecifier:
34223064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    return createCXString("C++ base class specifier");
342301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Constructor:
342401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConstructor");
342501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Destructor:
342601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXDestructor");
342701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_ConversionFunction:
342801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConversion");
3429fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTypeParameter:
3430fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTypeParameter");
3431fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_NonTypeTemplateParameter:
3432fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("NonTypeTemplateParameter");
3433fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTemplateParameter:
3434fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTemplateParameter");
3435fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_FunctionTemplate:
3436fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("FunctionTemplate");
343739d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  case CXCursor_ClassTemplate:
343839d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return createCXString("ClassTemplate");
343974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  case CXCursor_ClassTemplatePartialSpecialization:
344074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return createCXString("ClassTemplatePartialSpecialization");
34416931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceAlias:
34426931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceAlias");
34430a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  case CXCursor_UsingDirective:
34440a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("UsingDirective");
34457e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  case CXCursor_UsingDeclaration:
34467e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor    return createCXString("UsingDeclaration");
3447162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case CXCursor_TypeAliasDecl:
3448352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("TypeAliasDecl");
3449352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCSynthesizeDecl:
3450352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCSynthesizeDecl");
3451352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCDynamicDecl:
3452352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCDynamicDecl");
34532dfdb948bef51a601e763191e4becfe59880d382Argyrios Kyrtzidis  case CXCursor_CXXAccessSpecifier:
34542dfdb948bef51a601e763191e4becfe59880d382Argyrios Kyrtzidis    return createCXString("CXXAccessSpecifier");
345589922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  }
3456e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3457deb06bd3566e18f677e76bc435d478b033fe328bTed Kremenek  llvm_unreachable("Unhandled CXCursorKind");
3458600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
345989922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
3460064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidisstruct GetCursorData {
3461064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  SourceLocation TokenBeginLoc;
34624b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  bool PointsAtMacroArgExpansion;
3463064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor &BestCursor;
3464064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
34654b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  GetCursorData(SourceManager &SM,
34664b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                SourceLocation tokenBegin, CXCursor &outputCursor)
34674b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    : TokenBeginLoc(tokenBegin), BestCursor(outputCursor) {
34684b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    PointsAtMacroArgExpansion = SM.isMacroArgExpansion(tokenBegin);
34694b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  }
3470064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis};
3471064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
34724b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidisstatic enum CXChildVisitResult GetCursorVisitor(CXCursor cursor,
34734b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXCursor parent,
34744b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXClientData client_data) {
3475064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  GetCursorData *Data = static_cast<GetCursorData *>(client_data);
3476064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor *BestCursor = &Data->BestCursor;
34774b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis
34784b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // If we point inside a macro argument we should provide info of what the
34794b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // token is so use the actual cursor, don't replace it with a macro expansion
34804b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // cursor.
34814b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  if (cursor.kind == CXCursor_MacroExpansion && Data->PointsAtMacroArgExpansion)
34824b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    return CXChildVisit_Recurse;
348365ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis
348465ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis  if (clang_isDeclaration(cursor.kind)) {
348565ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis    // Avoid having the implicit methods override the property decls.
348616ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (ObjCMethodDecl *MD = dyn_cast_or_null<ObjCMethodDecl>(getCursorDecl(cursor)))
348765ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis      if (MD->isImplicit())
348865ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis        return CXChildVisit_Break;
348965ab90736ede3932b26848e39c64396c47f2941bArgyrios Kyrtzidis  }
3490064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
3491064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  if (clang_isExpression(cursor.kind) &&
3492064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis      clang_isDeclaration(BestCursor->kind)) {
349316ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (Decl *D = getCursorDecl(*BestCursor)) {
349416ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      // Avoid having the cursor of an expression replace the declaration cursor
349516ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      // when the expression source range overlaps the declaration range.
349616ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      // This can happen for C++ constructor expressions whose range generally
349716ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      // include the variable declaration, e.g.:
349816ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      //  MyCXXClass foo; // Make sure pointing at 'foo' returns a VarDecl cursor.
349916ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      if (D->getLocation().isValid() && Data->TokenBeginLoc.isValid() &&
350016ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis          D->getLocation() == Data->TokenBeginLoc)
350116ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis        return CXChildVisit_Break;
350216ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    }
3503064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  }
3504064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
350593798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // If our current best cursor is the construction of a temporary object,
350693798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // don't replace that cursor with a type reference, because we want
350793798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // clang_getCursor() to point at the constructor.
350893798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  if (clang_isExpression(BestCursor->kind) &&
350993798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor      isa<CXXTemporaryObjectExpr>(getCursorExpr(*BestCursor)) &&
3510aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      cursor.kind == CXCursor_TypeRef) {
3511aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    // Keep the cursor pointing at CXXTemporaryObjectExpr but also mark it
3512aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    // as having the actual point on the type reference.
3513aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    *BestCursor = getTypeRefedCallExprCursor(*BestCursor);
351493798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CXChildVisit_Recurse;
3515aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis  }
351693798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor
351733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  *BestCursor = cursor;
351833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  return CXChildVisit_Recurse;
351933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
3520e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3521b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXCursor clang_getCursor(CXTranslationUnit TU, CXSourceLocation Loc) {
3522b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!TU)
3523f462989fe8d6f59ab2d7d0fe2b4b96292ce706eaTed Kremenek    return clang_getNullCursor();
3524e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3525a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
3526bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
3527bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor
3528a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek  SourceLocation SLoc = cxloc::translateSourceLocation(Loc);
3529671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  CXCursor Result = cxcursor::getCursor(TU, SLoc);
3530a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek
353140749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  bool Logging = getenv("LIBCLANG_LOGGING");
353240749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  if (Logging) {
353340749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile SearchFile;
353440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned SearchLine, SearchColumn;
353540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile ResultFile;
353640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned ResultLine, ResultColumn;
35376653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    CXString SearchFileName, ResultFileName, KindSpelling, USR;
35386653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    const char *IsDef = clang_isCursorDefinition(Result)? " (Definition)" : "";
353940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXSourceLocation ResultLoc = clang_getCursorLocation(Result);
354040749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
354120174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(Loc, &SearchFile, &SearchLine, &SearchColumn, 0);
354220174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(ResultLoc, &ResultFile, &ResultLine,
354320174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                               &ResultColumn, 0);
354440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    SearchFileName = clang_getFileName(SearchFile);
354540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    ResultFileName = clang_getFileName(ResultFile);
354640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    KindSpelling = clang_getCursorKindSpelling(Result.kind);
35476653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    USR = clang_getCursorUSR(Result);
35486653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    fprintf(stderr, "clang_getCursor(%s:%d:%d) = %s(%s:%d:%d):%s%s\n",
354940749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(SearchFileName), SearchLine, SearchColumn,
355040749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(KindSpelling),
35516653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(ResultFileName), ResultLine, ResultColumn,
35526653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(USR), IsDef);
355340749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(SearchFileName);
355440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(ResultFileName);
355540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(KindSpelling);
35566653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    clang_disposeString(USR);
35570aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor
35580aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    CXCursor Definition = clang_getCursorDefinition(Result);
35590aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    if (!clang_equalCursors(Definition, clang_getNullCursor())) {
35600aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXSourceLocation DefinitionLoc = clang_getCursorLocation(Definition);
35610aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionKindSpelling
35620aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor                                = clang_getCursorKindSpelling(Definition.kind);
35630aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXFile DefinitionFile;
35640aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      unsigned DefinitionLine, DefinitionColumn;
356520174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth      clang_getExpansionLocation(DefinitionLoc, &DefinitionFile,
356620174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                 &DefinitionLine, &DefinitionColumn, 0);
35670aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionFileName = clang_getFileName(DefinitionFile);
35680aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      fprintf(stderr, "  -> %s(%s:%d:%d)\n",
35690aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionKindSpelling),
35700aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionFileName),
35710aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              DefinitionLine, DefinitionColumn);
35720aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionFileName);
35730aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionKindSpelling);
35740aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    }
357540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  }
357640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
3577e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  return Result;
357877128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
357977128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
3580738855554394a6afcf39cc8345fd22c3756b8dd0Ted KremenekCXCursor clang_getNullCursor(void) {
35815bfb8c128c2ac8eb4032afc180cdc400a0f953caDouglas Gregor  return MakeCXCursorInvalid(CXCursor_InvalidFile);
3582738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
3583738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek
3584738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenekunsigned clang_equalCursors(CXCursor X, CXCursor Y) {
3585283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  return X == Y;
3586738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
35870d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
35889ce5584553054d0cb934940586aca0186e87fa57Douglas Gregorunsigned clang_hashCursor(CXCursor C) {
35899ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  unsigned Index = 0;
35909ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  if (clang_isExpression(C.kind) || clang_isStatement(C.kind))
35919ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor    Index = 1;
35929ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35939ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  return llvm::DenseMapInfo<std::pair<unsigned, void*> >::getHashValue(
35949ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor                                        std::make_pair(C.kind, C.data[Index]));
35959ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor}
35969ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35979ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isInvalid(enum CXCursorKind K) {
359877128ddd3077fc045751a55bb3226802b15d5510Steve Naroff  return K >= CXCursor_FirstInvalid && K <= CXCursor_LastInvalid;
359977128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
360077128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
36019ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isDeclaration(enum CXCursorKind K) {
360289922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  return K >= CXCursor_FirstDecl && K <= CXCursor_LastDecl;
360389922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
36042d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff
36059ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isReference(enum CXCursorKind K) {
3606f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  return K >= CXCursor_FirstRef && K <= CXCursor_LastRef;
3607f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3608f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
360997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isExpression(enum CXCursorKind K) {
361097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstExpr && K <= CXCursor_LastExpr;
361197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
361297b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
361397b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isStatement(enum CXCursorKind K) {
361497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstStmt && K <= CXCursor_LastStmt;
361597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
361697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
36178be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregorunsigned clang_isAttribute(enum CXCursorKind K) {
36188be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor    return K >= CXCursor_FirstAttr && K <= CXCursor_LastAttr;
36198be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor}
36208be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor
36217eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregorunsigned clang_isTranslationUnit(enum CXCursorKind K) {
36227eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return K == CXCursor_TranslationUnit;
36237eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
36247eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
36259f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregorunsigned clang_isPreprocessing(enum CXCursorKind K) {
36269f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  return K >= CXCursor_FirstPreprocessing && K <= CXCursor_LastPreprocessing;
36279f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor}
36289f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor
3629ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenekunsigned clang_isUnexposed(enum CXCursorKind K) {
3630ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  switch (K) {
3631ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedDecl:
3632ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedExpr:
3633ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedStmt:
3634ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedAttr:
3635ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return true;
3636ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    default:
3637ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return false;
3638ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  }
3639ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek}
3640ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek
36419ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXCursorKind clang_getCursorKind(CXCursor C) {
36429efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff  return C.kind;
36439efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff}
36449efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff
364598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas GregorCXSourceLocation clang_getCursorLocation(CXCursor C) {
364698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (clang_isReference(C.kind)) {
3647f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    switch (C.kind) {
3648f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCSuperClassRef: {
3649f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3650f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCSuperClassRef(C);
3651a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3652f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3653f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3654f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
3655f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCProtocolDecl *, SourceLocation> P
3656f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCProtocolRef(C);
3657a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3658f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3659f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3660f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCClassRef: {
3661f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3662f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCClassRef(C);
3663a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3664f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
36657d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3666f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef: {
36677d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      std::pair<TypeDecl *, SourceLocation> P = getCursorTypeRef(C);
3668a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36697d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
36700b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
36710b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
36720b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      std::pair<TemplateDecl *, SourceLocation> P = getCursorTemplateRef(C);
36730b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36740b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
36750b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
36766931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
36776931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      std::pair<NamedDecl *, SourceLocation> P = getCursorNamespaceRef(C);
36786931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36796931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
36806931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
3681a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
3682a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      std::pair<FieldDecl *, SourceLocation> P = getCursorMemberRef(C);
3683a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3684a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
3685a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3686011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    case CXCursor_VariableRef: {
3687011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      std::pair<VarDecl *, SourceLocation> P = getCursorVariableRef(C);
3688011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3689011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    }
3690011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
36913064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
36921b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      CXXBaseSpecifier *BaseSpec = getCursorCXXBaseSpecifier(C);
36931b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (!BaseSpec)
36941b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return clang_getNullLocation();
36951b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
36961b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (TypeSourceInfo *TSInfo = BaseSpec->getTypeSourceInfo())
36971b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return cxloc::translateSourceLocation(getCursorContext(C),
36981b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor                                            TSInfo->getTypeLoc().getBeginLoc());
36991b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
37001b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
370196a0014f9b963d8a987f1cccd48808a47f9c6331Daniel Dunbar                                        BaseSpec->getLocStart());
37023064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
3703f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
370436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
370536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      std::pair<LabelStmt *, SourceLocation> P = getCursorLabelRef(C);
370636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C), P.second);
370736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
370836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
37091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
37101f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
37111f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                                          getCursorOverloadedDeclRef(C).second);
37121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3713f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    default:
3714f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3715f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      llvm_unreachable("Missed a reference kind");
3716f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
371798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  }
371897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
371997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3720f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    return cxloc::translateSourceLocation(getCursorContext(C),
372197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor                                   getLocationFromExpr(getCursorExpr(C)));
372297b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
372336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind))
372436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C),
372536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor                                          getCursorStmt(C)->getLocStart());
372636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
37279f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  if (C.kind == CXCursor_PreprocessingDirective) {
37289f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    SourceLocation L = cxcursor::getCursorPreprocessingDirective(C).getBegin();
37299f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
37309f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  }
37314807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
37329b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
37334ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor    SourceLocation L
37349e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth      = cxcursor::getCursorMacroExpansion(C)->getSourceRange().getBegin();
37354807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
37364807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor  }
3737572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3738572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition) {
3739572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    SourceLocation L = cxcursor::getCursorMacroDefinition(C)->getLocation();
3740572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3741572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  }
3742ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3743ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective) {
3744ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    SourceLocation L
3745ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      = cxcursor::getCursorInclusionDirective(C)->getSourceRange().getBegin();
3746ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3747ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  }
3748ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
37499a700d277c38d9afaa7cb3fe93a714bfe9b62eecTed Kremenek  if (C.kind < CXCursor_FirstDecl || C.kind > CXCursor_LastDecl)
37505352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullLocation();
375198258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor
3752f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  Decl *D = getCursorDecl(C);
375316ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis  if (!D)
375416ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    return clang_getNullLocation();
375516ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
3756f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  SourceLocation Loc = D->getLocation();
3757007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // FIXME: Multiple variables declared in a single declaration
3758007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // currently lack the information needed to correctly determine their
3759007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // ranges when accounting for the type-specifier.  We use context
3760007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3761007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // and if so, whether it is the first decl.
3762007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3763007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (!cxcursor::isFirstInDeclGroup(C))
3764007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      Loc = VD->getLocation();
3765007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
3766007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek
37672ca54feee89d7277fb967e3247a64f40ef155a82Douglas Gregor  return cxloc::translateSourceLocation(getCursorContext(C), Loc);
376888145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
3769a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor
3770a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor} // end extern "C"
3771a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3772671436e9e2794c56f3c2e62739d225571493af37Argyrios KyrtzidisCXCursor cxcursor::getCursor(CXTranslationUnit TU, SourceLocation SLoc) {
3773671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  assert(TU);
3774671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3775671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // Guard against an invalid SourceLocation, or we may assert in one
3776671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // of the following calls.
3777671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  if (SLoc.isInvalid())
3778671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    return clang_getNullCursor();
3779671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3780671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
3781671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3782671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // Translate the given source location to make it point at the beginning of
3783671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  // the token under the cursor.
3784671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  SLoc = Lexer::GetBeginningOfToken(SLoc, CXXUnit->getSourceManager(),
37854e4d08403ca5cfd4d558fa2936215d3a4e5a528dDavid Blaikie                                    CXXUnit->getASTContext().getLangOpts());
3786671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3787671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  CXCursor Result = MakeCXCursorInvalid(CXCursor_NoDeclFound);
3788671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  if (SLoc.isValid()) {
3789671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    GetCursorData ResultData(CXXUnit->getSourceManager(), SLoc, Result);
3790671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis    CursorVisitor CursorVis(TU, GetCursorVisitor, &ResultData,
3791671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis                            /*VisitPreprocessorLast=*/true,
3792e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                            /*VisitIncludedEntities=*/false,
3793671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis                            SourceLocation(SLoc));
3794dfb332d0081c6641d1dbae6a2aeff757c99cc740Argyrios Kyrtzidis    CursorVis.visitFileRegion();
3795671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  }
3796671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3797671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis  return Result;
3798671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis}
3799671436e9e2794c56f3c2e62739d225571493af37Argyrios Kyrtzidis
3800a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorstatic SourceRange getRawCursorExtent(CXCursor C) {
3801a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  if (clang_isReference(C.kind)) {
3802a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    switch (C.kind) {
3803a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCSuperClassRef:
3804a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return  getCursorObjCSuperClassRef(C).second;
3805f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3806a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCProtocolRef:
3807a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCProtocolRef(C).second;
3808f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3809a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCClassRef:
3810a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCClassRef(C).second;
38117d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3812a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_TypeRef:
3813a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorTypeRef(C).second;
38140b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
38150b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
38160b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return getCursorTemplateRef(C).second;
38170b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
38186931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
38196931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return getCursorNamespaceRef(C).second;
3820a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3821a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
3822a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return getCursorMemberRef(C).second;
3823a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
38243064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier:
38251b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return getCursorCXXBaseSpecifier(C)->getSourceRange();
3826f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
382736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
382836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return getCursorLabelRef(C).second;
382936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
38301f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
38311f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return getCursorOverloadedDeclRef(C).second;
38321f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3833011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    case CXCursor_VariableRef:
3834011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      return getCursorVariableRef(C).second;
3835011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
3836a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    default:
3837a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3838a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      llvm_unreachable("Missed a reference kind");
3839a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    }
3840a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  }
384197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
384297b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3843a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorExpr(C)->getSourceRange();
384433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
384533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (clang_isStatement(C.kind))
3846a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorStmt(C)->getSourceRange();
3847f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
38486639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  if (clang_isAttribute(C.kind))
38496639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis    return getCursorAttr(C)->getRange();
38506639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
3851a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (C.kind == CXCursor_PreprocessingDirective)
3852a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return cxcursor::getCursorPreprocessingDirective(C);
38534807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
3854ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_MacroExpansion) {
3855ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3856ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorMacroExpansion(C)->getSourceRange();
3857ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3858ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3859572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3860ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_MacroDefinition) {
3861ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3862ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorMacroDefinition(C)->getSourceRange();
3863ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3864ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3865ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3866ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (C.kind == CXCursor_InclusionDirective) {
3867ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    ASTUnit *TU = getCursorASTUnit(C);
3868ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    SourceRange Range = cxcursor::getCursorInclusionDirective(C)->getSourceRange();
3869ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return TU->mapRangeFromPreamble(Range);
3870ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
3871ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3872007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
3873007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    Decl *D = cxcursor::getCursorDecl(C);
387416ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (!D)
387516ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      return SourceRange();
387616ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
3877007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    SourceRange R = D->getSourceRange();
3878007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // FIXME: Multiple variables declared in a single declaration
3879007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // currently lack the information needed to correctly determine their
3880007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // ranges when accounting for the type-specifier.  We use context
3881007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3882007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // and if so, whether it is the first decl.
3883007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3884007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      if (!cxcursor::isFirstInDeclGroup(C))
3885007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek        R.setBegin(VD->getLocation());
3886007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    }
3887007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    return R;
3888007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
38896653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return SourceRange();
38906653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
38916653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38926653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// \brief Retrieves the "raw" cursor extent, which is then extended to include
38936653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// the decl-specifier-seq for declarations.
38946653798ff5ce6deb58112777e21307ccc453133dDouglas Gregorstatic SourceRange getFullCursorExtent(CXCursor C, SourceManager &SrcMgr) {
38956653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
38966653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    Decl *D = cxcursor::getCursorDecl(C);
389716ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (!D)
389816ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      return SourceRange();
389916ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
39006653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange R = D->getSourceRange();
39012494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
39022494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // Adjust the start of the location for declarations preceded by
39032494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // declaration specifiers.
39042494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
39056653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    if (const DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) {
39062494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
390796a0014f9b963d8a987f1cccd48808a47f9c6331Daniel Dunbar        StartLoc = TI->getTypeLoc().getLocStart();
39082494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    } else if (TypedefDecl *Typedef = dyn_cast<TypedefDecl>(D)) {
39092494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
391096a0014f9b963d8a987f1cccd48808a47f9c6331Daniel Dunbar        StartLoc = TI->getTypeLoc().getLocStart();
39112494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    }
39126653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
39132494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && R.getBegin().isValid() &&
39142494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, R.getBegin()))
39152494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      R.setBegin(StartLoc);
39162494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
39172494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // FIXME: Multiple variables declared in a single declaration
39182494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // currently lack the information needed to correctly determine their
39192494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // ranges when accounting for the type-specifier.  We use context
39202494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
39212494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // and if so, whether it is the first decl.
39222494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
39232494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (!cxcursor::isFirstInDeclGroup(C))
39242494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        R.setBegin(VD->getLocation());
39256653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    }
39266653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
39276653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    return R;
39286653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  }
39296653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
39306653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return getRawCursorExtent(C);
39316653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
3932a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3933a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorextern "C" {
3934a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3935a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas GregorCXSourceRange clang_getCursorExtent(CXCursor C) {
3936a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  SourceRange R = getRawCursorExtent(C);
3937a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R.isInvalid())
39385352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
3939f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3940a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  return cxloc::translateSourceRange(getCursorContext(C), R);
3941a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor}
3942c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
3943c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas GregorCXCursor clang_getCursorReferenced(CXCursor C) {
3944b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
3945b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
3946f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3947a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit tu = getCursorTU(C);
39481f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (clang_isDeclaration(C.kind)) {
39491f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getCursorDecl(C);
395016ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (!D)
395116ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis      return clang_getNullCursor();
39521f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
3953a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Using, D->getLocation(), tu);
39545f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
3955e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
3956e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return MakeCXCursor(Property, tu);
3957e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
3958c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return C;
39591f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
39601f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
396197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
39621f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Expr *E = getCursorExpr(C);
39631f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getDeclFromExpr(E);
3964aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    if (D) {
3965aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      CXCursor declCursor = MakeCXCursor(D, tu);
3966aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      declCursor = getSelectorIdentifierCursor(getSelectorIdentifierIndex(C),
3967aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis                                               declCursor);
3968aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis      return declCursor;
3969aed123ec3cc37e457fe20a6158fdadf8849ad916Argyrios Kyrtzidis    }
39701f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
39711f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (OverloadExpr *Ovl = dyn_cast_or_null<OverloadExpr>(E))
3972a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Ovl, tu);
39731f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
397497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    return clang_getNullCursor();
397597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
397697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
397736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
397836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
397936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (GotoStmt *Goto = dyn_cast_or_null<GotoStmt>(S))
398037c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek      if (LabelDecl *label = Goto->getLabel())
398137c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        if (LabelStmt *labelS = label->getStmt())
398237c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        return MakeCXCursor(labelS, getCursorDecl(C), tu);
398336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
398436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return clang_getNullCursor();
398536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
398636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
39879b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
39889e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    if (MacroDefinition *Def = getCursorMacroExpansion(C)->getDefinition())
3989a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeMacroDefinitionCursor(Def, tu);
3990bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor  }
3991bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
3992c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  if (!clang_isReference(C.kind))
3993c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return clang_getNullCursor();
3994f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3995c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  switch (C.kind) {
3996c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    case CXCursor_ObjCSuperClassRef:
3997a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCSuperClassRef(C).first, tu);
3998f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3999f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
400098c16b8b4fe7bb26b17a479d6872e390816e57d4Argyrios Kyrtzidis      ObjCProtocolDecl *Prot = getCursorObjCProtocolRef(C).first;
400198c16b8b4fe7bb26b17a479d6872e390816e57d4Argyrios Kyrtzidis      if (ObjCProtocolDecl *Def = Prot->getDefinition())
400298c16b8b4fe7bb26b17a479d6872e390816e57d4Argyrios Kyrtzidis        return MakeCXCursor(Def, tu);
400398c16b8b4fe7bb26b17a479d6872e390816e57d4Argyrios Kyrtzidis
4004c15707d8da08df2eb22f6ed047743fa3f7c9831bArgyrios Kyrtzidis      return MakeCXCursor(Prot, tu);
400598c16b8b4fe7bb26b17a479d6872e390816e57d4Argyrios Kyrtzidis    }
4006f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
40077723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor    case CXCursor_ObjCClassRef: {
40087723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor      ObjCInterfaceDecl *Class = getCursorObjCClassRef(C).first;
40097723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor      if (ObjCInterfaceDecl *Def = Class->getDefinition())
40107723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor        return MakeCXCursor(Def, tu);
40117723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor
4012c15707d8da08df2eb22f6ed047743fa3f7c9831bArgyrios Kyrtzidis      return MakeCXCursor(Class, tu);
40137723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor    }
40147d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
4015f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef:
4016a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTypeRef(C).first, tu );
40170b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
40180b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
4019a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTemplateRef(C).first, tu );
40200b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
40216931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
4022a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorNamespaceRef(C).first, tu );
40236931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
4024a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
4025a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorMemberRef(C).first, tu );
4026a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
40273064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
40283064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = cxcursor::getCursorCXXBaseSpecifier(C);
40293064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return clang_getTypeDeclaration(cxtype::MakeCXType(B->getType(),
4030a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                                         tu ));
40313064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
4032f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
403336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
403436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // FIXME: We end up faking the "parent" declaration here because we
403536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // don't want to make CXCursor larger.
403636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return MakeCXCursor(getCursorLabelRef(C).first,
4037a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek               static_cast<ASTUnit*>(tu->TUData)->getASTContext()
4038a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          .getTranslationUnitDecl(),
4039a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          tu);
404036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
40411f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
40421f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return C;
4043011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor
4044011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor    case CXCursor_VariableRef:
4045011d8b93b7cfa8492b8a9c909a850d6577e08dcaDouglas Gregor      return MakeCXCursor(getCursorVariableRef(C).first, tu);
40461f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
4047c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    default:
4048c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      // We would prefer to enumerate all non-reference cursor kinds here.
4049c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      llvm_unreachable("Unhandled reference cursor kind");
4050c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  }
4051c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor}
4052c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
4053b699866820102a69d83d6ac6941985c5ef4e8c40Douglas GregorCXCursor clang_getCursorDefinition(CXCursor C) {
4054b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
4055b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
4056f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4057a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(C);
4058f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4059b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  bool WasReference = false;
406097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isReference(C.kind) || clang_isExpression(C.kind)) {
4061b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    C = clang_getCursorReferenced(C);
4062b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    WasReference = true;
4063b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4064b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
40659b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
4066bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor    return clang_getCursorReferenced(C);
4067bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
4068b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
4069b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4070b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4071b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  Decl *D = getCursorDecl(C);
4072b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!D)
4073b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4074f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4075b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  switch (D->getKind()) {
4076b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't really separate the notions of
4077b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // declaration and definition.
4078b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Namespace:
4079b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Typedef:
4080162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case Decl::TypeAlias:
40813e4c6c4c79a03f5cb0c4671d7c282d623c6dc35eRichard Smith  case Decl::TypeAliasTemplate:
4082b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTypeParm:
4083b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::EnumConstant:
4084b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Field:
4085d98114647e16796a976b04af79975b4f0eacf22bBenjamin Kramer  case Decl::IndirectField:
4086b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCIvar:
4087b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCAtDefsField:
4088b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ImplicitParam:
4089b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ParmVar:
4090b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NonTypeTemplateParm:
4091b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTemplateParm:
4092b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategoryImpl:
4093b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCImplementation:
40946206d53f67613958ae1b023aba337ebb46f11a8bAbramo Bagnara  case Decl::AccessSpec:
4095b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::LinkageSpec:
4096b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCPropertyImpl:
4097b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FileScopeAsm:
4098b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::StaticAssert:
4099b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Block:
4100ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  case Decl::Label:  // FIXME: Is this right??
4101af0f4d0b2e38c810effc8b024ad2fb6604eec5d3Francois Pichet  case Decl::ClassScopeFunctionSpecialization:
410215de72cf580840c61e5704c2f8a2b56f9d0638e1Douglas Gregor  case Decl::Import:
4103b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return C;
4104b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4105b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't make any sense here, but are
4106b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // nonetheless harmless.
4107b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TranslationUnit:
4108b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4109b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4110b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds for which the definition is not resolvable.
4111b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingTypename:
4112b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingValue:
4113b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4114b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4115b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingDirective:
4116b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return MakeCXCursor(cast<UsingDirectiveDecl>(D)->getNominatedNamespace(),
4117a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                        TU);
4118b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4119b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NamespaceAlias:
4120a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<NamespaceAliasDecl>(D)->getNamespace(), TU);
4121b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4122b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Enum:
4123b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Record:
4124b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXRecord:
4125b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplateSpecialization:
4126b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplatePartialSpecialization:
4127952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor    if (TagDecl *Def = cast<TagDecl>(D)->getDefinition())
4128a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
4129b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4130b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4131b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Function:
4132b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXMethod:
4133b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConstructor:
4134b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXDestructor:
4135b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConversion: {
4136b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4137b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionDecl>(D)->getBody(Def))
4138a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(const_cast<FunctionDecl *>(Def), TU);
4139b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4140b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4141b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4142b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Var: {
414331310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    // Ask the variable if it has a definition.
414431310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    if (VarDecl *Def = cast<VarDecl>(D)->getDefinition())
4145a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
414631310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    return clang_getNullCursor();
4147b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4148f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4149b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FunctionTemplate: {
4150b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4151b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionTemplateDecl>(D)->getTemplatedDecl()->getBody(Def))
4152a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def->getDescribedFunctionTemplate(), TU);
4153b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4154b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4155f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4156b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplate: {
4157b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (RecordDecl *Def = cast<ClassTemplateDecl>(D)->getTemplatedDecl()
4158952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor                                                            ->getDefinition())
41590b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return MakeCXCursor(cast<CXXRecordDecl>(Def)->getDescribedClassTemplate(),
4160a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          TU);
4161b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4162b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4163b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41641f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::Using:
41651f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return MakeCursorOverloadedDeclRef(cast<UsingDecl>(D),
4166a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       D->getLocation(), TU);
4167b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4168b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingShadow:
4169b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getCursorDefinition(
4170f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                       MakeCXCursor(cast<UsingShadowDecl>(D)->getTargetDecl(),
4171a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                    TU));
4172b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4173b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCMethod: {
4174b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    ObjCMethodDecl *Method = cast<ObjCMethodDecl>(D);
4175b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (Method->isThisDeclarationADefinition())
4176b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      return C;
4177b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4178b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // Dig out the method definition in the associated
4179b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // @implementation, if we have it.
4180b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: The ASTs should make finding the definition easier.
4181b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4182b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                       = dyn_cast<ObjCInterfaceDecl>(Method->getDeclContext()))
4183b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (ObjCImplementationDecl *ClassImpl = Class->getImplementation())
4184b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor        if (ObjCMethodDecl *Def = ClassImpl->getMethod(Method->getSelector(),
4185b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                                                  Method->isInstanceMethod()))
4186b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          if (Def->isThisDeclarationADefinition())
4187a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            return MakeCXCursor(Def, TU);
4188b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4189b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4190b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4191b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4192b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategory:
4193b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCCategoryImplDecl *Impl
4194b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                               = cast<ObjCCategoryDecl>(D)->getImplementation())
4195a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4196b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4197b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4198b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProtocol:
41995e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor    if (ObjCProtocolDecl *Def = cast<ObjCProtocolDecl>(D)->getDefinition())
42005e2a1ff9f28d2eab256d2553e76a9c9d54693875Douglas Gregor      return MakeCXCursor(Def, TU);
4201b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4202b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4203375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor  case Decl::ObjCInterface: {
4204b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // There are two notions of a "definition" for an Objective-C
4205b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // class: the interface and its implementation. When we resolved a
4206b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // reference to an Objective-C class, produce the @interface as
4207b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // the definition; when we were provided with the interface,
4208b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // produce the @implementation as the definition.
4209375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor    ObjCInterfaceDecl *IFace = cast<ObjCInterfaceDecl>(D);
4210b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (WasReference) {
4211375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor      if (ObjCInterfaceDecl *Def = IFace->getDefinition())
42127723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor        return MakeCXCursor(Def, TU);
4213375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor    } else if (ObjCImplementationDecl *Impl = IFace->getImplementation())
4214a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4215b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4216375bb1413c041055262c8a416f20d10474a5eda9Douglas Gregor  }
4217f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4218b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProperty:
4219b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: We don't really know where to find the
4220b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // ObjCPropertyImplDecls that implement this property.
4221b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4222b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4223b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCompatibleAlias:
4224b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4225b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          = cast<ObjCCompatibleAliasDecl>(D)->getClassInterface())
42267723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor      if (ObjCInterfaceDecl *Def = Class->getDefinition())
42277723fec9b45b7258c0eddf4cbfd0d335348f5edcDouglas Gregor        return MakeCXCursor(Def, TU);
4228f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4229b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4230b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4231b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Friend:
4232b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendDecl>(D)->getFriendDecl())
4233a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4234b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4235b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4236b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FriendTemplate:
4237b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendTemplateDecl>(D)->getFriendDecl())
4238a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4239b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4240b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4241b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4242b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getNullCursor();
4243b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4244b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4245b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregorunsigned clang_isCursorDefinition(CXCursor C) {
4246b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
4247b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return 0;
4248b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4249b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getCursorDefinition(C) == C;
4250b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4251b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
42521a9d0503b67a499797141af0fd6d315d5045f0eaDouglas GregorCXCursor clang_getCanonicalCursor(CXCursor C) {
42531a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  if (!clang_isDeclaration(C.kind))
42541a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return C;
42551a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
4256e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  if (Decl *D = getCursorDecl(C)) {
4257debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis    if (ObjCCategoryImplDecl *CatImplD = dyn_cast<ObjCCategoryImplDecl>(D))
4258debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis      if (ObjCCategoryDecl *CatD = CatImplD->getCategoryDecl())
4259debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis        return MakeCXCursor(CatD, getCursorTU(C));
4260debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis
4261e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis    if (ObjCImplDecl *ImplD = dyn_cast<ObjCImplDecl>(D))
4262e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis      if (ObjCInterfaceDecl *IFD = ImplD->getClassInterface())
4263e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis        return MakeCXCursor(IFD, getCursorTU(C));
4264e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis
42651a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return MakeCXCursor(D->getCanonicalDecl(), getCursorTU(C));
4266e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  }
42671a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
42681a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  return C;
42691a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor}
42701a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
42711f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregorunsigned clang_getNumOverloadedDecls(CXCursor C) {
42727c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (C.kind != CXCursor_OverloadedDeclRef)
42731f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return 0;
42741f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42751f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
42761f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
42771f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return E->getNumDecls();
42781f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42791f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
42801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
42811f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return S->size();
42821f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42831f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
42841f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
4285826faa22bae112e01293a58534a40711043cce65Argyrios Kyrtzidis    return Using->shadow_size();
42861f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42871f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return 0;
42881f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
42891f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42901f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas GregorCXCursor clang_getOverloadedDecl(CXCursor cursor, unsigned index) {
42917c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (cursor.kind != CXCursor_OverloadedDeclRef)
42921f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42931f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42941f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (index >= clang_getNumOverloadedDecls(cursor))
42951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42961f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
4297a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(cursor);
42981f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(cursor).first;
42991f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
4300a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(E->decls_begin()[index], TU);
43011f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
43021f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
43031f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
4304a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(S->begin()[index], TU);
43051f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
43061f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
43071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D)) {
43081f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // FIXME: This is, unfortunately, linear time.
43091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    UsingDecl::shadow_iterator Pos = Using->shadow_begin();
43101f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    std::advance(Pos, index);
4311a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<UsingShadowDecl>(*Pos)->getTargetDecl(), TU);
43121f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
43131f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
43141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return clang_getNullCursor();
43151f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
43161f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
43170d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbarvoid clang_getDefinitionSpellingAndExtent(CXCursor C,
43184ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **startBuf,
43194ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **endBuf,
43204ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startLine,
43214ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startColumn,
43224ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *endLine,
43239ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          unsigned *endColumn) {
4324283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  assert(getCursorDecl(C) && "CXCursor has null decl");
4325283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  NamedDecl *ND = static_cast<NamedDecl *>(getCursorDecl(C));
43264ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  FunctionDecl *FD = dyn_cast<FunctionDecl>(ND);
43274ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  CompoundStmt *Body = dyn_cast<CompoundStmt>(FD->getBody());
4328f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
43294ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  SourceManager &SM = FD->getASTContext().getSourceManager();
43304ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startBuf = SM.getCharacterData(Body->getLBracLoc());
43314ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endBuf = SM.getCharacterData(Body->getRBracLoc());
43324ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startLine = SM.getSpellingLineNumber(Body->getLBracLoc());
43334ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startColumn = SM.getSpellingColumnNumber(Body->getLBracLoc());
43344ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endLine = SM.getSpellingLineNumber(Body->getRBracLoc());
43354ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endColumn = SM.getSpellingColumnNumber(Body->getRBracLoc());
43364ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff}
4337f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4338430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4339430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas GregorCXSourceRange clang_getCursorReferenceNameRange(CXCursor C, unsigned NameFlags,
4340430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                unsigned PieceIndex) {
4341430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  RefNamePieces Pieces;
4342430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4343430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  switch (C.kind) {
4344430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_MemberRefExpr:
4345430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (MemberExpr *E = dyn_cast<MemberExpr>(getCursorExpr(C)))
4346430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, true, E->getMemberNameInfo(),
4347430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange());
4348430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4349430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4350430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_DeclRefExpr:
4351430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (DeclRefExpr *E = dyn_cast<DeclRefExpr>(getCursorExpr(C)))
4352430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, false, E->getNameInfo(),
4353430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange(),
4354e4b92761b43ced611c417ae478568610f1ad7b1eAbramo Bagnara                           E->getOptionalExplicitTemplateArgs());
4355430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4356430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4357430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_CallExpr:
4358430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (CXXOperatorCallExpr *OCE =
4359430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        dyn_cast<CXXOperatorCallExpr>(getCursorExpr(C))) {
4360430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Expr *Callee = OCE->getCallee();
4361430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(Callee))
4362430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Callee = ICE->getSubExpr();
4363430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4364430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Callee))
4365430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Pieces = buildPieces(NameFlags, false, DRE->getNameInfo(),
4366430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                             DRE->getQualifierLoc().getSourceRange());
4367430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
4368430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4369430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4370430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  default:
4371430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4372430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4373430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4374430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (Pieces.empty()) {
4375430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (PieceIndex == 0)
4376430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      return clang_getCursorExtent(C);
4377430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  } else if (PieceIndex < Pieces.size()) {
4378430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      SourceRange R = Pieces[PieceIndex];
4379430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (R.isValid())
4380430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        return cxloc::translateSourceRange(getCursorContext(C), R);
4381430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4382430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4383430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  return clang_getNullRange();
4384430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
4385430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
43860a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregorvoid clang_enableStackTraces(void) {
43870a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  llvm::sys::PrintStackTraceOnErrorSignal();
43880a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor}
43890a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor
4390995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbarvoid clang_executeOnThread(void (*fn)(void*), void *user_data,
4391995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar                           unsigned stack_size) {
4392995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar  llvm::llvm_execute_on_thread(fn, user_data, stack_size);
4393995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar}
4394995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar
4395fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
4396fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
4397fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
4398fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor// Token-based Operations.
4399fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
4400fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4401fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor/* CXToken layout:
4402fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[0]: a CXTokenKind
4403fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[1]: starting token location
4404fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[2]: token length
4405fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[3]: reserved
4406f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek *   ptr_data: for identifiers and keywords, an IdentifierInfo*.
4407fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   otherwise unused.
4408fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor */
4409fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorextern "C" {
4410fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4411fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXTokenKind clang_getTokenKind(CXToken CXTok) {
4412fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return static_cast<CXTokenKind>(CXTok.int_data[0]);
4413fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4414fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4415fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXString clang_getTokenSpelling(CXTranslationUnit TU, CXToken CXTok) {
4416fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  switch (clang_getTokenKind(CXTok)) {
4417fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Identifier:
4418fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Keyword:
4419fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We know we have an IdentifierInfo*, so use that.
4420ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(static_cast<IdentifierInfo *>(CXTok.ptr_data)
4421ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                            ->getNameStart());
4422fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4423fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Literal: {
4424fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We have stashed the starting pointer in the ptr_data field. Use it.
4425fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *Text = static_cast<const char *>(CXTok.ptr_data);
44265f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    return createCXString(StringRef(Text, CXTok.int_data[2]));
4427fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4428f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4429fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Punctuation:
4430fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Comment:
4431fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    break;
4432fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4433f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4434f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // We have to find the starting buffer pointer the hard way, by
4435fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // deconstructing the source location.
4436a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4437fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4438ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
4439f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4440fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceLocation Loc = SourceLocation::getFromRawEncoding(CXTok.int_data[1]);
4441fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> LocInfo
4442a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    = CXXUnit->getSourceManager().getDecomposedSpellingLoc(Loc);
4443f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
44445f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4445f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = CXXUnit->getSourceManager().getBufferData(LocInfo.first, &Invalid);
4446f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  if (Invalid)
4447aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor    return createCXString("");
4448fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4449f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  return createCXString(Buffer.substr(LocInfo.second, CXTok.int_data[2]));
4450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4451f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceLocation clang_getTokenLocation(CXTranslationUnit TU, CXToken CXTok) {
4453a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return clang_getNullLocation();
4456f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4457fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return cxloc::translateSourceLocation(CXXUnit->getASTContext(),
4458fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4459fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4460fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4461fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceRange clang_getTokenExtent(CXTranslationUnit TU, CXToken CXTok) {
4462a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
44635352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  if (!CXXUnit)
44645352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
4465f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4466f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return cxloc::translateSourceRange(CXXUnit->getASTContext(),
4467fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4468fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4469f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4470ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisstatic void getTokens(ASTUnit *CXXUnit, SourceRange Range,
4471ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                      SmallVectorImpl<CXToken> &CXTokens) {
4472fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceManager &SourceMgr = CXXUnit->getSourceManager();
4473fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> BeginLocInfo
4474ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(Range.getBegin());
4475fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> EndLocInfo
4476ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(Range.getEnd());
4477f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4478fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Cannot tokenize across files.
4479fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (BeginLocInfo.first != EndLocInfo.first)
4480fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4481f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4482f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Create a lexer
4483f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
44845f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4485f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid);
448647a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor  if (Invalid)
448747a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor    return;
4488aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor
4489fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
44904e4d08403ca5cfd4d558fa2936215d3a4e5a528dDavid Blaikie            CXXUnit->getASTContext().getLangOpts(),
4491f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer            Buffer.begin(), Buffer.data() + BeginLocInfo.second, Buffer.end());
4492fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lex.SetCommentRetentionState(true);
4493f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4494fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Lex tokens until we hit the end of the range.
4495f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  const char *EffectiveBufferEnd = Buffer.data() + EndLocInfo.second;
4496fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Token Tok;
4497096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall  bool previousWasAt = false;
4498fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  do {
4499fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Lex the next token
4500fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    Lex.LexFromRawLexer(Tok);
4501fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.is(tok::eof))
4502fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
4503f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4504fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Initialize the CXToken.
4505fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXToken CXTok;
4506f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4507fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Common fields
4508fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[1] = Tok.getLocation().getRawEncoding();
4509fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[2] = Tok.getLength();
4510fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[3] = 0;
4511f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4512fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Kind-specific fields
4513fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.isLiteral()) {
4514fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Literal;
4515fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = (void *)Tok.getLiteralData();
4516c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara    } else if (Tok.is(tok::raw_identifier)) {
4517aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor      // Lookup the identifier to determine whether we have a keyword.
4518fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      IdentifierInfo *II
4519c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        = CXXUnit->getPreprocessor().LookUpIdentifierInfo(Tok);
4520aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek
4521096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall      if ((II->getObjCKeywordID() != tok::objc_not_keyword) && previousWasAt) {
4522aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek        CXTok.int_data[0] = CXToken_Keyword;
4523aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4524aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      else {
4525c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        CXTok.int_data[0] = Tok.is(tok::identifier)
4526c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          ? CXToken_Identifier
4527c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          : CXToken_Keyword;
4528aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4529fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = II;
4530fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else if (Tok.is(tok::comment)) {
4531fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Comment;
4532fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4533fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else {
4534fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Punctuation;
4535fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4536fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
4537fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTokens.push_back(CXTok);
4538096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall    previousWasAt = Tok.is(tok::at);
4539fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  } while (Lex.getBufferLocation() <= EffectiveBufferEnd);
4540ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis}
4541ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4542ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisvoid clang_tokenize(CXTranslationUnit TU, CXSourceRange Range,
4543ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                    CXToken **Tokens, unsigned *NumTokens) {
4544ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (Tokens)
4545ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    *Tokens = 0;
4546ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (NumTokens)
4547ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    *NumTokens = 0;
4548ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4549ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4550ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (!CXXUnit || !Tokens || !NumTokens)
4551ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4552ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4553ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
4554ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4555ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SourceRange R = cxloc::translateCXSourceRange(Range);
4556ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (R.isInvalid())
4557ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
4558ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
4559ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SmallVector<CXToken, 32> CXTokens;
4560ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  getTokens(CXXUnit, R, CXTokens);
4561f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4562fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (CXTokens.empty())
4563fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4564f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4565fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *Tokens = (CXToken *)malloc(sizeof(CXToken) * CXTokens.size());
4566fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memmove(*Tokens, CXTokens.data(), sizeof(CXToken) * CXTokens.size());
4567fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *NumTokens = CXTokens.size();
4568fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
45690045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
45706db610934bedc6896393c1e1099525b35380acd6Ted Kremenekvoid clang_disposeTokens(CXTranslationUnit TU,
45716db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                         CXToken *Tokens, unsigned NumTokens) {
45726db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  free(Tokens);
45736db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
45746db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45756db610934bedc6896393c1e1099525b35380acd6Ted Kremenek} // end: extern "C"
45766db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45776db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
45786db610934bedc6896393c1e1099525b35380acd6Ted Kremenek// Token annotation APIs.
45796db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
45806db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45810045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregortypedef llvm::DenseMap<unsigned, CXCursor> AnnotateTokensData;
4582fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
4583fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXCursor parent,
4584fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXClientData client_data);
45856db610934bedc6896393c1e1099525b35380acd6Ted Kremeneknamespace {
45866db610934bedc6896393c1e1099525b35380acd6Ted Kremenekclass AnnotateTokensWorker {
45876db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  AnnotateTokensData &Annotated;
458811949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXToken *Tokens;
458911949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXCursor *Cursors;
459011949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  unsigned NumTokens;
4591fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned TokIdx;
45924419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  unsigned PreprocessingTokIdx;
4593fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CursorVisitor AnnotateVis;
4594fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceManager &SrcMgr;
4595f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool HasContextSensitiveKeywords;
4596f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4597fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  bool MoreTokens() const { return TokIdx < NumTokens; }
4598fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned NextToken() const { return TokIdx; }
4599fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void AdvanceToken() { ++TokIdx; }
4600fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation GetTokenLoc(unsigned tokI) {
4601fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4602fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
46035f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  bool isFunctionMacroToken(unsigned tokI) const {
4604a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return Tokens[tokI].int_data[3] != 0;
4605a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
46065f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  SourceLocation getFunctionMacroTokenLoc(unsigned tokI) const {
4607a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[3]);
4608a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4609a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4610a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  void annotateAndAdvanceTokens(CXCursor, RangeComparisonResult, SourceRange);
46115f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void annotateAndAdvanceFunctionMacroTokens(CXCursor, RangeComparisonResult,
46125f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                             SourceRange);
4613fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
46146db610934bedc6896393c1e1099525b35380acd6Ted Kremenekpublic:
461511949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  AnnotateTokensWorker(AnnotateTokensData &annotated,
4616fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                       CXToken *tokens, CXCursor *cursors, unsigned numTokens,
4617a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                       CXTranslationUnit tu, SourceRange RegionOfInterest)
461811949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    : Annotated(annotated), Tokens(tokens), Cursors(cursors),
46194419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      NumTokens(numTokens), TokIdx(0), PreprocessingTokIdx(0),
4620a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      AnnotateVis(tu,
4621f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  AnnotateTokensVisitor, this,
4622f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  /*VisitPreprocessorLast=*/true,
4623e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                  /*VisitIncludedEntities=*/false,
4624f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                  RegionOfInterest),
4625f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      SrcMgr(static_cast<ASTUnit*>(tu->TUData)->getSourceManager()),
4626f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      HasContextSensitiveKeywords(false) { }
462711949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4628fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void VisitChildren(CXCursor C) { AnnotateVis.VisitChildren(C); }
46296db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  enum CXChildVisitResult Visit(CXCursor cursor, CXCursor parent);
463003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  void AnnotateTokens();
4631f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4632f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// \brief Determine whether the annotator saw any cursors that have
4633f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// context-sensitive keywords.
4634f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool hasContextSensitiveKeywords() const {
4635f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    return HasContextSensitiveKeywords;
4636f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
46376db610934bedc6896393c1e1099525b35380acd6Ted Kremenek};
46386db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
46390045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
464003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidisvoid AnnotateTokensWorker::AnnotateTokens() {
4641fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Walk the AST within the region of interest, annotating tokens
4642fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // along the way.
464303ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis  AnnotateVis.visitFileRegion();
4644fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4645fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = 0 ; I < TokIdx ; ++I) {
464611949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
46474419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    if (Pos != Annotated.end() &&
46484419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        (clang_isInvalid(Cursors[I].kind) ||
46494419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor         Pos->second.kind != CXCursor_PreprocessingDirective))
4650fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      Cursors[I] = Pos->second;
4651fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4652fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4653fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Finish up annotating any tokens left.
4654fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (!MoreTokens())
4655fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return;
465611949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4657fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor &C = clang_getNullCursor();
4658fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = TokIdx ; I < NumTokens ; ++I) {
465903ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis    if (I < PreprocessingTokIdx && clang_isPreprocessing(Cursors[I].kind))
466003ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis      continue;
466103ee2dd9fc5d5fc62b5eb0fb88ee56e553f8cda7Argyrios Kyrtzidis
4662fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
4663fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = (Pos == Annotated.end()) ? C : Pos->second;
466411949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  }
466511949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek}
466611949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4667a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief It annotates and advances tokens with a cursor until the comparison
4668a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis//// between the cursor location and the source range is the same as
4669a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \arg compResult.
4670a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis///
4671a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeBefore to annotate tokens with a cursor until a range is reached.
4672a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeOverlap to annotate tokens inside a range.
4673a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceTokens(CXCursor updateC,
4674a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4675a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
4676a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  while (MoreTokens()) {
4677a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    const unsigned I = NextToken();
46785f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    if (isFunctionMacroToken(I))
46795f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      return annotateAndAdvanceFunctionMacroTokens(updateC, compResult, range);
4680a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4681a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceLocation TokLoc = GetTokenLoc(I);
4682a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4683a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      Cursors[I] = updateC;
4684a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      AdvanceToken();
4685a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue;
4686a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4687a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    break;
4688a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4689a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4690a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4691a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Special annotation handling for macro argument tokens.
46925f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceFunctionMacroTokens(
46935f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                               CXCursor updateC,
4694a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4695a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
46965f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(MoreTokens());
46975f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(isFunctionMacroToken(NextToken()) &&
4698a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis         "Should be called only for macro arg tokens");
4699a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4700a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // This works differently than annotateAndAdvanceTokens; because expanded
4701a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // macro arguments can have arbitrary translation-unit source order, we do not
4702a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // advance the token index one by one until a token fails the range test.
4703a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // We only advance once past all of the macro arg tokens if all of them
4704a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // pass the range test. If one of them fails we keep the token index pointing
4705a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // at the start of the macro arg tokens so that the failing token will be
4706a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // annotated by a subsequent annotation try.
4707a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4708a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  bool atLeastOneCompFail = false;
4709a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4710a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned I = NextToken();
47115f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  for (; I < NumTokens && isFunctionMacroToken(I); ++I) {
47125f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    SourceLocation TokLoc = getFunctionMacroTokenLoc(I);
4713a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (TokLoc.isFileID())
4714a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue; // not macro arg token, it's parens or comma.
4715a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4716a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (clang_isInvalid(clang_getCursorKind(Cursors[I])))
4717a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        Cursors[I] = updateC;
4718a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    } else
4719a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      atLeastOneCompFail = true;
4720a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4721a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4722a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (!atLeastOneCompFail)
4723a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    TokIdx = I; // All of the tokens were handled, advance beyond all of them.
4724a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4725a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
47266db610934bedc6896393c1e1099525b35380acd6Ted Kremenekenum CXChildVisitResult
47274419b675577d7c281a659fab1fec10e1bfbe04c5Douglas GregorAnnotateTokensWorker::Visit(CXCursor cursor, CXCursor parent) {
4728fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CXSourceLocation Loc = clang_getCursorLocation(cursor);
47294419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  SourceRange cursorRange = getRawCursorExtent(cursor);
473081d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor  if (cursorRange.isInvalid())
473181d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor    return CXChildVisit_Recurse;
4732f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4733f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (!HasContextSensitiveKeywords) {
4734f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C properties can have context-sensitive keywords.
4735f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    if (cursor.kind == CXCursor_ObjCPropertyDecl) {
4736f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCPropertyDecl *Property
4737f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(cursor)))
4738f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        HasContextSensitiveKeywords = Property->getPropertyAttributesAsWritten() != 0;
4739f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4740f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C methods can have context-sensitive keywords.
4741f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_ObjCInstanceMethodDecl ||
4742f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ObjCClassMethodDecl) {
4743f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCMethodDecl *Method
4744f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            = dyn_cast_or_null<ObjCMethodDecl>(getCursorDecl(cursor))) {
4745f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->getObjCDeclQualifier())
4746f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4747f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        else {
4748f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          for (ObjCMethodDecl::param_iterator P = Method->param_begin(),
4749f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                                           PEnd = Method->param_end();
4750f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor               P != PEnd; ++P) {
4751f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            if ((*P)->getObjCDeclQualifier()) {
4752f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              HasContextSensitiveKeywords = true;
4753f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              break;
4754f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            }
4755f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          }
4756f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
4757f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4758f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4759f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ methods can have context-sensitive keywords.
4760f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_CXXMethod) {
4761f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (CXXMethodDecl *Method
4762f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<CXXMethodDecl>(getCursorDecl(cursor))) {
4763f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->hasAttr<FinalAttr>() || Method->hasAttr<OverrideAttr>())
4764f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4765f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4766f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4767f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ classes can have context-sensitive keywords.
4768f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_StructDecl ||
4769f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassDecl ||
4770f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplate ||
4771f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplatePartialSpecialization) {
4772f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Decl *D = getCursorDecl(cursor))
4773f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (D->hasAttr<FinalAttr>())
4774f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4775f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4776f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
4777f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
47784419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  if (clang_isPreprocessing(cursor.kind)) {
4779cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // For macro expansions, just note where the beginning of the macro
4780cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // expansion occurs.
47819b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    if (cursor.kind == CXCursor_MacroExpansion) {
47824419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      Annotated[Loc.int_data] = cursor;
47834419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      return CXChildVisit_Recurse;
47844419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47854419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47864419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Items in the preprocessing record are kept separate from items in
47874419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // declarations, so we keep a separate token index.
47884419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    unsigned SavedTokIdx = TokIdx;
47894419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = PreprocessingTokIdx;
47904419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47914419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Skip tokens up until we catch up to the beginning of the preprocessing
47924419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // entry.
47934419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
47944419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
47954419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
47964419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
47974419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
47984419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
47994419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
48004419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
48014419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
48024419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
48034419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
48044419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
48054419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
48064419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
48074419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Look at all of the tokens within this range.
48084419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
48094419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
48104419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
48114419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
48124419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
4813b219cfc4d75f0a03630b7c4509ef791b7e97b2c8David Blaikie        llvm_unreachable("Infeasible");
48144419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
48154419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
48164419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
48174419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        Cursors[I] = cursor;
48184419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
48194419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
48204419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
48214419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
48224419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
48234419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
48244419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Save the preprocessing token index; restore the non-preprocessing
48254419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // token index.
48264419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    PreprocessingTokIdx = TokIdx;
48274419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = SavedTokIdx;
48280045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    return CXChildVisit_Recurse;
48290045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  }
4830fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4831fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (cursorRange.isInvalid())
4832fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return CXChildVisit_Continue;
4833a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek
4834fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation L = SourceLocation::getFromRawEncoding(Loc.int_data);
4835fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4836a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  // Adjust the annotated range based specific declarations.
4837a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  const enum CXCursorKind cursorK = clang_getCursorKind(cursor);
4838a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  if (cursorK >= CXCursor_FirstDecl && cursorK <= CXCursor_LastDecl) {
483923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    Decl *D = cxcursor::getCursorDecl(cursor);
48402494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
48412494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
484216ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    if (const DeclaratorDecl *DD = dyn_cast_or_null<DeclaratorDecl>(D)) {
48432494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
484496a0014f9b963d8a987f1cccd48808a47f9c6331Daniel Dunbar        StartLoc = TI->getTypeLoc().getLocStart();
484516ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    } else if (TypedefDecl *Typedef = dyn_cast_or_null<TypedefDecl>(D)) {
48462494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
484796a0014f9b963d8a987f1cccd48808a47f9c6331Daniel Dunbar        StartLoc = TI->getTypeLoc().getLocStart();
4848a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek    }
48492494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
48502494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && L.isValid() &&
48512494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, L))
48522494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      cursorRange.setBegin(StartLoc);
4853a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  }
485481d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor
48553f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // If the location of the cursor occurs within a macro instantiation, record
48563f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // the spelling location of the cursor in our annotation map.  We can then
48573f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // paper over the token labelings during a post-processing step to try and
48583f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // get cursor mappings for tokens that are the *arguments* of a macro
48593f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // instantiation.
48603f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  if (L.isMacroID()) {
48613f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    unsigned rawEncoding = SrcMgr.getSpellingLoc(L).getRawEncoding();
48623f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // Only invalidate the old annotation if it isn't part of a preprocessing
48633f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // directive.  Here we assume that the default construction of CXCursor
48643f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // results in CXCursor.kind being an initialized value (i.e., 0).  If
48653f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // this isn't the case, we can fix by doing lookup + insertion.
48664419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
48673f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    CXCursor &oldC = Annotated[rawEncoding];
48683f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    if (!clang_isPreprocessing(oldC.kind))
48693f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek      oldC = cursor;
48703f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  }
48713f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek
4872fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const enum CXCursorKind K = clang_getCursorKind(parent);
4873fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor updateC =
4874d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek    (clang_isInvalid(K) || K == CXCursor_TranslationUnit)
4875d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek     ? clang_getNullCursor() : parent;
4876fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4877a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(updateC, RangeBefore, cursorRange);
4878fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
48795517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // Avoid having the cursor of an expression "overwrite" the annotation of the
48805517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // variable declaration that it belongs to.
48815517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // This can happen for C++ constructor expressions whose range generally
48825517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // include the variable declaration, e.g.:
48835517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  //  MyCXXClass foo; // Make sure we don't annotate 'foo' as a CallExpr cursor.
48845517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  if (clang_isExpression(cursorK)) {
48855517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    Expr *E = getCursorExpr(cursor);
48868ccac3de1335f1cfd7cea56ba1cefcf0b724ce3fArgyrios Kyrtzidis    if (Decl *D = getCursorParentDecl(cursor)) {
48875517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      const unsigned I = NextToken();
48885517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      if (E->getLocStart().isValid() && D->getLocation().isValid() &&
48895517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == D->getLocation() &&
48905517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == GetTokenLoc(I)) {
48915517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        Cursors[I] = updateC;
48925517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        AdvanceToken();
48935517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      }
48945517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    }
48955517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  }
48965517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis
4897fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Visit children to get their cursor information.
4898fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned BeforeChildren = NextToken();
4899fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  VisitChildren(cursor);
4900fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned AfterChildren = NextToken();
4901fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4902a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // Scan the tokens that are at the end of the cursor, but are not captured
4903a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // but the child cursors.
4904a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(cursor, RangeOverlap, cursorRange);
49056db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
4906fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Scan the tokens that are at the beginning of the cursor, but are not
4907fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // capture by the child cursors.
4908fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = BeforeChildren; I != AfterChildren; ++I) {
4909fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    if (!clang_isInvalid(clang_getCursorKind(Cursors[I])))
4910fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      break;
49114419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
4912fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = cursor;
4913fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4914fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4915fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  return CXChildVisit_Continue;
49160045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor}
49170045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
49186db610934bedc6896393c1e1099525b35380acd6Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
49196db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXCursor parent,
49206db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXClientData client_data) {
49216db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  return static_cast<AnnotateTokensWorker*>(client_data)->Visit(cursor, parent);
49226db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
49236db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
49246628a614c504263ae539462f049d523dd07ac1baTed Kremeneknamespace {
4925a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4926a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Uses the macro expansions in the preprocessing record to find
4927a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// and mark tokens that are macro arguments. This info is used by the
4928a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// AnnotateTokensWorker.
4929a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisclass MarkMacroArgTokensVisitor {
4930a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceManager &SM;
4931a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXToken *Tokens;
4932a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned NumTokens;
4933a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned CurIdx;
4934a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4935a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidispublic:
4936a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  MarkMacroArgTokensVisitor(SourceManager &SM,
4937a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                            CXToken *tokens, unsigned numTokens)
4938a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    : SM(SM), Tokens(tokens), NumTokens(numTokens), CurIdx(0) { }
4939a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4940a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXChildVisitResult visit(CXCursor cursor, CXCursor parent) {
4941a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (cursor.kind != CXCursor_MacroExpansion)
4942a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue;
4943a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4944a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceRange macroRange = getCursorMacroExpansion(cursor)->getSourceRange();
4945a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (macroRange.getBegin() == macroRange.getEnd())
4946a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue; // it's not a function macro.
4947a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4948a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4949a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(getTokenLoc(CurIdx),
4950a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                        macroRange.getBegin()))
4951a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4952a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4953a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4954a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4955a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4956a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4957a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4958a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      SourceLocation tokLoc = getTokenLoc(CurIdx);
4959a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(tokLoc, macroRange.getEnd()))
4960a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4961a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
49625f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      setFunctionMacroTokenLoc(CurIdx, SM.getMacroArgExpandedLocation(tokLoc));
4963a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4964a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4965a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4966a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4967a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4968a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return CXChildVisit_Continue;
4969a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4970a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4971a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisprivate:
4972a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceLocation getTokenLoc(unsigned tokI) {
4973a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4974a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4975a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
49765f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void setFunctionMacroTokenLoc(unsigned tokI, SourceLocation loc) {
4977a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // The third field is reserved and currently not used. Use it here
4978a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // to mark macro arg expanded tokens with their expanded locations.
4979a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    Tokens[tokI].int_data[3] = loc.getRawEncoding();
4980a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4981a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis};
4982a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4983a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis} // end anonymous namespace
4984a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4985a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisstatic CXChildVisitResult
4986a676379b26edc959193f9f919ba9c6d296a57824Argyrios KyrtzidisMarkMacroArgTokensVisitorDelegate(CXCursor cursor, CXCursor parent,
4987a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                  CXClientData client_data) {
4988a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  return static_cast<MarkMacroArgTokensVisitor*>(client_data)->visit(cursor,
4989a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                                                     parent);
4990a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4991a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4992a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisnamespace {
49936628a614c504263ae539462f049d523dd07ac1baTed Kremenek  struct clang_annotateTokens_Data {
49946628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXTranslationUnit TU;
49956628a614c504263ae539462f049d523dd07ac1baTed Kremenek    ASTUnit *CXXUnit;
49966628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXToken *Tokens;
49976628a614c504263ae539462f049d523dd07ac1baTed Kremenek    unsigned NumTokens;
49986628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXCursor *Cursors;
49996628a614c504263ae539462f049d523dd07ac1baTed Kremenek  };
5000ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek}
5001ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek
5002ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidisstatic void annotatePreprocessorTokens(CXTranslationUnit TU,
5003ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       SourceRange RegionOfInterest,
5004ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                       AnnotateTokensData &Annotated) {
5005ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
5006ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5007ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  SourceManager &SourceMgr = CXXUnit->getSourceManager();
5008ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  std::pair<FileID, unsigned> BeginLocInfo
5009ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(RegionOfInterest.getBegin());
5010ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  std::pair<FileID, unsigned> EndLocInfo
5011ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    = SourceMgr.getDecomposedLoc(RegionOfInterest.getEnd());
5012ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5013ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (BeginLocInfo.first != EndLocInfo.first)
5014ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
5015ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5016ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  StringRef Buffer;
5017ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  bool Invalid = false;
5018ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Buffer = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid);
5019ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  if (Buffer.empty() || Invalid)
5020ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    return;
5021ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5022ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
50234e4d08403ca5cfd4d558fa2936215d3a4e5a528dDavid Blaikie            CXXUnit->getASTContext().getLangOpts(),
5024ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis            Buffer.begin(), Buffer.data() + BeginLocInfo.second,
5025ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis            Buffer.end());
5026ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  Lex.SetCommentRetentionState(true);
5027ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5028ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  // Lex tokens in raw mode until we hit the end of the range, to avoid
5029ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  // entering #includes or expanding macros.
5030ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  while (true) {
5031ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    Token Tok;
5032ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    Lex.LexFromRawLexer(Tok);
5033ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5034ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  reprocess:
5035ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    if (Tok.is(tok::hash) && Tok.isAtStartOfLine()) {
5036ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // We have found a preprocessing directive. Gobble it up so that we
5037ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // don't see it while preprocessing these tokens later, but keep track
5038ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // of all of the token locations inside this preprocessing directive so
5039ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // that we can annotate them appropriately.
5040ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      //
5041ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // FIXME: Some simple tests here could identify macro definitions and
5042ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      // #undefs, to provide specific cursor kinds for those.
5043ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      SmallVector<SourceLocation, 32> Locations;
5044ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      do {
5045ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Locations.push_back(Tok.getLocation());
5046ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Lex.LexFromRawLexer(Tok);
5047ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      } while (!Tok.isAtStartOfLine() && !Tok.is(tok::eof));
5048ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5049ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      using namespace cxcursor;
5050ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      CXCursor Cursor
5051ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      = MakePreprocessingDirectiveCursor(SourceRange(Locations.front(),
5052ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                                     Locations.back()),
5053ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis                                         TU);
5054ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      for (unsigned I = 0, N = Locations.size(); I != N; ++I) {
5055ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        Annotated[Locations[I].getRawEncoding()] = Cursor;
5056ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      }
5057ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5058ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      if (Tok.isAtStartOfLine())
5059ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis        goto reprocess;
5060ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5061ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      continue;
5062ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    }
5063ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5064ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis    if (Tok.is(tok::eof))
5065ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis      break;
5066ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  }
5067ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis}
5068ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
50696628a614c504263ae539462f049d523dd07ac1baTed Kremenek// This gets run a separate thread to avoid stack blowout.
50706628a614c504263ae539462f049d523dd07ac1baTed Kremenekstatic void clang_annotateTokensImpl(void *UserData) {
50716628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXTranslationUnit TU = ((clang_annotateTokens_Data*)UserData)->TU;
50726628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = ((clang_annotateTokens_Data*)UserData)->CXXUnit;
50736628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXToken *Tokens = ((clang_annotateTokens_Data*)UserData)->Tokens;
50746628a614c504263ae539462f049d523dd07ac1baTed Kremenek  const unsigned NumTokens = ((clang_annotateTokens_Data*)UserData)->NumTokens;
50756628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor *Cursors = ((clang_annotateTokens_Data*)UserData)->Cursors;
5076fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
50770396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Determine the region of interest, which contains all of the tokens.
50780045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  SourceRange RegionOfInterest;
50796628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setBegin(
50806628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU, Tokens[0])));
50816628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setEnd(
50826628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU,
50836628a614c504263ae539462f049d523dd07ac1baTed Kremenek                                                         Tokens[NumTokens-1])));
5084fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
50850396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // A mapping from the source locations found when re-lexing or traversing the
50860396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // region of interest to the corresponding cursors.
50870045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  AnnotateTokensData Annotated;
5088ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis
5089fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Relex the tokens within the source range to look for preprocessing
50900396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // directives.
5091ee0f84fc84ed7de7975e102668d8e53a778f7a8cArgyrios Kyrtzidis  annotatePreprocessorTokens(TU, RegionOfInterest, Annotated);
50926628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5093a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (CXXUnit->getPreprocessor().getPreprocessingRecord()) {
5094a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // Search and mark tokens that are macro argument expansions.
5095a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MarkMacroArgTokensVisitor Visitor(CXXUnit->getSourceManager(),
5096a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                      Tokens, NumTokens);
5097a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    CursorVisitor MacroArgMarker(TU,
5098a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                 MarkMacroArgTokensVisitorDelegate, &Visitor,
5099f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                 /*VisitPreprocessorLast=*/true,
5100e70984629f3accf7e1e7187d06bd653dc8e315f2Argyrios Kyrtzidis                                 /*VisitIncludedEntities=*/false,
5101f226ff9fe8c8db6c5b74a61ce649eda1491c3502Argyrios Kyrtzidis                                 RegionOfInterest);
5102a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MacroArgMarker.visitPreprocessedEntitiesInRegion();
5103a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
5104a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
51050396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Annotate all of the source locations in the region of interest that map to
5106fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // a specific cursor.
5107fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  AnnotateTokensWorker W(Annotated, Tokens, Cursors, NumTokens,
5108a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                         TU, RegionOfInterest);
51096628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51106c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // FIXME: We use a ridiculous stack size here because the data-recursion
51116c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm uses a large stack frame than the non-data recursive version,
51126c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // and AnnotationTokensWorker currently transforms the data-recursion
51136c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm back into a traditional recursion by explicitly calling
51146c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // VisitChildren().  We will need to remove this explicit recursive call.
51156628a614c504263ae539462f049d523dd07ac1baTed Kremenek  W.AnnotateTokens();
51166628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5117f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // If we ran into any entities that involve context-sensitive keywords,
5118f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // take another pass through the tokens to mark them as such.
5119f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (W.hasContextSensitiveKeywords()) {
5120f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    for (unsigned I = 0; I != NumTokens; ++I) {
5121f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (clang_getTokenKind(Tokens[I]) != CXToken_Identifier)
5122f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5123f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5124f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCPropertyDecl) {
5125f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5126f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (ObjCPropertyDecl *Property
51276628a614c504263ae539462f049d523dd07ac1baTed Kremenek            = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(Cursors[I]))) {
5128f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          if (Property->getPropertyAttributesAsWritten() != 0 &&
5129f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              llvm::StringSwitch<bool>(II->getName())
51306628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readonly", true)
51316628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("assign", true)
5132f85e193739c953358c865005855253af4f68a497John McCall              .Case("unsafe_unretained", true)
51336628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readwrite", true)
51346628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("retain", true)
51356628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("copy", true)
51366628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("nonatomic", true)
51376628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("atomic", true)
51386628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("getter", true)
51396628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("setter", true)
5140f85e193739c953358c865005855253af4f68a497John McCall              .Case("strong", true)
5141f85e193739c953358c865005855253af4f68a497John McCall              .Case("weak", true)
51426628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Default(false))
5143f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            Tokens[I].int_data[0] = CXToken_Keyword;
5144f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
5145f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5146f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5147f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5148f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCInstanceMethodDecl ||
5149f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Cursors[I].kind == CXCursor_ObjCClassMethodDecl) {
5150f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5151f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (llvm::StringSwitch<bool>(II->getName())
51526628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("in", true)
51536628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("out", true)
51546628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("inout", true)
51556628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("oneway", true)
51566628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("bycopy", true)
51576628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("byref", true)
51586628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Default(false))
5159f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Tokens[I].int_data[0] = CXToken_Keyword;
5160f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5161f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
51626639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
51636639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      if (Cursors[I].kind == CXCursor_CXXFinalAttr ||
51646639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis          Cursors[I].kind == CXCursor_CXXOverrideAttr) {
51656639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis        Tokens[I].int_data[0] = CXToken_Keyword;
51666628a614c504263ae539462f049d523dd07ac1baTed Kremenek        continue;
5167f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5168f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
5169f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
5170fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
51716628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51726628a614c504263ae539462f049d523dd07ac1baTed Kremenekextern "C" {
51736628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51746628a614c504263ae539462f049d523dd07ac1baTed Kremenekvoid clang_annotateTokens(CXTranslationUnit TU,
51756628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXToken *Tokens, unsigned NumTokens,
51766628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXCursor *Cursors) {
51776628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51786628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (NumTokens == 0 || !Tokens || !Cursors)
51796628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
51806628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51816628a614c504263ae539462f049d523dd07ac1baTed Kremenek  // Any token we don't specifically annotate will have a NULL cursor.
51826628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor C = clang_getNullCursor();
51836628a614c504263ae539462f049d523dd07ac1baTed Kremenek  for (unsigned I = 0; I != NumTokens; ++I)
51846628a614c504263ae539462f049d523dd07ac1baTed Kremenek    Cursors[I] = C;
51856628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51866628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
51876628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!CXXUnit)
51886628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
51896628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51906628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
51916628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51926628a614c504263ae539462f049d523dd07ac1baTed Kremenek  clang_annotateTokens_Data data = { TU, CXXUnit, Tokens, NumTokens, Cursors };
51936628a614c504263ae539462f049d523dd07ac1baTed Kremenek  llvm::CrashRecoveryContext CRC;
51946628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!RunSafely(CRC, clang_annotateTokensImpl, &data,
51956628a614c504263ae539462f049d523dd07ac1baTed Kremenek                 GetSafetyThreadStackSize() * 2)) {
51966628a614c504263ae539462f049d523dd07ac1baTed Kremenek    fprintf(stderr, "libclang: crash detected while annotating tokens\n");
51976628a614c504263ae539462f049d523dd07ac1baTed Kremenek  }
51986628a614c504263ae539462f049d523dd07ac1baTed Kremenek}
51996628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5200fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor} // end: extern "C"
5201fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
5202fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
520316b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek// Operations for querying linkage of a cursor.
520416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
520516b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
520616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenekextern "C" {
520716b4259aecaa22b642d35d36fd89965ed700c1e0Ted KremenekCXLinkageKind clang_getCursorLinkage(CXCursor cursor) {
52080396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  if (!clang_isDeclaration(cursor.kind))
52090396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    return CXLinkage_Invalid;
52100396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor
521116b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  Decl *D = cxcursor::getCursorDecl(cursor);
521216b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  if (NamedDecl *ND = dyn_cast_or_null<NamedDecl>(D))
521316b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    switch (ND->getLinkage()) {
521416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case NoLinkage: return CXLinkage_NoLinkage;
521516b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case InternalLinkage: return CXLinkage_Internal;
521616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case UniqueExternalLinkage: return CXLinkage_UniqueExternal;
521716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case ExternalLinkage: return CXLinkage_External;
521816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    };
521916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
522016b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  return CXLinkage_Invalid;
522116b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek}
522216b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek} // end: extern "C"
522316b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
522416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
522545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek// Operations for querying language of a cursor.
522645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
522745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
522845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekstatic CXLanguageKind getDeclLanguage(const Decl *D) {
522916ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis  if (!D)
523016ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis    return CXLanguage_C;
523116ed0e6fda74800595cb98d11969fc4afcc00bacArgyrios Kyrtzidis
523245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  switch (D->getKind()) {
523345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    default:
523445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      break;
523545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ImplicitParam:
523645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCAtDefsField:
523745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategory:
523845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategoryImpl:
523945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCompatibleAlias:
524045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCImplementation:
524145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCInterface:
524245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCIvar:
524345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCMethod:
524445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProperty:
524545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCPropertyImpl:
524645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProtocol:
524745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_ObjC;
524845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConstructor:
524945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConversion:
525045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXDestructor:
525145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXMethod:
525245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXRecord:
525345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplate:
525445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplatePartialSpecialization:
525545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplateSpecialization:
525645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Friend:
525745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FriendTemplate:
525845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FunctionTemplate:
525945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::LinkageSpec:
526045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Namespace:
526145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NamespaceAlias:
526245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NonTypeTemplateParm:
526345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::StaticAssert:
526445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTemplateParm:
526545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTypeParm:
526645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingTypename:
526745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingValue:
526845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Using:
526945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingDirective:
527045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingShadow:
527145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_CPlusPlus;
527245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  }
527345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
527445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_C;
527545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
527645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
527745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekextern "C" {
527858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
527958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregorenum CXAvailabilityKind clang_getCursorAvailability(CXCursor cursor) {
528058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  if (clang_isDeclaration(cursor.kind))
528158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    if (Decl *D = cxcursor::getCursorDecl(cursor)) {
52820a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      if (isa<FunctionDecl>(D) && cast<FunctionDecl>(D)->isDeleted())
528358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Available;
528458ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
52850a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      switch (D->getAvailability()) {
52860a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Available:
52870a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_NotYetIntroduced:
52880a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_Available;
52890a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52900a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Deprecated:
529158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Deprecated;
52920a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52930a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Unavailable:
52940a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_NotAvailable;
52950a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      }
529658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
52970a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
529858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  return CXAvailability_Available;
529958ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor}
530058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
530145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted KremenekCXLanguageKind clang_getCursorLanguage(CXCursor cursor) {
530245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  if (clang_isDeclaration(cursor.kind))
530345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    return getDeclLanguage(cxcursor::getCursorDecl(cursor));
530445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
530545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_Invalid;
530645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
53073910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53083910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// \brief If the given cursor is the "templated" declaration
53093910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// descibing a class or function template, return the class or
53103910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// function template.
53113910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregorstatic Decl *maybeGetTemplateCursor(Decl *D) {
53123910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (!D)
53133910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    return 0;
53143910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53153910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
53163910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (FunctionTemplateDecl *FunTmpl = FD->getDescribedFunctionTemplate())
53173910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return FunTmpl;
53183910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53193910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D))
53203910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (ClassTemplateDecl *ClassTmpl = RD->getDescribedClassTemplate())
53213910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return ClassTmpl;
53223910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53233910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  return D;
53243910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor}
53253910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53262be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorSemanticParent(CXCursor cursor) {
53272be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
53282be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
53292be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getDeclContext();
53303910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
53313910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
53323910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53333910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
53343910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
53352be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
53362be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
53372be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
53382be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isStatement(cursor.kind) || clang_isExpression(cursor.kind)) {
53392be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor))
5340a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(D, getCursorTU(cursor));
53412be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
53422be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
53432be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
53442be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
53452be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
53462be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorLexicalParent(CXCursor cursor) {
53472be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
53482be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
53492be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getLexicalDeclContext();
53503910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
53513910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
53523910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
53533910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
53543910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
53552be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
53562be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
53572be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
53582be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // FIXME: Note that we can't easily compute the lexical context of a
53592be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // statement or expression, so we return nothing.
53602be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
53612be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
53622be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
53639f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_getOverriddenCursors(CXCursor cursor,
53649f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                CXCursor **overridden,
53659f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                unsigned *num_overridden) {
53669f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (overridden)
53679f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *overridden = 0;
53689f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (num_overridden)
53699f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *num_overridden = 0;
53709f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!overridden || !num_overridden)
53719f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53729f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
5373b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  SmallVector<CXCursor, 8> Overridden;
5374b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  cxcursor::getOverriddenCursors(cursor, Overridden);
53759f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53762407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek  // Don't allocate memory if we have no overriden cursors.
53772407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek  if (Overridden.size() == 0)
53782407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek    return;
53792407712d90cb1cce3bb2713d342c4df8222a2a47Ted Kremenek
5380b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  *num_overridden = Overridden.size();
5381b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  *overridden = new CXCursor [Overridden.size()];
5382b11be041e4f05519a2eabf6a99429ba6110f1ca9Argyrios Kyrtzidis  std::copy(Overridden.begin(), Overridden.end(), *overridden);
53839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53849f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_disposeOverriddenCursors(CXCursor *overridden) {
53869f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  delete [] overridden;
53879f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53889f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
5389ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas GregorCXFile clang_getIncludedFile(CXCursor cursor) {
5390ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (cursor.kind != CXCursor_InclusionDirective)
5391ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return 0;
5392ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
5393ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  InclusionDirective *ID = getCursorInclusionDirective(cursor);
5394ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  return (void *)ID->getFile();
5395ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor}
5396ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
539745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek} // end: extern "C"
539845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
53999ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
54009ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
54019ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek// C++ AST instrospection.
54029ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
54039ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
54049ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekextern "C" {
54059ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekunsigned clang_CXXMethod_isStatic(CXCursor C) {
54069ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek  if (!clang_isDeclaration(C.kind))
54079ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek    return 0;
540849f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor
540949f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  CXXMethodDecl *Method = 0;
541049f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
541149f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
541249f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
541349f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  else
541449f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
541549f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  return (Method && Method->isStatic()) ? 1 : 0;
541640b492a43bac3ed0c465772aa6921d011cfc273fTed Kremenek}
5417b12903e1a4b8d1b611b8c7e4f910665d628e68cdTed Kremenek
5418211924b563aa31421836cee7655be729ad02733fDouglas Gregorunsigned clang_CXXMethod_isVirtual(CXCursor C) {
5419211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (!clang_isDeclaration(C.kind))
5420211924b563aa31421836cee7655be729ad02733fDouglas Gregor    return 0;
5421211924b563aa31421836cee7655be729ad02733fDouglas Gregor
5422211924b563aa31421836cee7655be729ad02733fDouglas Gregor  CXXMethodDecl *Method = 0;
5423211924b563aa31421836cee7655be729ad02733fDouglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
5424211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
5425211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
5426211924b563aa31421836cee7655be729ad02733fDouglas Gregor  else
5427211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
5428211924b563aa31421836cee7655be729ad02733fDouglas Gregor  return (Method && Method->isVirtual()) ? 1 : 0;
5429211924b563aa31421836cee7655be729ad02733fDouglas Gregor}
54309ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek} // end: extern "C"
54319ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
543245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
543395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek// Attribute introspection.
543495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
543595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
543695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenekextern "C" {
543795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted KremenekCXType clang_getIBOutletCollectionType(CXCursor C) {
543895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  if (C.kind != CXCursor_IBOutletCollectionAttr)
5439a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return cxtype::MakeCXType(QualType(), cxcursor::getCursorTU(C));
544095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
544195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  IBOutletCollectionAttr *A =
544295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(C));
544395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
544418aa2ff4641847d7f8866e8c5912d4d0ddb858ceArgyrios Kyrtzidis  return cxtype::MakeCXType(A->getInterface(), cxcursor::getCursorTU(C));
544595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek}
544695f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek} // end: extern "C"
544795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
544895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
544959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek// Inspecting memory usage.
545059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
545159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5452f787002478f09af1741fb0f82a562002e6799c49Ted Kremenektypedef std::vector<CXTUResourceUsageEntry> MemUsageEntries;
545359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5454f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekstatic inline void createCXTUResourceUsageEntry(MemUsageEntries &entries,
5455f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek                                              enum CXTUResourceUsageKind k,
5456ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek                                              unsigned long amount) {
5457f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsageEntry entry = { k, amount };
545859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.push_back(entry);
545959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
546059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
546159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekextern "C" {
546259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5463f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekconst char *clang_getTUResourceUsageName(CXTUResourceUsageKind kind) {
546459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  const char *str = "";
546559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  switch (kind) {
5466f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_AST:
546759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: expressions, declarations, and types";
546859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5469f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Identifiers:
547059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: identifiers";
547159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5472f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Selectors:
547359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: selectors";
5474e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5475f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_GlobalCompletionResults:
54764e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      str = "Code completion: cached global results";
5477e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5478457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek    case CXTUResourceUsage_SourceManagerContentCache:
5479457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      str = "SourceManager: content cache allocator";
5480457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      break;
5481ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    case CXTUResourceUsage_AST_SideTables:
5482ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      str = "ASTContext: side tables";
5483ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      break;
5484f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_Malloc:
5485f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: malloc'ed memory buffers";
5486f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5487f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_MMap:
5488f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: mmap'ed memory buffers";
5489f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5490e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc:
5491e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: malloc'ed memory buffers";
5492e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
5493e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_MMap:
5494e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: mmap'ed memory buffers";
5495e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
54965e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_Preprocessor:
54975e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: malloc'ed memory";
54985e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
54995e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_PreprocessingRecord:
55005e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: PreprocessingRecord";
55015e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
5502ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek    case CXTUResourceUsage_SourceManager_DataStructures:
5503ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      str = "SourceManager: data structures and tables";
5504ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      break;
5505d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek    case CXTUResourceUsage_Preprocessor_HeaderSearch:
5506d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      str = "Preprocessor: header search tables";
5507d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      break;
550859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
550959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return str;
551059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
551159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5512f787002478f09af1741fb0f82a562002e6799c49Ted KremenekCXTUResourceUsage clang_getCXTUResourceUsage(CXTranslationUnit TU) {
551359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (!TU) {
5514f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    CXTUResourceUsage usage = { (void*) 0, 0, 0 };
551559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return usage;
551659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
551759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
551859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTUnit *astUnit = static_cast<ASTUnit*>(TU->TUData);
55191e4c01b79273b9cd4e9e9ecfd3422df3900b8356Dylan Noblesmith  OwningPtr<MemUsageEntries> entries(new MemUsageEntries());
552059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTContext &astContext = astUnit->getASTContext();
552159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
552259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by AST nodes and types?
5523f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST,
5524ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getASTAllocatedMemory());
552559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
552659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by identifiers?
5527f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Identifiers,
552859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Idents.getAllocator().getTotalMemory());
552959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
553059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used for selectors?
5531f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Selectors,
553259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Selectors.getTotalMemory());
553359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5534ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  // How much memory is used by ASTContext's side tables?
5535ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST_SideTables,
5536ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getSideTableAllocatedMemory());
5537ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek
55384e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  // How much memory is used for caching global code completion results?
55394e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned long completionBytes = 0;
55404e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  if (GlobalCodeCompletionAllocator *completionAllocator =
55414e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      astUnit->getCachedCompletionAllocator().getPtr()) {
55425e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    completionBytes = completionAllocator->getTotalMemory();
55434e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  }
5544457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5545457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               CXTUResourceUsage_GlobalCompletionResults,
5546457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               completionBytes);
5547457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek
5548457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  // How much memory is being used by SourceManager's content cache?
5549457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5550457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          CXTUResourceUsage_SourceManagerContentCache,
5551457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          (unsigned long) astContext.getSourceManager().getContentCacheSize());
5552f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5553f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  // How much memory is being used by the MemoryBuffer's in SourceManager?
5554f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  const SourceManager::MemoryBufferSizes &srcBufs =
5555f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    astUnit->getSourceManager().getMemoryBufferSizes();
5556f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5557f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5558f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_Malloc,
5559f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.malloc_bytes);
5560ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5561f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_MMap,
5562f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.mmap_bytes);
5563ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5564ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               CXTUResourceUsage_SourceManager_DataStructures,
5565ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               (unsigned long) astContext.getSourceManager()
5566ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                                .getDataStructureSizes());
5567e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5568e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  // How much memory is being used by the ExternalASTSource?
5569e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  if (ExternalASTSource *esrc = astContext.getExternalSource()) {
5570e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    const ExternalASTSource::MemoryBufferSizes &sizes =
5571e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      esrc->getMemoryBufferSizes();
5572e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5573e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5574e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc,
5575e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.malloc_bytes);
5576e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5577e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_MMap,
5578e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.mmap_bytes);
5579e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  }
55805e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
55815e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  // How much memory is being used by the Preprocessor?
55825e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  Preprocessor &pp = astUnit->getPreprocessor();
55835e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  createCXTUResourceUsageEntry(*entries,
55845e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                               CXTUResourceUsage_Preprocessor,
5585c5c5e92ec53f7e6ac7ebbbf77c6d8e4b7d88daecArgyrios Kyrtzidis                               pp.getTotalMemory());
55865e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
55875e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  if (PreprocessingRecord *pRec = pp.getPreprocessingRecord()) {
55885e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    createCXTUResourceUsageEntry(*entries,
55895e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 CXTUResourceUsage_PreprocessingRecord,
55905e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 pRec->getTotalMemory());
55915e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  }
55925e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5593d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5594d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               CXTUResourceUsage_Preprocessor_HeaderSearch,
5595d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               pp.getHeaderSearchInfo().getTotalMemory());
55965e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5597f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = { (void*) entries.get(),
559859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            (unsigned) entries->size(),
559959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            entries->size() ? &(*entries)[0] : 0 };
560059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.take();
560159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return usage;
560259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
560359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5604f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekvoid clang_disposeCXTUResourceUsage(CXTUResourceUsage usage) {
560559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (usage.data)
560659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    delete (MemUsageEntries*) usage.data;
560759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
560859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
560959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek} // end extern "C"
561059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
56116df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregorvoid clang::PrintLibclangResourceUsage(CXTranslationUnit TU) {
56126df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  CXTUResourceUsage Usage = clang_getCXTUResourceUsage(TU);
56136df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  for (unsigned I = 0; I != Usage.numEntries; ++I)
56146df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    fprintf(stderr, "  %s: %lu\n",
56156df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            clang_getTUResourceUsageName(Usage.entries[I].kind),
56166df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            Usage.entries[I].amount);
56176df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
56186df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  clang_disposeCXTUResourceUsage(Usage);
56196df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor}
56206df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
562159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
562204bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek// Misc. utility functions.
562304bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek//===----------------------------------------------------------------------===//
5624f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
5625abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbar/// Default to using an 8 MB stack size on "safety" threads.
5626abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbarstatic unsigned SafetyStackThreadSize = 8 << 20;
5627bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5628bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarnamespace clang {
5629bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5630bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarbool RunSafely(llvm::CrashRecoveryContext &CRC,
56316c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               void (*Fn)(void*), void *UserData,
56326c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               unsigned Size) {
56336c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (!Size)
56346c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek    Size = GetSafetyThreadStackSize();
56356c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (Size)
5636bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar    return CRC.RunSafelyOnThread(Fn, UserData, Size);
5637bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return CRC.RunSafely(Fn, UserData);
5638bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5639bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5640bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarunsigned GetSafetyThreadStackSize() {
5641bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return SafetyStackThreadSize;
5642bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5643bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5644bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarvoid SetSafetyThreadStackSize(unsigned Value) {
5645bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  SafetyStackThreadSize = Value;
5646bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5647bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5648bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5649bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
565004bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenekextern "C" {
565104bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
5652a2a9d6e4e5b6001b86b7dfc5db1ea296ce29a3d3Ted KremenekCXString clang_getClangVersion() {
5653ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(getClangFullVersion());
565404bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek}
565504bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
565604bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek} // end: extern "C"
565759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5658