CIndex.cpp revision 92ddef1bf843e1e18c040d69f48a6bf0bc7c776a
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"
22ab1889321f6f8f200f2b318ac26883ac18e49d03Ted Kremenek
2304bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek#include "clang/Basic/Version.h"
24936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor
2550398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff#include "clang/AST/DeclVisitor.h"
26fb5704295c6137685a7b90b92cd6b958028740c8Steve Naroff#include "clang/AST/StmtVisitor.h"
277d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor#include "clang/AST/TypeLocVisitor.h"
28b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Basic/Diagnostic.h"
29b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Frontend/ASTUnit.h"
30b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Frontend/CompilerInstance.h"
31936ea3b590117d2cd73b1b92621d06c4a7edbe60Douglas Gregor#include "clang/Frontend/FrontendDiagnostic.h"
32d8210650ed948de65a08a8daf16d291b747717c4Ted Kremenek#include "clang/Lex/Lexer.h"
33dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor#include "clang/Lex/HeaderSearch.h"
34b846debc1b22a37228efe4aa87b34482d15b6a3cBenjamin Kramer#include "clang/Lex/PreprocessingRecord.h"
3533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor#include "clang/Lex/Preprocessor.h"
36a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor#include "llvm/ADT/STLExtras.h"
37d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek#include "llvm/ADT/Optional.h"
38f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor#include "llvm/ADT/StringSwitch.h"
39d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek#include "clang/Analysis/Support/SaveAndRestore.h"
40c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar#include "llvm/Support/CrashRecoveryContext.h"
4148615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar#include "llvm/Support/PrettyStackTrace.h"
4202465750c8c3fa96b1e7e596b02297e24361dc4fDouglas Gregor#include "llvm/Support/MemoryBuffer.h"
43358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor#include "llvm/Support/raw_ostream.h"
447a07fcb8f10fe45ea65a0a41798eb1c40777bde4Douglas Gregor#include "llvm/Support/Timer.h"
4503013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Mutex.h"
4603013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Program.h"
4703013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Signals.h"
4803013fa9a0bf1ef4b907f5fec006c8f4000fdd21Michael J. Spencer#include "llvm/Support/Threading.h"
4937f1ea0eb08a00fa90edbecb427cfbb50ca0f4d0Ted Kremenek#include "llvm/Support/Compiler.h"
50fc0622155fa61349698a8fd0053773c37d9f7ac4Ted Kremenek
5150398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroffusing namespace clang;
5216c440a377b7ec8b722a2e2c7c864f75c95bd305Ted Kremenekusing namespace clang::cxcursor;
53ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenekusing namespace clang::cxstring;
5450398199fb10e196a8d92fbf7a062dbe42ed88fdSteve Naroff
55a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenekstatic CXTranslationUnit 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();
61a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return D;
62a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek}
63a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek
6433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// \brief The result of comparing two source ranges.
6533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregorenum RangeComparisonResult {
6633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief Either the ranges overlap or one of the ranges is invalid.
6733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  RangeOverlap,
68f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
6933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The first range ends before the second range starts.
7033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  RangeBefore,
71f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The first range starts after the second range ends.
7333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  RangeAfter
7433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor};
7533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
76f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek/// \brief Compare two source ranges to determine their relative position in
7733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// the translation unit.
78f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekstatic RangeComparisonResult RangeCompare(SourceManager &SM,
79f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                          SourceRange R1,
8033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor                                          SourceRange R2) {
8133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  assert(R1.isValid() && "First range is invalid?");
8233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  assert(R2.isValid() && "Second range is invalid?");
83a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R1.getEnd() != R2.getBegin() &&
84d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar      SM.isBeforeInTranslationUnit(R1.getEnd(), R2.getBegin()))
8533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    return RangeBefore;
86a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R2.getEnd() != R1.getBegin() &&
87d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar      SM.isBeforeInTranslationUnit(R2.getEnd(), R1.getBegin()))
8833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    return RangeAfter;
8933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  return RangeOverlap;
9033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
9133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
92fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek/// \brief Determine if a source location falls within, before, or after a
93fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek///   a given source range.
94fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekstatic RangeComparisonResult LocationCompare(SourceManager &SM,
95fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                             SourceLocation L, SourceRange R) {
96fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  assert(R.isValid() && "First range is invalid?");
97fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  assert(L.isValid() && "Second range is invalid?");
98a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (L == R.getBegin() || L == R.getEnd())
99fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeOverlap;
100fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (SM.isBeforeInTranslationUnit(L, R.getBegin()))
101fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeBefore;
102fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (SM.isBeforeInTranslationUnit(R.getEnd(), L))
103fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return RangeAfter;
104fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  return RangeOverlap;
105fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek}
106fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
10776dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// \brief Translate a Clang source range into a CIndex source range.
10876dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar///
10976dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// Clang internally represents ranges where the end location points to the
11076dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// start of the token at the end. However, for external clients it is more
11176dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// useful to have a CXSourceRange be a proper half-open interval. This routine
11276dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar/// does the appropriate translation.
113f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed KremenekCXSourceRange cxloc::translateSourceRange(const SourceManager &SM,
11476dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                                          const LangOptions &LangOpts,
1150a76aae8c03cb7dd7bdbe683485560afaf695959Chris Lattner                                          const CharSourceRange &R) {
11676dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  // We want the last character in this location, so we will adjust the
1176a5a23f8e7fb65e028c8092bc1d1a1d9dfe2e9bcDouglas Gregor  // location accordingly.
11876dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  SourceLocation EndLoc = R.getEnd();
119a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (EndLoc.isValid() && EndLoc.isMacroID())
120edc3dccece244a584f8ebdb81da6c962c08e79beChandler Carruth    EndLoc = SM.getExpansionRange(EndLoc).second;
1210a76aae8c03cb7dd7bdbe683485560afaf695959Chris Lattner  if (R.isTokenRange() && !EndLoc.isInvalid() && EndLoc.isFileID()) {
1226a5a23f8e7fb65e028c8092bc1d1a1d9dfe2e9bcDouglas Gregor    unsigned Length = Lexer::MeasureTokenLength(EndLoc, SM, LangOpts);
123a64ccefdf0ea4e03ec88805d71b0af74950c7472Argyrios Kyrtzidis    EndLoc = EndLoc.getLocWithOffset(Length);
12476dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  }
12576dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar
12676dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  CXSourceRange Result = { { (void *)&SM, (void *)&LangOpts },
12776dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                           R.getBegin().getRawEncoding(),
12876dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar                           EndLoc.getRawEncoding() };
12976dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar  return Result;
13076dd3c2ff274772dc5771cb73f856d14aaf5ee2fDaniel Dunbar}
1311db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
1328a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek//===----------------------------------------------------------------------===//
13333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor// Cursor visitor.
1348a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek//===----------------------------------------------------------------------===//
1358a8da7d17d4eb281b61d08d603c7bb180d280d5aTed Kremenek
13689922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroffnamespace {
137c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
138c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekclass VisitorJob {
139c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekpublic:
140cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek  enum Kind { DeclVisitKind, StmtVisitKind, MemberExprPartsKind,
141e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek              TypeLocVisitKind, OverloadExprPartsKind,
14260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek              DeclRefExprPartsKind, LabelRefVisitKind,
143f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek              ExplicitTemplateArgsVisitKind,
144f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor              NestedNameSpecifierLocVisitKind,
145cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek              DeclarationNameInfoVisitKind,
14694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor              MemberRefVisitKind, SizeOfPackExprPartsKind };
147c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekprotected:
148f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void *data[3];
149c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  CXCursor parent;
150c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  Kind K;
151f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  VisitorJob(CXCursor C, Kind k, void *d1, void *d2 = 0, void *d3 = 0)
152f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    : parent(C), K(k) {
153f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    data[0] = d1;
154f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    data[1] = d2;
155f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    data[2] = d3;
156f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
157c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekpublic:
158c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  Kind getKind() const { return K; }
159c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  const CXCursor &getParent() const { return parent; }
160c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  static bool classof(VisitorJob *VJ) { return true; }
161c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek};
162c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
1635f9e272e632e951b1efe824cd16acb4d96077930Chris Lattnertypedef SmallVector<VisitorJob, 10> VisitorWorkList;
164c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
165b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor// Cursor visitor.
1667d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorclass CursorVisitor : public DeclVisitor<CursorVisitor, bool>,
167cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek                      public TypeLocVisitor<CursorVisitor, bool>
1687d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor{
16933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The translation unit we are traversing.
170a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU;
171a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *AU;
172f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
17333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The parent cursor whose children we are traversing.
174b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  CXCursor Parent;
175f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
17633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The declaration that serves at the parent of any statement or
17733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// expression nodes.
178f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  Decl *StmtParent;
179f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
18033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The visitor function.
181b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  CXCursorVisitor Visitor;
182f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
18333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief The opaque client data, to be passed along to the visitor.
184b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  CXClientData ClientData;
185f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
18604a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor  /// \brief Whether we should visit the preprocessing record entries last,
18704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor  /// after visiting other declarations.
18804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor  bool VisitPreprocessorLast;
18904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
19033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// \brief When valid, a source range to which the cursor should restrict
19133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  /// its search.
19233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  SourceRange RegionOfInterest;
193f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
194d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually remove.  This part of a hack to support proper
195d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // iteration over all Decls contained lexically within an ObjC container.
196d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  DeclContext::decl_iterator *DI_current;
197d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  DeclContext::decl_iterator DE_current;
198d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
199d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  // Cache of pre-allocated worklists for data-recursion walk of Stmts.
2005f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<VisitorWorkList*, 5> WorkListFreeList;
2015f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<VisitorWorkList*, 5> WorkListCache;
202d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek
203b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  using DeclVisitor<CursorVisitor, bool>::Visit;
2047d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  using TypeLocVisitor<CursorVisitor, bool>::Visit;
205f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
206f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  /// \brief Determine whether this particular source range comes before, comes
207f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  /// after, or overlaps the region of interest.
20833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  ///
209d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar  /// \param R a half-open source range retrieved from the abstract syntax tree.
210f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  RangeComparisonResult CompareRegionOfInterest(SourceRange R);
211f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2120f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  class SetParentRAII {
2130f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    CXCursor &Parent;
2140f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    Decl *&StmtParent;
2150f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    CXCursor OldParent;
2160f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek
2170f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  public:
2180f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    SetParentRAII(CXCursor &Parent, Decl *&StmtParent, CXCursor NewParent)
2190f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek      : Parent(Parent), StmtParent(StmtParent), OldParent(Parent)
2200f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    {
2210f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek      Parent = NewParent;
2220f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek      if (clang_isDeclaration(Parent.kind))
2230f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek        StmtParent = getCursorDecl(Parent);
2240f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    }
2250f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek
2260f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    ~SetParentRAII() {
2270f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek      Parent = OldParent;
2280f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek      if (clang_isDeclaration(Parent.kind))
2290f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek        StmtParent = getCursorDecl(Parent);
2300f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek    }
2310f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  };
2320f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek
233b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorpublic:
234a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CursorVisitor(CXTranslationUnit TU, CXCursorVisitor Visitor,
235a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                CXClientData ClientData,
23604a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                bool VisitPreprocessorLast,
23733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor                SourceRange RegionOfInterest = SourceRange())
238a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    : TU(TU), AU(static_cast<ASTUnit*>(TU->TUData)),
239a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      Visitor(Visitor), ClientData(ClientData),
24008e0bc16b3312c27e87d33be7dcf3d4fe5bdd2e2Douglas Gregor      VisitPreprocessorLast(VisitPreprocessorLast),
24104a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      RegionOfInterest(RegionOfInterest), DI_current(0)
242b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  {
243b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Parent.kind = CXCursor_NoDeclFound;
244b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Parent.data[0] = 0;
245b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Parent.data[1] = 0;
246b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Parent.data[2] = 0;
247f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    StmtParent = 0;
248b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
249f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
250d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  ~CursorVisitor() {
251d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    // Free the pre-allocated worklists for data-recursion.
2525f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    for (SmallVectorImpl<VisitorWorkList*>::iterator
253d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek          I = WorkListCache.begin(), E = WorkListCache.end(); I != E; ++I) {
254d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek      delete *I;
255d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    }
256d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
257d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek
258a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *getASTUnit() const { return static_cast<ASTUnit*>(TU->TUData); }
259a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit getTU() const { return TU; }
260ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek
26133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  bool Visit(CXCursor Cursor, bool CheckedRegionOfInterest = false);
262788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
2634c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  bool visitPreprocessedEntitiesInRegion();
2644c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
2654c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  template<typename InputIterator>
2664c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  bool visitPreprocessedEntities(InputIterator First, InputIterator Last);
267788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
268b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  bool VisitChildren(CXCursor Parent);
269f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2707d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  // Declaration visitors
271162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  bool VisitTypeAliasDecl(TypeAliasDecl *D);
27209dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek  bool VisitAttributes(Decl *D);
2731ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek  bool VisitBlockDecl(BlockDecl *B);
2743064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  bool VisitCXXRecordDecl(CXXRecordDecl *D);
275d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  llvm::Optional<bool> shouldVisitCursor(CXCursor C);
276b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  bool VisitDeclContext(DeclContext *DC);
27779758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitTranslationUnitDecl(TranslationUnitDecl *D);
27879758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitTypedefDecl(TypedefDecl *D);
27979758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitTagDecl(TagDecl *D);
2800ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  bool VisitClassTemplateSpecializationDecl(ClassTemplateSpecializationDecl *D);
28174dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  bool VisitClassTemplatePartialSpecializationDecl(
28274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor                                     ClassTemplatePartialSpecializationDecl *D);
283fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  bool VisitTemplateTypeParmDecl(TemplateTypeParmDecl *D);
2844540c9c73787d6ef736792f24209727b64997c90Ted Kremenek  bool VisitEnumConstantDecl(EnumConstantDecl *D);
28579758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitDeclaratorDecl(DeclaratorDecl *DD);
2864540c9c73787d6ef736792f24209727b64997c90Ted Kremenek  bool VisitFunctionDecl(FunctionDecl *ND);
28779758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitFieldDecl(FieldDecl *D);
28879758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitVarDecl(VarDecl *);
28984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  bool VisitNonTypeTemplateParmDecl(NonTypeTemplateParmDecl *D);
290fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  bool VisitFunctionTemplateDecl(FunctionTemplateDecl *D);
29139d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  bool VisitClassTemplateDecl(ClassTemplateDecl *D);
29284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  bool VisitTemplateTemplateParmDecl(TemplateTemplateParmDecl *D);
29379758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCMethodDecl(ObjCMethodDecl *ND);
2944540c9c73787d6ef736792f24209727b64997c90Ted Kremenek  bool VisitObjCContainerDecl(ObjCContainerDecl *D);
29579758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCCategoryDecl(ObjCCategoryDecl *ND);
29679758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCProtocolDecl(ObjCProtocolDecl *PID);
29723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  bool VisitObjCPropertyDecl(ObjCPropertyDecl *PD);
29879758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCInterfaceDecl(ObjCInterfaceDecl *D);
2994540c9c73787d6ef736792f24209727b64997c90Ted Kremenek  bool VisitObjCImplDecl(ObjCImplDecl *D);
30079758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D);
3011ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  bool VisitObjCImplementationDecl(ObjCImplementationDecl *D);
30279758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  // FIXME: ObjCCompatibleAliasDecl requires aliased-class locations.
30379758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D);
30479758f614c83ba17757618de1459e2b481f19dd2Ted Kremenek  bool VisitObjCClassDecl(ObjCClassDecl *D);
305a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  bool VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *PD);
306a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  bool VisitLinkageSpecDecl(LinkageSpecDecl *D);
3078f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  bool VisitNamespaceDecl(NamespaceDecl *D);
3086931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  bool VisitNamespaceAliasDecl(NamespaceAliasDecl *D);
3090a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  bool VisitUsingDirectiveDecl(UsingDirectiveDecl *D);
3107e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  bool VisitUsingDecl(UsingDecl *D);
3117e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  bool VisitUnresolvedUsingValueDecl(UnresolvedUsingValueDecl *D);
3127e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  bool VisitUnresolvedUsingTypenameDecl(UnresolvedUsingTypenameDecl *D);
3130a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
31401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  // Name visitor
31501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  bool VisitDeclarationNameInfo(DeclarationNameInfo Name);
316c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  bool VisitNestedNameSpecifier(NestedNameSpecifier *NNS, SourceRange Range);
317dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  bool VisitNestedNameSpecifierLoc(NestedNameSpecifierLoc NNS);
31801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
319fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // Template visitors
320fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  bool VisitTemplateParameters(const TemplateParameterList *Params);
3210b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  bool VisitTemplateName(TemplateName Name, SourceLocation Loc);
322fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  bool VisitTemplateArgumentLoc(const TemplateArgumentLoc &TAL);
323fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
3247d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  // Type visitors
325427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#define ABSTRACT_TYPELOC(CLASS, PARENT)
326427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#define TYPELOC(CLASS, PARENT) \
327427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  bool Visit##CLASS##TypeLoc(CLASS##TypeLoc TyLoc);
328427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#include "clang/AST/TypeLocNodes.def"
329427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
330f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  bool VisitTagTypeLoc(TagTypeLoc TL);
331f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  bool VisitArrayTypeLoc(ArrayTypeLoc TL);
332427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  bool VisitFunctionTypeLoc(FunctionTypeLoc TL, bool SkipResultType = false);
333427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
334c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  // Data-recursive visitor functions.
335c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  bool IsInRegionOfInterest(CXCursor C);
336c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  bool RunVisitorWorkList(VisitorWorkList &WL);
337c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  void EnqueueWorkList(VisitorWorkList &WL, Stmt *S);
338cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  LLVM_ATTRIBUTE_NOINLINE bool Visit(Stmt *S);
339b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor};
340f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
341b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor} // end anonymous namespace
3420d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
343a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorstatic SourceRange getRawCursorExtent(CXCursor C);
3446653798ff5ce6deb58112777e21307ccc453133dDouglas Gregorstatic SourceRange getFullCursorExtent(CXCursor C, SourceManager &SrcMgr);
3456653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
346a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
34733e9abd21083a0191a7676a04b497006d2da184dDouglas GregorRangeComparisonResult CursorVisitor::CompareRegionOfInterest(SourceRange R) {
348a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return RangeCompare(AU->getSourceManager(), R, RegionOfInterest);
34933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
35033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
351b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \brief Visit the given cursor and, if requested by the visitor,
352b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// its children.
353b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor///
35433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// \param Cursor the cursor to visit.
35533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor///
35633e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// \param CheckRegionOfInterest if true, then the caller already checked that
35733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor/// this cursor is within the region of interest.
35833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor///
359b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \returns true if the visitation should be aborted, false if it
360b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// should continue.
36133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregorbool CursorVisitor::Visit(CXCursor Cursor, bool CheckedRegionOfInterest) {
362b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isInvalid(Cursor.kind))
363b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return false;
364f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
365b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isDeclaration(Cursor.kind)) {
366b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Decl *D = getCursorDecl(Cursor);
367b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    assert(D && "Invalid declaration cursor");
368b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    if (D->isImplicit())
369b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return false;
370b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
3710d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
37233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // If we have a range of interest, and this cursor doesn't intersect with it,
37333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  // we're done.
37433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (RegionOfInterest.isValid() && !CheckedRegionOfInterest) {
375a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    SourceRange Range = getRawCursorExtent(Cursor);
376f408f32aa9ae3d97bc656267dc5d78fa7d03499bDaniel Dunbar    if (Range.isInvalid() || CompareRegionOfInterest(Range))
37733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor      return false;
37833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  }
379f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
380b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  switch (Visitor(Cursor, Parent, ClientData)) {
381b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Break:
382b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
3830d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
384b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Continue:
385b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return false;
3862e331b938b38057e333fab0ba841130ea8467794Douglas Gregor
387b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  case CXChildVisit_Recurse:
388b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return VisitChildren(Cursor);
389b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
3900d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
391fd64377225a6a140bddb3f997d52a036486f9360Douglas Gregor  return false;
392b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
3930d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
3944c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntitiesInRegion() {
395788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  PreprocessingRecord &PPRec
396a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    = *AU->getPreprocessor().getPreprocessingRecord();
397788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
39892ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (RegionOfInterest.isValid()) {
39992ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis    std::pair<PreprocessingRecord::iterator, PreprocessingRecord::iterator>
40092ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis      Entities = PPRec.getPreprocessedEntitiesInRange(RegionOfInterest);
40192ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis    return visitPreprocessedEntities(Entities.first, Entities.second);
40292ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  }
40392ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis
404788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  bool OnlyLocalDecls
40532038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor    = !AU->isMainFileAST() && AU->getOnlyLocalDecls();
40632038bb8486a1f31e8bd8e19ef388049669e9ed2Douglas Gregor
40792ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  if (OnlyLocalDecls)
40892ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis    return visitPreprocessedEntities(PPRec.local_begin(), PPRec.local_end());
4094c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
41092ddef1bf843e1e18c040d69f48a6bf0bc7c776aArgyrios Kyrtzidis  return visitPreprocessedEntities(PPRec.begin(), PPRec.end());
4114c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor}
4124c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4134c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregortemplate<typename InputIterator>
4144c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregorbool CursorVisitor::visitPreprocessedEntities(InputIterator First,
4154c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor                                              InputIterator Last) {
4164c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  for (; First != Last; ++First) {
4174c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    if (MacroExpansion *ME = dyn_cast<MacroExpansion>(*First)) {
4184c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroExpansionCursor(ME, TU)))
4194c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4204c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4214c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4224c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4234c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4244c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    if (MacroDefinition *MD = dyn_cast<MacroDefinition>(*First)) {
4254c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeMacroDefinitionCursor(MD, TU)))
4264c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
42789d9980bbc2e4a4ac86673e6ec16fb9f5babb63bDouglas Gregor
4284c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
4294c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    }
4304c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4314c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor    if (InclusionDirective *ID = dyn_cast<InclusionDirective>(*First)) {
4324c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (Visit(MakeInclusionDirectiveCursor(ID, TU)))
4334c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        return true;
4344c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor
4354c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      continue;
436788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor    }
437788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor  }
438788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
4394c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor  return false;
440788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor}
441788f5a1242c04762f91eaa7565c07b9865846d88Douglas Gregor
442b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \brief Visit the children of the given cursor.
443a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek///
444b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// \returns true if the visitation should be aborted, false if it
445b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor/// should continue.
446f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekbool CursorVisitor::VisitChildren(CXCursor Cursor) {
447c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (clang_isReference(Cursor.kind) &&
448c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      Cursor.kind != CXCursor_CXXBaseSpecifier) {
449a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    // By definition, references have no children.
450a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return false;
451a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  }
452f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
453f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Set the Parent field to Cursor, then back to its old value once we're
454b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // done.
4550f91f6a2eacf29fb69b1b0c3131eb0385aad5777Ted Kremenek  SetParentRAII SetParent(Parent, StmtParent, Cursor);
456f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
457b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isDeclaration(Cursor.kind)) {
458b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    Decl *D = getCursorDecl(Cursor);
45906d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (!D)
46006d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return false;
46106d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
462539311e0221df256c70c1c3080c8af847cd29dffTed Kremenek    return VisitAttributes(D) || Visit(D);
463b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
464f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
46506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isStatement(Cursor.kind)) {
46606d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Stmt *S = getCursorStmt(Cursor))
46706d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(S);
46806d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
46906d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
47006d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
47106d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
47206d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  if (clang_isExpression(Cursor.kind)) {
47306d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    if (Expr *E = getCursorExpr(Cursor))
47406d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor      return Visit(E);
47506d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor
47606d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor    return false;
47706d9b1ad0bca7230cbae57e3e3207dda77a9eac0Douglas Gregor  }
478f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
479b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (clang_isTranslationUnit(Cursor.kind)) {
480a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    CXTranslationUnit tu = getCursorTU(Cursor);
481a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    ASTUnit *CXXUnit = static_cast<ASTUnit*>(tu->TUData);
48204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
48304a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    int VisitOrder[2] = { VisitPreprocessorLast, !VisitPreprocessorLast };
48404a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor    for (unsigned I = 0; I != 2; ++I) {
48504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      if (VisitOrder[I]) {
48604a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        if (!CXXUnit->isMainFileAST() && CXXUnit->getOnlyLocalDecls() &&
48704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor            RegionOfInterest.isInvalid()) {
48804a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          for (ASTUnit::top_level_iterator TL = CXXUnit->top_level_begin(),
48904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                        TLEnd = CXXUnit->top_level_end();
49004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor               TL != TLEnd; ++TL) {
49104a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor            if (Visit(MakeCXCursor(*TL, tu), true))
49204a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor              return true;
49304a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor          }
49404a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        } else if (VisitDeclContext(
49504a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                                CXXUnit->getASTContext().getTranslationUnitDecl()))
4967b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor          return true;
49704a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor        continue;
4987b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor      }
4993178cb674ac8c3b59e1791e14d38d48619a1b621Bob Wilson
50004a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor      // Walk the preprocessing record.
5014c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor      if (CXXUnit->getPreprocessor().getPreprocessingRecord())
5024c30bb148b53c8063e940ca3e049ba4d270dc9d5Douglas Gregor        visitPreprocessedEntitiesInRegion();
5030396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    }
50404a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor
5057b691f33829e6a302e256e138b3917390c2665bbDouglas Gregor    return false;
506b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
507f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
508c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  if (Cursor.kind == CXCursor_CXXBaseSpecifier) {
509c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    if (CXXBaseSpecifier *Base = getCursorCXXBaseSpecifier(Cursor)) {
510c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      if (TypeSourceInfo *BaseTSInfo = Base->getTypeSourceInfo()) {
511c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor        return Visit(BaseTSInfo->getTypeLoc());
512c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor      }
513c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor    }
514c314aa484524c6a310a51e7b83bc194774e95e90Douglas Gregor  }
515221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
516221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  if (Cursor.kind == CXCursor_IBOutletCollectionAttr) {
517221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    IBOutletCollectionAttr *A =
518221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(Cursor));
519221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis    if (const ObjCInterfaceType *InterT = A->getInterface()->getAs<ObjCInterfaceType>())
520221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis      return Visit(cxcursor::MakeCursorObjCClassRef(InterT->getInterface(),
521221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis                                                    A->getInterfaceLoc(), TU));
522221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis  }
523221d5a5005b3dbde50b8b0a40bb5c5e6da8c1173Argyrios Kyrtzidis
524b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  // Nothing to visit at the moment.
525b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
526dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
527dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5281ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenekbool CursorVisitor::VisitBlockDecl(BlockDecl *B) {
52913c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor  if (TypeSourceInfo *TSInfo = B->getSignatureAsWritten())
53013c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor    if (Visit(TSInfo->getTypeLoc()))
53113c8ccb59b38e9e7133f1c80a00f210b6514a0b1Douglas Gregor        return true;
5321ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
533664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  if (Stmt *Body = B->getBody())
534664cffd330611d78fc0286f539589920a37ca328Ted Kremenek    return Visit(MakeCXCursor(Body, StmtParent, TU));
535664cffd330611d78fc0286f539589920a37ca328Ted Kremenek
536664cffd330611d78fc0286f539589920a37ca328Ted Kremenek  return false;
5371ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek}
5381ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek
539d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekllvm::Optional<bool> CursorVisitor::shouldVisitCursor(CXCursor Cursor) {
540d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (RegionOfInterest.isValid()) {
5416653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange Range = getFullCursorExtent(Cursor, AU->getSourceManager());
542d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Range.isInvalid())
543d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
5446653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
545d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    switch (CompareRegionOfInterest(Range)) {
546d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeBefore:
547d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes before the region of interest; skip it.
548d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return llvm::Optional<bool>();
54923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
550d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeAfter:
551d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration comes after the region of interest; we're done.
552d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
553d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar
554d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    case RangeOverlap:
555d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      // This declaration overlaps the region of interest; visit it.
556d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      break;
557d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
558d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
559d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return true;
560d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
561f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
562d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenekbool CursorVisitor::VisitDeclContext(DeclContext *DC) {
563d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  DeclContext::decl_iterator I = DC->decls_begin(), E = DC->decls_end();
564f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
565d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually remove.  This part of a hack to support proper
566d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // iteration over all Decls contained lexically within an ObjC container.
567d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator*> DI_saved(DI_current, &I);
568d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SaveAndRestore<DeclContext::decl_iterator> DE_saved(DE_current, E);
569f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
570d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for ( ; I != E; ++I) {
571d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *D = *I;
572d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (D->getLexicalDeclContext() != DC)
573d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
574d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    CXCursor Cursor = MakeCXCursor(D, TU);
575d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
576d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
577d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
578d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
579d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
580d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar    if (Visit(Cursor, true))
581b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
582b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  }
583b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
584dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
585dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
5861ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTranslationUnitDecl(TranslationUnitDecl *D) {
5871ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  llvm_unreachable("Translation units are visited directly by Visit()");
5881ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
5891ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
5901ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
591162e1c1b487352434552147967c3dd296ebee2f7Richard Smithbool CursorVisitor::VisitTypeAliasDecl(TypeAliasDecl *D) {
592162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
593162e1c1b487352434552147967c3dd296ebee2f7Richard Smith    return Visit(TSInfo->getTypeLoc());
594162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
595162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return false;
596162e1c1b487352434552147967c3dd296ebee2f7Richard Smith}
597162e1c1b487352434552147967c3dd296ebee2f7Richard Smith
5981ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTypedefDecl(TypedefDecl *D) {
5991ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (TypeSourceInfo *TSInfo = D->getTypeSourceInfo())
6001ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(TSInfo->getTypeLoc());
601f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
6021ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
6031ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6041ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6051ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitTagDecl(TagDecl *D) {
6061ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitDeclContext(D);
6071ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6081ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6090ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregorbool CursorVisitor::VisitClassTemplateSpecializationDecl(
6100ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor                                          ClassTemplateSpecializationDecl *D) {
6110ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  bool ShouldVisitBody = false;
6120ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  switch (D->getSpecializationKind()) {
6130ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_Undeclared:
6140ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ImplicitInstantiation:
6150ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    // Nothing to visit
6160ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return false;
6170ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6180ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDeclaration:
6190ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitInstantiationDefinition:
6200ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6210ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6220ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  case TSK_ExplicitSpecialization:
6230ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    ShouldVisitBody = true;
6240ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    break;
6250ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6260ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6270ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  // Visit the template arguments used in the specialization.
6280ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (TypeSourceInfo *SpecType = D->getTypeAsWritten()) {
6290ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    TypeLoc TL = SpecType->getTypeLoc();
6300ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    if (TemplateSpecializationTypeLoc *TSTLoc
6310ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          = dyn_cast<TemplateSpecializationTypeLoc>(&TL)) {
6320ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor      for (unsigned I = 0, N = TSTLoc->getNumArgs(); I != N; ++I)
6330ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor        if (VisitTemplateArgumentLoc(TSTLoc->getArgLoc(I)))
6340ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor          return true;
6350ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    }
6360ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  }
6370ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6380ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  if (ShouldVisitBody && VisitCXXRecordDecl(D))
6390ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor    return true;
6400ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
6410ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor  return false;
6420ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor}
6430ab1e9f672a86cf3f094780d99251553a2b69fc7Douglas Gregor
64474dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregorbool CursorVisitor::VisitClassTemplatePartialSpecializationDecl(
64574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor                                   ClassTemplatePartialSpecializationDecl *D) {
64674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
64774dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // before visiting these template parameters.
64874dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
64974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return true;
65074dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
65174dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  // Visit the partial specialization arguments.
65274dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  const TemplateArgumentLoc *TemplateArgs = D->getTemplateArgsAsWritten();
65374dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  for (unsigned I = 0, N = D->getNumTemplateArgsAsWritten(); I != N; ++I)
65474dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    if (VisitTemplateArgumentLoc(TemplateArgs[I]))
65574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor      return true;
65674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
65774dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  return VisitCXXRecordDecl(D);
65874dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor}
65974dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor
660fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmDecl(TemplateTypeParmDecl *D) {
66184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  // Visit the default argument.
66284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
66384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (TypeSourceInfo *DefArg = D->getDefaultArgumentInfo())
66484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      if (Visit(DefArg->getTypeLoc()))
66584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor        return true;
66684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
667fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
668fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
669fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
6701ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitEnumConstantDecl(EnumConstantDecl *D) {
6711ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInitExpr())
6721ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(MakeCXCursor(Init, StmtParent, TU));
6731ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
6741ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
6751ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
6767d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitDeclaratorDecl(DeclaratorDecl *DD) {
6777d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  if (TypeSourceInfo *TSInfo = DD->getTypeSourceInfo())
6787d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
6797d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      return true;
6807d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
681c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
682c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = DD->getQualifierLoc())
683c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
684c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
685c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
6867d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor  return false;
6877d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
6887d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
689a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor/// \brief Compare two base or member initializers based on their source order.
690cbb67480094b3bcb5b715acd827cbad55e2a204cSean Huntstatic int CompareCXXCtorInitializers(const void* Xp, const void *Yp) {
691cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *X
692cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Xp);
693cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt  CXXCtorInitializer const * const *Y
694cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt    = static_cast<CXXCtorInitializer const * const *>(Yp);
695a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
696a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  if ((*X)->getSourceOrder() < (*Y)->getSourceOrder())
697a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return -1;
698a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else if ((*X)->getSourceOrder() > (*Y)->getSourceOrder())
699a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 1;
700a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  else
701a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return 0;
702a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor}
703a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
704b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitFunctionDecl(FunctionDecl *ND) {
70501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (TypeSourceInfo *TSInfo = ND->getTypeSourceInfo()) {
70601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function declaration's syntactic components in the order
70701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // written. This requires a bit of work.
708723df245307a530da5433dfb43accf187dc3e243Abramo Bagnara    TypeLoc TL = TSInfo->getTypeLoc().IgnoreParens();
70901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    FunctionTypeLoc *FTL = dyn_cast<FunctionTypeLoc>(&TL);
71001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
71101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // If we have a function declared directly (without the use of a typedef),
71201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // visit just the return type. Otherwise, just visit the function's type
71301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // now.
71401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if ((FTL && !isa<CXXConversionDecl>(ND) && Visit(FTL->getResultLoc())) ||
71501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor        (!FTL && Visit(TL)))
71601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
71701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
718c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // Visit the nested-name-specifier, if present.
719c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (NestedNameSpecifierLoc QualifierLoc = ND->getQualifierLoc())
720c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      if (VisitNestedNameSpecifierLoc(QualifierLoc))
721c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor        return true;
72201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the declaration name.
72401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (VisitDeclarationNameInfo(ND->getNameInfo()))
72501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
72601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Visit explicitly-specified template arguments!
72801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
72901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // Visit the function parameters, if we have a function type.
73001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (FTL && VisitFunctionTypeLoc(*FTL, true))
73101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return true;
73201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Attributes?
73401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
73501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
73610620eb5164e31208fcbf0437cd79ae535ed0559Sean Hunt  if (ND->doesThisDeclarationHaveABody() && !ND->isLateTemplateParsed()) {
737a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    if (CXXConstructorDecl *Constructor = dyn_cast<CXXConstructorDecl>(ND)) {
738a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Find the initializers that were written in the source.
7395f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner      SmallVector<CXXCtorInitializer *, 4> WrittenInits;
740a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (CXXConstructorDecl::init_iterator I = Constructor->init_begin(),
741a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                          IEnd = Constructor->init_end();
742a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor           I != IEnd; ++I) {
743a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (!(*I)->isWritten())
744a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor          continue;
745a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
746a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        WrittenInits.push_back(*I);
747a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
748a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
749a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Sort the initializers in source order
750a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      llvm::array_pod_sort(WrittenInits.begin(), WrittenInits.end(),
751cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt                           &CompareCXXCtorInitializers);
752a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
753a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      // Visit the initializers in source order
754a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      for (unsigned I = 0, N = WrittenInits.size(); I != N; ++I) {
755cbb67480094b3bcb5b715acd827cbad55e2a204cSean Hunt        CXXCtorInitializer *Init = WrittenInits[I];
75600eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet        if (Init->isAnyMemberInitializer()) {
75700eb3f9c5b33e3d99aee1f8b75dd9c9678fdd66bFrancois Pichet          if (Visit(MakeCursorMemberRef(Init->getAnyMember(),
758a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor                                        Init->getMemberLocation(), TU)))
759a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
760a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        } else if (TypeSourceInfo *BaseInfo = Init->getBaseClassInfo()) {
761a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor          if (Visit(BaseInfo->getTypeLoc()))
762a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
763a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        }
764a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
765a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        // Visit the initializer value.
766a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor        if (Expr *Initializer = Init->getInit())
767a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor          if (Visit(MakeCXCursor(Initializer, ND, TU)))
768a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor            return true;
769a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      }
770a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
771a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
772a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    if (Visit(MakeCXCursor(ND->getBody(), StmtParent, TU)))
773a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return true;
774a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  }
775f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
776b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return false;
777b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
778dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
7791ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitFieldDecl(FieldDecl *D) {
7801ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7811ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
782f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7831ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *BitWidth = D->getBitWidth())
7841ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(MakeCXCursor(BitWidth, StmtParent, TU));
785f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7861ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7871ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7881ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
7891ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitVarDecl(VarDecl *D) {
7901ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (VisitDeclaratorDecl(D))
7911ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
792f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7931ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (Expr *Init = D->getInit())
7941ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return Visit(MakeCXCursor(Init, StmtParent, TU));
795f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
7961ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
7971ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
7981ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
79984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitNonTypeTemplateParmDecl(NonTypeTemplateParmDecl *D) {
80084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitDeclaratorDecl(D))
80184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
80284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
80384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited())
80484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    if (Expr *DefArg = D->getDefaultArgument())
80584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      return Visit(MakeCXCursor(DefArg, StmtParent, TU));
80684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
80784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
80884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
80984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
810fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitFunctionTemplateDecl(FunctionTemplateDecl *D) {
811fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // FIXME: Visit the "outer" template parameter lists on the FunctionDecl
812fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // before visiting these template parameters.
813fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
814fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return true;
815fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
816fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return VisitFunctionDecl(D->getTemplatedDecl());
817fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
818fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
81939d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregorbool CursorVisitor::VisitClassTemplateDecl(ClassTemplateDecl *D) {
82039d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // FIXME: Visit the "outer" template parameter lists on the TagDecl
82139d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  // before visiting these template parameters.
82239d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
82339d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return true;
82439d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82539d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  return VisitCXXRecordDecl(D->getTemplatedDecl());
82639d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor}
82739d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor
82884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregorbool CursorVisitor::VisitTemplateTemplateParmDecl(TemplateTemplateParmDecl *D) {
82984b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (VisitTemplateParameters(D->getTemplateParameters()))
83084b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
83184b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
83284b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  if (D->hasDefaultArgument() && !D->defaultArgumentWasInherited() &&
83384b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor      VisitTemplateArgumentLoc(D->getDefaultArgument()))
83484b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor    return true;
83584b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
83684b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor  return false;
83784b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor}
83884b51d77e11aaf6ea4607e01187343423ce6c8aeDouglas Gregor
8391ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCMethodDecl(ObjCMethodDecl *ND) {
8404bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor  if (TypeSourceInfo *TSInfo = ND->getResultTypeSourceInfo())
8414bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor    if (Visit(TSInfo->getTypeLoc()))
8424bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor      return true;
8434bc1cb6aa635a5bf8fae99bf69c56c724c1e786cDouglas Gregor
844f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  for (ObjCMethodDecl::param_iterator P = ND->param_begin(),
8451ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       PEnd = ND->param_end();
8461ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       P != PEnd; ++P) {
8471ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    if (Visit(MakeCXCursor(*P, TU)))
8481ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
8491ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  }
850f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8511ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (ND->isThisDeclarationADefinition() &&
8521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      Visit(MakeCXCursor(ND->getBody(), StmtParent, TU)))
8531ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    return true;
854f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
8551ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
8561ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
8571ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
858d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremeneknamespace {
859d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  struct ContainerDeclsSort {
860d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    SourceManager &SM;
861d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    ContainerDeclsSort(SourceManager &sm) : SM(sm) {}
862d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    bool operator()(Decl *A, Decl *B) {
863d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_A = A->getLocStart();
864d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L_B = B->getLocStart();
865d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      assert(L_A.isValid() && L_B.isValid());
866d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return SM.isBeforeInTranslationUnit(L_A, L_B);
867d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
868d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  };
869d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek}
870d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
871a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregorbool CursorVisitor::VisitObjCContainerDecl(ObjCContainerDecl *D) {
872d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // FIXME: Eventually convert back to just 'VisitDeclContext()'.  Essentially
873d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // an @implementation can lexically contain Decls that are not properly
874d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // nested in the AST.  When we identify such cases, we need to retrofit
875d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // this nesting here.
876d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (!DI_current)
877d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
878d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
879d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Scan the Decls that immediately come after the container
880d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // in the current DeclContext.  If any fall within the
881d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // container's lexical region, stash them into a vector
882d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // for later processing.
8835f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<Decl *, 24> DeclsInContainer;
884d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  SourceLocation EndLoc = D->getSourceRange().getEnd();
885a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  SourceManager &SM = AU->getSourceManager();
886d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (EndLoc.isValid()) {
887d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    DeclContext::decl_iterator next = *DI_current;
888d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    while (++next != DE_current) {
889d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      Decl *D_next = *next;
890d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      if (!D_next)
891d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek        break;
892d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      SourceLocation L = D_next->getLocStart();
893d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      if (!L.isValid())
894d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek        break;
895d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      if (SM.isBeforeInTranslationUnit(L, EndLoc)) {
896d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek        *DI_current = next;
897d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek        DeclsInContainer.push_back(D_next);
898d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek        continue;
899d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      }
900d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      break;
901d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    }
902d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
903d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
904d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // The common case.
905d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  if (DeclsInContainer.empty())
906d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    return VisitDeclContext(D);
907d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
908d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Get all the Decls in the DeclContext, and sort them with the
909d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // additional ones we've collected.  Then visit them.
910d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  for (DeclContext::decl_iterator I = D->decls_begin(), E = D->decls_end();
911d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek       I!=E; ++I) {
912d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    Decl *subDecl = *I;
9130582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek    if (!subDecl || subDecl->getLexicalDeclContext() != D ||
9140582c897ec7261b4c6af0fe26dc2a0b6b54d266cTed Kremenek        subDecl->getLocStart().isInvalid())
915d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
916d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    DeclsInContainer.push_back(subDecl);
917d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
918d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
919d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now sort the Decls so that they appear in lexical order.
920d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  std::sort(DeclsInContainer.begin(), DeclsInContainer.end(),
921d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek            ContainerDeclsSort(SM));
922d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek
923d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  // Now visit the decls.
9245f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  for (SmallVectorImpl<Decl*>::iterator I = DeclsInContainer.begin(),
925d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek         E = DeclsInContainer.end(); I != E; ++I) {
926d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    CXCursor Cursor = MakeCXCursor(*I, TU);
927d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    const llvm::Optional<bool> &V = shouldVisitCursor(Cursor);
928d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.hasValue())
929d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      continue;
930d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (!V.getValue())
931d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return false;
932d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek    if (Visit(Cursor, true))
933d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek      return true;
934d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  }
935d8c370ca95a3bccc36c4a6b92a4042809ff51ee4Ted Kremenek  return false;
936a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor}
937a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor
938b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCCategoryDecl(ObjCCategoryDecl *ND) {
939b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (Visit(MakeCursorObjCClassRef(ND->getClassInterface(), ND->getLocation(),
940b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                   TU)))
941b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
942f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
94378db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCCategoryDecl::protocol_loc_iterator PL = ND->protocol_loc_begin();
94478db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCCategoryDecl::protocol_iterator I = ND->protocol_begin(),
94578db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = ND->protocol_end(); I != E; ++I, ++PL)
946b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
947b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
948f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
949a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(ND);
950dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
951dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
9521ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCProtocolDecl(ObjCProtocolDecl *PID) {
9531ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  ObjCProtocolDecl::protocol_loc_iterator PL = PID->protocol_loc_begin();
9541ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  for (ObjCProtocolDecl::protocol_iterator I = PID->protocol_begin(),
9551ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       E = PID->protocol_end(); I != E; ++I, ++PL)
9561ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
9571ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
958f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
9591ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(PID);
9601ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
9611ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
96223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenekbool CursorVisitor::VisitObjCPropertyDecl(ObjCPropertyDecl *PD) {
96383cb94269015bf2770ade71e616c5322ea7e76e1Douglas Gregor  if (PD->getTypeSourceInfo() && Visit(PD->getTypeSourceInfo()->getTypeLoc()))
964fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall    return true;
965fc929208193eff37e1d3a28b1ea3bd1c9a7913e0John McCall
96623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // FIXME: This implements a workaround with @property declarations also being
96723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // installed in the DeclContext for the @interface.  Eventually this code
96823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // should be removed.
96923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCCategoryDecl *CDecl = dyn_cast<ObjCCategoryDecl>(PD->getDeclContext());
97023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!CDecl || !CDecl->IsClassExtension())
97123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
97223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
97323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCInterfaceDecl *ID = CDecl->getClassInterface();
97423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!ID)
97523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
97623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
97723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  IdentifierInfo *PropertyId = PD->getIdentifier();
97823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  ObjCPropertyDecl *prevDecl =
97923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    ObjCPropertyDecl::findPropertyDecl(cast<DeclContext>(ID), PropertyId);
98023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
98123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (!prevDecl)
98223173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    return false;
98323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
98423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // Visit synthesized methods since they will be skipped when visiting
98523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  // the @interface.
98623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getGetterMethodDecl())
987a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
98823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek      if (Visit(MakeCXCursor(MD, TU)))
98923173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
99023173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99123173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  if (ObjCMethodDecl *MD = prevDecl->getSetterMethodDecl())
992a054fb46b1fb596d1719b89d2d9a5be3c32a4b0dTed Kremenek    if (MD->isSynthesized() && MD->getLexicalDeclContext() == CDecl)
99323173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek      if (Visit(MakeCXCursor(MD, TU)))
99423173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek        return true;
99523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
99623173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek  return false;
99723173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek}
99823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek
999b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregorbool CursorVisitor::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
1000dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek  // Issue callbacks for super class.
1001b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  if (D->getSuperClass() &&
1002b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1003f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
1004b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor                                        TU)))
1005b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor    return true;
1006f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
100778db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  ObjCInterfaceDecl::protocol_loc_iterator PL = D->protocol_loc_begin();
100878db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor  for (ObjCInterfaceDecl::protocol_iterator I = D->protocol_begin(),
100978db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor         E = D->protocol_end(); I != E; ++I, ++PL)
1010b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
1011b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
1012f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
1013a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor  return VisitObjCContainerDecl(D);
1014dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1015dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10161ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplDecl(ObjCImplDecl *D) {
10171ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCContainerDecl(D);
10181ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10191ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10201ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *D) {
1021ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  // 'ID' could be null when dealing with invalid code.
1022ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek  if (ObjCInterfaceDecl *ID = D->getClassInterface())
1023ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek    if (Visit(MakeCursorObjCClassRef(ID, D->getLocation(), TU)))
1024ebfa339321f8a4df9d5011e591a615d5765107d5Ted Kremenek      return true;
1025f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10261ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
10271ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor}
10281ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor
10291ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCImplementationDecl(ObjCImplementationDecl *D) {
10301ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#if 0
10311ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // Issue callbacks for super class.
10321ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  // FIXME: No source location information!
10331ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  if (D->getSuperClass() &&
10341ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      Visit(MakeCursorObjCSuperClassRef(D->getSuperClass(),
1035f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                                        D->getSuperClassLoc(),
10361ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor                                        TU)))
1037a59e390ed6d722f0eaaa9f7eb106eaaf470df3f1Douglas Gregor    return true;
10381ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor#endif
1039f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
10401ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return VisitObjCImplDecl(D);
1041dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1042dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10431ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCForwardProtocolDecl(ObjCForwardProtocolDecl *D) {
10441ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  ObjCForwardProtocolDecl::protocol_loc_iterator PL = D->protocol_loc_begin();
10451ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  for (ObjCForwardProtocolDecl::protocol_iterator I = D->protocol_begin(),
10461ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor                                                  E = D->protocol_end();
10471ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor       I != E; ++I, ++PL)
1048b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    if (Visit(MakeCursorObjCProtocolRef(*I, *PL, TU)))
1049b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor      return true;
1050f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
1051f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return false;
1052dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
1053dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek
10541ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregorbool CursorVisitor::VisitObjCClassDecl(ObjCClassDecl *D) {
105595ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian  if (Visit(MakeCursorObjCClassRef(D->getForwardInterfaceDecl(),
105695ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian                                   D->getForwardDecl()->getLocation(), TU)))
10571ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor      return true;
10581ef2fc1888d8d14e97ebe561ccbd421282de3ce2Douglas Gregor  return false;
1059dd6bcc5f79666b9298d91a0a6ee7a0b537bde601Ted Kremenek}
10605e4bc590b0ea010e38372d0b4a0aab578a746fe6Benjamin Kramer
1061a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregorbool CursorVisitor::VisitObjCPropertyImplDecl(ObjCPropertyImplDecl *PD) {
1062a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  if (ObjCIvarDecl *Ivar = PD->getPropertyIvarDecl())
1063a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor    return Visit(MakeCursorMemberRef(Ivar, PD->getPropertyIvarDeclLoc(), TU));
1064a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
1065a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor  return false;
1066a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor}
1067a4ffd85a6684e42f900aad5459e58ad91bb88755Douglas Gregor
10688f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenekbool CursorVisitor::VisitNamespaceDecl(NamespaceDecl *D) {
10698f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  return VisitDeclContext(D);
10708f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek}
10718f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek
10726931900f43cea558c6974075256c07728dbfecc6Douglas Gregorbool CursorVisitor::VisitNamespaceAliasDecl(NamespaceAliasDecl *D) {
1073c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
10740cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
10750cfaf6a270ecd0f5c7e541a8047c87948317548bDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1076c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
10776931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10786931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  return Visit(MakeCursorNamespaceRef(D->getAliasedNamespace(),
10796931900f43cea558c6974075256c07728dbfecc6Douglas Gregor                                      D->getTargetNameLoc(), TU));
10806931900f43cea558c6974075256c07728dbfecc6Douglas Gregor}
10816931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
10827e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUsingDecl(UsingDecl *D) {
1083c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1084dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1085dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1086c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1087dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
10887e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
10891f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (Visit(MakeCursorOverloadedDeclRef(D, D->getLocation(), TU)))
10901f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return true;
10911f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
10927e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
10937e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
10947e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
10950a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregorbool CursorVisitor::VisitUsingDirectiveDecl(UsingDirectiveDecl *D) {
1096c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1097db9924191092b4d426cc066637d81698211846aaDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1098db9924191092b4d426cc066637d81698211846aaDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1099c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
11000a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11010a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  return Visit(MakeCursorNamespaceRef(D->getNominatedNamespaceAsWritten(),
11020a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor                                      D->getIdentLocation(), TU));
11030a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor}
11040a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
11057e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingValueDecl(UnresolvedUsingValueDecl *D) {
1106c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1107dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc()) {
1108dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1109c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1110dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1111c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11127e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return VisitDeclarationNameInfo(D->getNameInfo());
11137e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11147e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
11157e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypenameDecl(
11167e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor                                               UnresolvedUsingTypenameDecl *D) {
1117c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Visit nested-name-specifier.
1118dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1119dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1120c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return true;
1121c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
11227e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  return false;
11237e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor}
11247e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor
112501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitDeclarationNameInfo(DeclarationNameInfo Name) {
112601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  switch (Name.getName().getNameKind()) {
112701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::Identifier:
112801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXLiteralOperatorName:
112901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXOperatorName:
113001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXUsingDirective:
113101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
113201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
113301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConstructorName:
113401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXDestructorName:
113501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::CXXConversionFunctionName:
113601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    if (TypeSourceInfo *TSInfo = Name.getNamedTypeInfo())
113701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor      return Visit(TSInfo->getTypeLoc());
113801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
113901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
114001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCZeroArgSelector:
114101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCOneArgSelector:
114201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case clang::DeclarationName::ObjCMultiArgSelector:
114301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    // FIXME: Per-identifier location info?
114401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return false;
114501829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  }
114601829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
114701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  return false;
114801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
114901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1150c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregorbool CursorVisitor::VisitNestedNameSpecifier(NestedNameSpecifier *NNS,
1151c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                             SourceRange Range) {
1152c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // FIXME: This whole routine is a hack to work around the lack of proper
1153c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // source information in nested-name-specifiers (PR5791). Since we do have
1154c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // a beginning source location, we can visit the first component of the
1155c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // nested-name-specifier, if it's a single-token component.
1156c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  if (!NNS)
1157c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return false;
1158c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1159c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  // Get the first component in the nested-name-specifier.
1160c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  while (NestedNameSpecifier *Prefix = NNS->getPrefix())
1161c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    NNS = Prefix;
1162c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1163c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  switch (NNS->getKind()) {
1164c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Namespace:
1165c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(), Range.getBegin(),
1166c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                        TU));
1167c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
116814aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor  case NestedNameSpecifier::NamespaceAlias:
116914aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor    return Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
117014aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor                                        Range.getBegin(), TU));
117114aba76042e041b2c5e439bf4ae353a0a3c7fd73Douglas Gregor
1172c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpec: {
1173c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // If the type has a form where we know that the beginning of the source
1174c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // range matches up with a reference cursor. Visit the appropriate reference
1175c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    // cursor.
1176f4c7371fb1d3cebcfb40abad4537bb82515704eaJohn McCall    const Type *T = NNS->getAsType();
1177c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TypedefType *Typedef = dyn_cast<TypedefType>(T))
1178c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Typedef->getDecl(), Range.getBegin(), TU));
1179c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TagType *Tag = dyn_cast<TagType>(T))
1180c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return Visit(MakeCursorTypeRef(Tag->getDecl(), Range.getBegin(), TU));
1181c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    if (const TemplateSpecializationType *TST
1182c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor                                      = dyn_cast<TemplateSpecializationType>(T))
1183c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor      return VisitTemplateName(TST->getTemplateName(), Range.getBegin());
1184c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1185c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1186c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1187c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::TypeSpecWithTemplate:
1188c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Global:
1189c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  case NestedNameSpecifier::Identifier:
1190c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor    break;
1191c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  }
1192c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1193c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor  return false;
1194c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor}
1195c5ade2e3644a5822df63e442788d68c591ccdc97Douglas Gregor
1196dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregorbool
1197dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas GregorCursorVisitor::VisitNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier) {
11985f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<NestedNameSpecifierLoc, 4> Qualifiers;
1199dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  for (; Qualifier; Qualifier = Qualifier.getPrefix())
1200dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    Qualifiers.push_back(Qualifier);
1201dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1202dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  while (!Qualifiers.empty()) {
1203dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifierLoc Q = Qualifiers.pop_back_val();
1204dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    NestedNameSpecifier *NNS = Q.getNestedNameSpecifier();
1205dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    switch (NNS->getKind()) {
1206dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Namespace:
1207dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespace(),
1208c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1209dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1210dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1211dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1212dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1213dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1214dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::NamespaceAlias:
1215dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(MakeCursorNamespaceRef(NNS->getAsNamespaceAlias(),
1216c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor                                       Q.getLocalBeginLoc(),
1217dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor                                       TU)))
1218dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1219dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1220dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1221dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1222dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpec:
1223dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::TypeSpecWithTemplate:
1224dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      if (Visit(Q.getTypeLoc()))
1225dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor        return true;
1226dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1227dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1228dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1229dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Global:
1230dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    case NestedNameSpecifier::Identifier:
1231dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor      break;
1232dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor    }
1233dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  }
1234dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1235dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor  return false;
1236dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor}
1237dc355713be51fcb4ee52d9fd6b4548ceff47fadfDouglas Gregor
1238fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateParameters(
1239fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          const TemplateParameterList *Params) {
1240fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  if (!Params)
1241fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1242fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1243fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (TemplateParameterList::const_iterator P = Params->begin(),
1244fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                          PEnd = Params->end();
1245fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor       P != PEnd; ++P) {
1246fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Visit(MakeCXCursor(*P, TU)))
1247fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1248fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
1249fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1250fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1251fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1252fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
12530b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregorbool CursorVisitor::VisitTemplateName(TemplateName Name, SourceLocation Loc) {
12540b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  switch (Name.getKind()) {
12550b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::Template:
12560b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(Name.getAsTemplateDecl(), Loc, TU));
12570b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12580b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::OverloadedTemplate:
12591f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // Visit the overloaded template set.
12601f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (Visit(MakeCursorOverloadedDeclRef(Name, Loc, TU)))
12611f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return true;
12621f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
12630b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12640b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12650b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::DependentTemplate:
12660b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12670b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return false;
12680b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12690b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case TemplateName::QualifiedTemplate:
12700b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    // FIXME: Visit nested-name-specifier.
12710b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return Visit(MakeCursorTemplateRef(
12720b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                  Name.getAsQualifiedTemplateName()->getDecl(),
12730b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                                       Loc, TU));
1274146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall
1275146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall  case TemplateName::SubstTemplateTemplateParm:
1276146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall    return Visit(MakeCursorTemplateRef(
1277146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                         Name.getAsSubstTemplateTemplateParm()->getParameter(),
1278146060435c3efce95c95a092c7a1eb651cfb9ae0John McCall                                       Loc, TU));
12791aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor
12801aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor  case TemplateName::SubstTemplateTemplateParmPack:
12811aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor    return Visit(MakeCursorTemplateRef(
12821aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                  Name.getAsSubstTemplateTemplateParmPack()->getParameterPack(),
12831aee05d08b2184acadeb36de300e216390780d6cDouglas Gregor                                       Loc, TU));
12840b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  }
12850b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
12860b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  return false;
12870b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor}
12880b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
1289fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateArgumentLoc(const TemplateArgumentLoc &TAL) {
1290fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  switch (TAL.getArgument().getKind()) {
1291fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Null:
1292fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Integral:
1293fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Pack:
1294fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
129587dd697dcc8ecb64df73ae64d61b8c80ff0c157cDouglas Gregor
1296fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Type:
1297fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (TypeSourceInfo *TSInfo = TAL.getTypeSourceInfo())
1298fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return Visit(TSInfo->getTypeLoc());
1299fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1300fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1301fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Declaration:
1302fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceDeclExpression())
1303fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return Visit(MakeCXCursor(E, StmtParent, TU));
1304fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1305fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1306fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Expression:
1307fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (Expr *E = TAL.getSourceExpression())
1308fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return Visit(MakeCXCursor(E, StmtParent, TU));
1309fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return false;
1310fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1311fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case TemplateArgument::Template:
1312a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor  case TemplateArgument::TemplateExpansion:
1313b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor    if (VisitNestedNameSpecifierLoc(TAL.getTemplateQualifierLoc()))
1314b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor      return true;
1315b6744efecba58792cce20d2d7b9ee39927c5422eDouglas Gregor
1316a7fc901a2e39bfe55bfcff5934b2d9fdf9656491Douglas Gregor    return VisitTemplateName(TAL.getArgument().getAsTemplateOrTemplatePattern(),
13170b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                             TAL.getTemplateNameLoc());
1318fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  }
1319fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1320fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1321fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1322fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1323a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenekbool CursorVisitor::VisitLinkageSpecDecl(LinkageSpecDecl *D) {
1324a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  return VisitDeclContext(D);
1325a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek}
1326a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek
132701829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitQualifiedTypeLoc(QualifiedTypeLoc TL) {
132801829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  return Visit(TL.getUnqualifiedLoc());
132901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor}
133001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor
1331f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBuiltinTypeLoc(BuiltinTypeLoc TL) {
1332a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTContext &Context = AU->getASTContext();
1333f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1334f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // Some builtin types (such as Objective-C's "id", "sel", and
1335f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  // "Class") have associated declarations. Create cursors for those.
1336f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  QualType VisitType;
1337f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  switch (TL.getType()->getAs<BuiltinType>()->getKind()) {
13386b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Void:
1339f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Bool:
13406b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Char_U:
13416b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::UChar:
1342f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Char16:
1343f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Char32:
13446b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::UShort:
13456b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::UInt:
13466b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::ULong:
13476b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::ULongLong:
13486b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::UInt128:
1349f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Char_S:
13506b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::SChar:
13513f59c975aa5d047f7edd1b900b5e885c38af0ef7Chris Lattner  case BuiltinType::WChar_U:
13523f59c975aa5d047f7edd1b900b5e885c38af0ef7Chris Lattner  case BuiltinType::WChar_S:
13536b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Short:
1354f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Int:
1355f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Long:
1356c4174cc4b9b657abb77d0825de473ea29cf48297Ted Kremenek  case BuiltinType::LongLong:
13576b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Int128:
13586b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Float:
13596b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Double:
13606b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::LongDouble:
1361f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::NullPtr:
1362f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::Overload:
1363864c041e118155c2b1ce0ba36942a3da5a4a055eJohn McCall  case BuiltinType::BoundMember:
13646b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::Dependent:
13651de4d4e8cb2e9c88809fea8092bc6e835a5473d2John McCall  case BuiltinType::UnknownAny:
1366f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13676b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1368f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCId:
1369f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCIdType();
1370f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
13716b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
13726b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek  case BuiltinType::ObjCClass:
13736b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    VisitType = Context.getObjCClassType();
13746b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek    break;
13756b3b514e312b4caa403bbffe5673aa31cfbad051Ted Kremenek
1376f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  case BuiltinType::ObjCSel:
1377f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    VisitType = Context.getObjCSelType();
1378f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    break;
1379f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1380f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1381f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (!VisitType.isNull()) {
1382f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (const TypedefType *Typedef = VisitType->getAs<TypedefType>())
1383f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek      return Visit(MakeCursorTypeRef(Typedef->getDecl(), TL.getBuiltinLoc(),
1384f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                     TU));
1385f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1386f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1387f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1388f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1389f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
13907d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregorbool CursorVisitor::VisitTypedefTypeLoc(TypedefTypeLoc TL) {
1391162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  return Visit(MakeCursorTypeRef(TL.getTypedefNameDecl(), TL.getNameLoc(), TU));
13927d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor}
13937d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
1394f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitUnresolvedUsingTypeLoc(UnresolvedUsingTypeLoc TL) {
1395f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1396f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1397f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1398f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitTagTypeLoc(TagTypeLoc TL) {
13996f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis  if (TL.isDefinition())
14006f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis    return Visit(MakeCXCursor(TL.getDecl(), TU));
14016f155de99c59af890817146ec8526bafb6560f1fArgyrios Kyrtzidis
1402f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1403f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1404f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1405fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateTypeParmTypeLoc(TemplateTypeParmTypeLoc TL) {
1406960d13dde337a59dacc9dc3936c26d4aa8478986Chandler Carruth  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1407fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1408fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1409f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCInterfaceTypeLoc(ObjCInterfaceTypeLoc TL) {
1410f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(MakeCursorObjCClassRef(TL.getIFaceDecl(), TL.getNameLoc(), TU)))
1411f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1412f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1413c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return false;
1414c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall}
1415c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1416c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCallbool CursorVisitor::VisitObjCObjectTypeLoc(ObjCObjectTypeLoc TL) {
1417c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  if (TL.hasBaseTypeAsWritten() && Visit(TL.getBaseLoc()))
1418c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall    return true;
1419c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall
1420f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumProtocols(); I != N; ++I) {
1421f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    if (Visit(MakeCursorObjCProtocolRef(TL.getProtocol(I), TL.getProtocolLoc(I),
1422f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor                                        TU)))
1423f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor      return true;
1424f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  }
1425f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1426f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1427f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1428f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1429f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitObjCObjectPointerTypeLoc(ObjCObjectPointerTypeLoc TL) {
1430c12c5bba6ceb6acd4e51e7a0fc03257da9cfd44eJohn McCall  return Visit(TL.getPointeeLoc());
1431f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1432f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1433075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnarabool CursorVisitor::VisitParenTypeLoc(ParenTypeLoc TL) {
1434075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara  return Visit(TL.getInnerLoc());
1435075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara}
1436075f8f1b6bed4d1b224c74f87508534cc6392ce6Abramo Bagnara
1437f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitPointerTypeLoc(PointerTypeLoc TL) {
1438f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1439f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1440f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1441f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitBlockPointerTypeLoc(BlockPointerTypeLoc TL) {
1442f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1443f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1444f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1445f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitMemberPointerTypeLoc(MemberPointerTypeLoc TL) {
1446f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return Visit(TL.getPointeeLoc());
1447f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1448f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1449f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitLValueReferenceTypeLoc(LValueReferenceTypeLoc TL) {
1450f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1451f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1452f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1453f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitRValueReferenceTypeLoc(RValueReferenceTypeLoc TL) {
1454f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return Visit(TL.getPointeeLoc());
1455f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1456f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
14573422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidisbool CursorVisitor::VisitAttributedTypeLoc(AttributedTypeLoc TL) {
14583422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis  return Visit(TL.getModifiedLoc());
14593422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis}
14603422fbc38f35d9e486879850c5bf0175bd2eee16Argyrios Kyrtzidis
146101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregorbool CursorVisitor::VisitFunctionTypeLoc(FunctionTypeLoc TL,
146201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor                                         bool SkipResultType) {
146301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  if (!SkipResultType && Visit(TL.getResultLoc()))
1464f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1465f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1466f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
14675dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek    if (Decl *D = TL.getArg(I))
14685dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek      if (Visit(MakeCXCursor(D, TU)))
14695dbacb4179c759eef36bcaa6466b91518e3b98a9Ted Kremenek        return true;
1470f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1471f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1472f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1473f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1474f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregorbool CursorVisitor::VisitArrayTypeLoc(ArrayTypeLoc TL) {
1475f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Visit(TL.getElementLoc()))
1476f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return true;
1477f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1478f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  if (Expr *Size = TL.getSizeExpr())
1479f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor    return Visit(MakeCXCursor(Size, StmtParent, TU));
1480f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1481f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor  return false;
1482f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor}
1483f20dfbcdaaf96757ea67c94376bdca0bd64db02cDouglas Gregor
1484fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregorbool CursorVisitor::VisitTemplateSpecializationTypeLoc(
1485fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor                                             TemplateSpecializationTypeLoc TL) {
14860b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  // Visit the template name.
14870b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  if (VisitTemplateName(TL.getTypePtr()->getTemplateName(),
14880b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor                        TL.getTemplateNameLoc()))
14890b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    return true;
1490fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1491fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  // Visit the template arguments.
1492fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
1493fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
1494fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor      return true;
1495fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
1496fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  return false;
1497fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor}
1498fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor
14992332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfExprTypeLoc(TypeOfExprTypeLoc TL) {
15002332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return Visit(MakeCXCursor(TL.getUnderlyingExpr(), StmtParent, TU));
15012332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
15022332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15032332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregorbool CursorVisitor::VisitTypeOfTypeLoc(TypeOfTypeLoc TL) {
15042332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
1505ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt    return Visit(TSInfo->getTypeLoc());
1506ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1507ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  return false;
1508ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt}
1509ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt
1510ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Huntbool CursorVisitor::VisitUnaryTransformTypeLoc(UnaryTransformTypeLoc TL) {
1511ca63c200346c0ca9e00194ec6e34a5a7b0ed9321Sean Hunt  if (TypeSourceInfo *TSInfo = TL.getUnderlyingTInfo())
15122332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor    return Visit(TSInfo->getTypeLoc());
15132332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15142332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor  return false;
15152332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor}
15162332c117b1ab498322eb99c238630f28c32c7b14Douglas Gregor
15172494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregorbool CursorVisitor::VisitDependentNameTypeLoc(DependentNameTypeLoc TL) {
15182494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15192494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    return true;
15202494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
15212494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor  return false;
15222494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor}
15232494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
152494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregorbool CursorVisitor::VisitDependentTemplateSpecializationTypeLoc(
152594fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor                                    DependentTemplateSpecializationTypeLoc TL) {
152694fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the nested-name-specifier, if there is one.
152794fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  if (TL.getQualifierLoc() &&
152894fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
152994fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    return true;
153094fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
153194fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  // Visit the template arguments.
153294fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  for (unsigned I = 0, N = TL.getNumArgs(); I != N; ++I)
153394fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor    if (VisitTemplateArgumentLoc(TL.getArgLoc(I)))
153494fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor      return true;
153594fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
153694fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor  return false;
153794fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor}
153894fdffa4a572fc14ac296f5f1aae9db3734c72f1Douglas Gregor
15399e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregorbool CursorVisitor::VisitElaboratedTypeLoc(ElaboratedTypeLoc TL) {
15409e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  if (VisitNestedNameSpecifierLoc(TL.getQualifierLoc()))
15419e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor    return true;
15429e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15439e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor  return Visit(TL.getNamedTypeLoc());
15449e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor}
15459e876876afc13aa671cc11a17c19907c599b9ab9Douglas Gregor
15467536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregorbool CursorVisitor::VisitPackExpansionTypeLoc(PackExpansionTypeLoc TL) {
15477536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor  return Visit(TL.getPatternLoc());
15487536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor}
15497536dd5e6c99584481b7dab68b7e7d8df9c54054Douglas Gregor
1550427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitDecltypeTypeLoc(DecltypeTypeLoc TL) {
1551427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  if (Expr *E = TL.getUnderlyingExpr())
1552427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis    return Visit(MakeCXCursor(E, StmtParent, TU));
1553427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1554427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return false;
1555427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1556427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1557427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::VisitInjectedClassNameTypeLoc(InjectedClassNameTypeLoc TL) {
1558427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit(MakeCursorTypeRef(TL.getDecl(), TL.getNameLoc(), TU));
1559427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1560427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1561427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis#define DEFAULT_TYPELOC_IMPL(CLASS, PARENT) \
1562427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidisbool CursorVisitor::Visit##CLASS##TypeLoc(CLASS##TypeLoc TL) { \
1563427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis  return Visit##PARENT##Loc(TL); \
1564427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis}
1565427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
1566427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Complex, Type)
1567427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ConstantArray, ArrayType)
1568427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(IncompleteArray, ArrayType)
1569427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(VariableArray, ArrayType)
1570427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedArray, ArrayType)
1571427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(DependentSizedExtVector, Type)
1572427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Vector, Type)
1573427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(ExtVector, VectorType)
1574427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionProto, FunctionType)
1575427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(FunctionNoProto, FunctionType)
1576427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Record, TagType)
1577427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Enum, TagType)
1578427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParm, Type)
1579427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(SubstTemplateTypeParmPack, Type)
1580427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios KyrtzidisDEFAULT_TYPELOC_IMPL(Auto, Type)
1581427964e15f1b9595659cea3fcb4dd808a00f37b5Argyrios Kyrtzidis
15823064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenekbool CursorVisitor::VisitCXXRecordDecl(CXXRecordDecl *D) {
1583c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  // Visit the nested-name-specifier, if present.
1584c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = D->getQualifierLoc())
1585c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor    if (VisitNestedNameSpecifierLoc(QualifierLoc))
1586c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor      return true;
1587c22b5fff39a7520207f165fb16a27a34b944bd9cDouglas Gregor
15883064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  if (D->isDefinition()) {
15893064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    for (CXXRecordDecl::base_class_iterator I = D->bases_begin(),
15903064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek         E = D->bases_end(); I != E; ++I) {
15913064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      if (Visit(cxcursor::MakeCursorCXXBaseSpecifier(I, TU)))
15923064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek        return true;
15933064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
15943064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  }
15953064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
15963064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  return VisitTagDecl(D);
15973064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek}
15983064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek
159909dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenekbool CursorVisitor::VisitAttributes(Decl *D) {
1600cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt  for (AttrVec::const_iterator i = D->attr_begin(), e = D->attr_end();
1601cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt       i != e; ++i)
1602cf807c4dfdb23e8fa3f400e0b24ef5b79db7a530Sean Hunt    if (Visit(MakeCXCursor(*i, D, TU)))
160309dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek        return true;
160409dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
160509dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek  return false;
160609dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek}
160709dfa37dda8e430fb6129bfd70365a51b4e12243Ted Kremenek
1608c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1609c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Data-recursive visitor methods.
1610c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
1611c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
161228a719433411ef782b582946823bc648ddcc4533Ted Kremeneknamespace {
1613035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#define DEF_JOB(NAME, DATA, KIND)\
1614035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass NAME : public VisitorJob {\
1615035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:\
1616035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  NAME(DATA *d, CXCursor parent) : VisitorJob(parent, VisitorJob::KIND, d) {} \
1617035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) { return VJ->getKind() == KIND; }\
1618f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DATA *get() const { return static_cast<DATA*>(data[0]); }\
1619035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1620035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1621035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(StmtVisit, Stmt, StmtVisitKind)
1622035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(MemberExprParts, MemberExpr, MemberExprPartsKind)
1623e4979ccb5960608edce73f3b274eb7c2de15dac5Ted KremenekDEF_JOB(DeclRefExprParts, DeclRefExpr, DeclRefExprPartsKind)
1624035dc41b509fcc470ceb6764aa64837505a2ece3Ted KremenekDEF_JOB(OverloadExprParts, OverloadExpr, OverloadExprPartsKind)
162560608ec12d17168a3d1f415409a6a6eaf6d94508Ted KremenekDEF_JOB(ExplicitTemplateArgsVisit, ExplicitTemplateArgumentList,
162660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        ExplicitTemplateArgsVisitKind)
162794d96291cd041adc5731a2294828a9c20e450b74Douglas GregorDEF_JOB(SizeOfPackExprParts, SizeOfPackExpr, SizeOfPackExprPartsKind)
1628035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek#undef DEF_JOB
1629035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1630035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass DeclVisit : public VisitorJob {
1631035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1632035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  DeclVisit(Decl *d, CXCursor parent, bool isFirst) :
1633035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::DeclVisitKind,
1634035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               d, isFirst ? (void*) 1 : (void*) 0) {}
1635035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
163682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    return VJ->getKind() == DeclVisitKind;
1637035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1638f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  Decl *get() const { return static_cast<Decl*>(data[0]); }
1639f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  bool isFirst() const { return data[1] ? true : false; }
1640035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1641035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekclass TypeLocVisit : public VisitorJob {
1642035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekpublic:
1643035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  TypeLocVisit(TypeLoc tl, CXCursor parent) :
1644035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    VisitorJob(parent, VisitorJob::TypeLocVisitKind,
1645035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek               tl.getType().getAsOpaquePtr(), tl.getOpaqueData()) {}
1646035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1647035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1648035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return VJ->getKind() == TypeLocVisitKind;
1649035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1650035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
165182f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek  TypeLoc get() const {
1652f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    QualType T = QualType::getFromOpaquePtr(data[0]);
1653f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return TypeLoc(T, data[1]);
1654035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1655035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek};
1656035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek
1657ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekclass LabelRefVisit : public VisitorJob {
1658ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekpublic:
1659ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelRefVisit(LabelDecl *LD, SourceLocation labelLoc, CXCursor parent)
1660ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner    : VisitorJob(parent, VisitorJob::LabelRefVisitKind, LD,
1661dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 labelLoc.getPtrEncoding()) {}
1662ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek
1663ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1664ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek    return VJ->getKind() == VisitorJob::LabelRefVisitKind;
1665ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  }
1666ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  LabelDecl *get() const { return static_cast<LabelDecl*>(data[0]); }
1667ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  SourceLocation getLoc() const {
1668dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin    return SourceLocation::getFromPtrEncoding(data[1]); }
1669f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek};
1670f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1671f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorclass NestedNameSpecifierLocVisit : public VisitorJob {
1672f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorpublic:
1673f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLocVisit(NestedNameSpecifierLoc Qualifier, CXCursor parent)
1674f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    : VisitorJob(parent, VisitorJob::NestedNameSpecifierLocVisitKind,
1675f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getNestedNameSpecifier(),
1676f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                 Qualifier.getOpaqueData()) { }
1677f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1678f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  static bool classof(const VisitorJob *VJ) {
1679f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return VJ->getKind() == VisitorJob::NestedNameSpecifierLocVisitKind;
1680f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1681f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1682f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  NestedNameSpecifierLoc get() const {
1683f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    return NestedNameSpecifierLoc(static_cast<NestedNameSpecifier*>(data[0]),
1684f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor                                  data[1]);
1685f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  }
1686f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor};
1687f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1688f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekclass DeclarationNameInfoVisit : public VisitorJob {
1689f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekpublic:
1690f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfoVisit(Stmt *S, CXCursor parent)
1691f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    : VisitorJob(parent, VisitorJob::DeclarationNameInfoVisitKind, S) {}
1692f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  static bool classof(const VisitorJob *VJ) {
1693f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    return VJ->getKind() == VisitorJob::DeclarationNameInfoVisitKind;
1694f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1695f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  DeclarationNameInfo get() const {
1696f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    Stmt *S = static_cast<Stmt*>(data[0]);
1697f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    switch (S->getStmtClass()) {
1698f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    default:
1699f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      llvm_unreachable("Unhandled Stmt");
1700f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::CXXDependentScopeMemberExprClass:
1701f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<CXXDependentScopeMemberExpr>(S)->getMemberNameInfo();
1702f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    case Stmt::DependentScopeDeclRefExprClass:
1703f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      return cast<DependentScopeDeclRefExpr>(S)->getNameInfo();
1704f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    }
1705f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  }
1706ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek};
1707cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekclass MemberRefVisit : public VisitorJob {
1708cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekpublic:
1709cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  MemberRefVisit(FieldDecl *D, SourceLocation L, CXCursor parent)
1710cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    : VisitorJob(parent, VisitorJob::MemberRefVisitKind, D,
1711dec0984fce504a39a7f085774fb67cfd9957be58Jeffrey Yasskin                 L.getPtrEncoding()) {}
1712cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  static bool classof(const VisitorJob *VJ) {
1713cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return VJ->getKind() == VisitorJob::MemberRefVisitKind;
1714cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1715cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  FieldDecl *get() const {
1716cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return static_cast<FieldDecl*>(data[0]);
1717cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1718cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  SourceLocation getLoc() const {
1719cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    return SourceLocation::getFromRawEncoding((unsigned)(uintptr_t) data[1]);
1720cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1721cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek};
172228a719433411ef782b582946823bc648ddcc4533Ted Kremenekclass EnqueueVisitor : public StmtVisitor<EnqueueVisitor, void> {
172328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitorWorkList &WL;
172428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  CXCursor Parent;
172528a719433411ef782b582946823bc648ddcc4533Ted Kremenekpublic:
172628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueVisitor(VisitorWorkList &wl, CXCursor parent)
172728a719433411ef782b582946823bc648ddcc4533Ted Kremenek    : WL(wl), Parent(parent) {}
172828a719433411ef782b582946823bc648ddcc4533Ted Kremenek
1729ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitAddrLabelExpr(AddrLabelExpr *E);
173073d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitBlockExpr(BlockExpr *B);
173128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCompoundLiteralExpr(CompoundLiteralExpr *E);
1732083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  void VisitCompoundStmt(CompoundStmt *S);
173311b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXDefaultArgExpr(CXXDefaultArgExpr *E) { /* Do nothing. */ }
1734f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitCXXDependentScopeMemberExpr(CXXDependentScopeMemberExpr *E);
173511b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  void VisitCXXNewExpr(CXXNewExpr *E);
17366d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  void VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E);
173728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitCXXOperatorCallExpr(CXXOperatorCallExpr *E);
1738cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E);
173973d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E);
1740b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  void VisitCXXTypeidExpr(CXXTypeidExpr *E);
174155b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  void VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr *E);
17421e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  void VisitCXXUuidofExpr(CXXUuidofExpr *E);
1743e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  void VisitDeclRefExpr(DeclRefExpr *D);
1744035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  void VisitDeclStmt(DeclStmt *S);
1745f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E);
1746cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitDesignatedInitExpr(DesignatedInitExpr *E);
174728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitExplicitCastExpr(ExplicitCastExpr *E);
174828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitForStmt(ForStmt *FS);
1749ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  void VisitGotoStmt(GotoStmt *GS);
175028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitIfStmt(IfStmt *If);
175128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitInitListExpr(InitListExpr *IE);
175228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitMemberExpr(MemberExpr *M);
1753cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void VisitOffsetOfExpr(OffsetOfExpr *E);
175473d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  void VisitObjCEncodeExpr(ObjCEncodeExpr *E);
175528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitObjCMessageExpr(ObjCMessageExpr *M);
175628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitOverloadExpr(OverloadExpr *E);
1757f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne  void VisitUnaryExprOrTypeTraitExpr(UnaryExprOrTypeTraitExpr *E);
175828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitStmt(Stmt *S);
175928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitSwitchStmt(SwitchStmt *S);
176028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitWhileStmt(WhileStmt *W);
17612939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  void VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E);
17626ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  void VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E);
176321ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  void VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E);
1764552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  void VisitExpressionTraitExpr(ExpressionTraitExpr *E);
176528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U);
17669d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  void VisitVAArgExpr(VAArgExpr *E);
176794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  void VisitSizeOfPackExpr(SizeOfPackExpr *E);
1768ee8aff06f6a96214731de17b2cb6df407c6c1820Douglas Gregor
176928a719433411ef782b582946823bc648ddcc4533Ted Kremenekprivate:
1770f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  void AddDeclarationNameInfo(Stmt *S);
1771f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  void AddNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier);
177260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  void AddExplicitTemplateArgs(const ExplicitTemplateArgumentList *A);
1773cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  void AddMemberRef(FieldDecl *D, SourceLocation L);
177428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void AddStmt(Stmt *S);
1775035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  void AddDecl(Decl *D, bool isFirst = true);
177628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void AddTypeLoc(TypeSourceInfo *TI);
177728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  void EnqueueChildren(Stmt *S);
177828a719433411ef782b582946823bc648ddcc4533Ted Kremenek};
177928a719433411ef782b582946823bc648ddcc4533Ted Kremenek} // end anonyous namespace
178028a719433411ef782b582946823bc648ddcc4533Ted Kremenek
1781f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::AddDeclarationNameInfo(Stmt *S) {
1782f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  // 'S' should always be non-null, since it comes from the
1783f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  // statement we are visiting.
1784f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  WL.push_back(DeclarationNameInfoVisit(S, Parent));
1785f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
1786f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
1787f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregorvoid
1788f3db29fff6a583ecda823cf909ab7737d8d30129Douglas GregorEnqueueVisitor::AddNestedNameSpecifierLoc(NestedNameSpecifierLoc Qualifier) {
1789f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  if (Qualifier)
1790f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    WL.push_back(NestedNameSpecifierLocVisit(Qualifier, Parent));
1791f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor}
1792f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
179328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::AddStmt(Stmt *S) {
179428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (S)
179528a719433411ef782b582946823bc648ddcc4533Ted Kremenek    WL.push_back(StmtVisit(S, Parent));
179628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1797035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekvoid EnqueueVisitor::AddDecl(Decl *D, bool isFirst) {
179828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (D)
1799035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    WL.push_back(DeclVisit(D, Parent, isFirst));
180028a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
180160608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenekvoid EnqueueVisitor::
180260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  AddExplicitTemplateArgs(const ExplicitTemplateArgumentList *A) {
180360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  if (A)
180460608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek    WL.push_back(ExplicitTemplateArgsVisit(
180560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek                        const_cast<ExplicitTemplateArgumentList*>(A), Parent));
180660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek}
1807cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::AddMemberRef(FieldDecl *D, SourceLocation L) {
1808cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  if (D)
1809cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    WL.push_back(MemberRefVisit(D, L, Parent));
1810cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
181128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::AddTypeLoc(TypeSourceInfo *TI) {
181228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (TI)
181328a719433411ef782b582946823bc648ddcc4533Ted Kremenek    WL.push_back(TypeLocVisit(TI->getTypeLoc(), Parent));
181428a719433411ef782b582946823bc648ddcc4533Ted Kremenek }
181528a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::EnqueueChildren(Stmt *S) {
1816a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  unsigned size = WL.size();
18177502c1d3ce8bb97bcc4f7bebef507040bd93b26fJohn McCall  for (Stmt::child_range Child = S->children(); Child; ++Child) {
181828a719433411ef782b582946823bc648ddcc4533Ted Kremenek    AddStmt(*Child);
1819a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  }
1820a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  if (size == WL.size())
1821a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek    return;
1822a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  // Now reverse the entries we just added.  This will match the DFS
1823a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  // ordering performed by the worklist.
1824a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  VisitorWorkList::iterator I = WL.begin() + size, E = WL.end();
1825a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek  std::reverse(I, E);
1826a6b70435ae49f2c7da1ad2b200c1aeb9f8761fddTed Kremenek}
1827ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekvoid EnqueueVisitor::VisitAddrLabelExpr(AddrLabelExpr *E) {
1828ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  WL.push_back(LabelRefVisit(E->getLabel(), E->getLabelLoc(), Parent));
1829ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek}
183073d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitBlockExpr(BlockExpr *B) {
183173d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddDecl(B->getBlockDecl());
183273d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
183328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitCompoundLiteralExpr(CompoundLiteralExpr *E) {
183428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(E);
183528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(E->getTypeSourceInfo());
183628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1837083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenekvoid EnqueueVisitor::VisitCompoundStmt(CompoundStmt *S) {
1838083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  for (CompoundStmt::reverse_body_iterator I = S->body_rbegin(),
1839083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek        E = S->body_rend(); I != E; ++I) {
1840083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek    AddStmt(*I);
1841083c7e2d564033af87e507fbbd02f1c77ff462b1Ted Kremenek  }
184211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek}
1843f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::
1844f64d80306144f978148ba92f36f7cea7b671dd34Ted KremenekVisitCXXDependentScopeMemberExpr(CXXDependentScopeMemberExpr *E) {
1845f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
1846f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddDeclarationNameInfo(E);
18477c3179cf463c3b3b8c21dbb955f933ba50b74f28Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = E->getQualifierLoc())
18487c3179cf463c3b3b8c21dbb955f933ba50b74f28Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1849f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  if (!E->isImplicitAccess())
1850f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek    AddStmt(E->getBase());
1851f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
185211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenekvoid EnqueueVisitor::VisitCXXNewExpr(CXXNewExpr *E) {
185311b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the initializer or constructor arguments.
185411b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  for (unsigned I = E->getNumConstructorArgs(); I > 0; --I)
185511b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek    AddStmt(E->getConstructorArg(I-1));
185611b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the array size, if any.
185711b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddStmt(E->getArraySize());
185811b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the allocated type.
185911b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  AddTypeLoc(E->getAllocatedTypeSourceInfo());
186011b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  // Enqueue the placement arguments.
186111b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek  for (unsigned I = E->getNumPlacementArgs(); I > 0; --I)
186211b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek    AddStmt(E->getPlacementArg(I-1));
186311b8e3e7fa67795acc968a9d5ebfd687feaf2b2cTed Kremenek}
186428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitCXXOperatorCallExpr(CXXOperatorCallExpr *CE) {
18658b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek  for (unsigned I = CE->getNumArgs(); I > 1 /* Yes, this is 1 */; --I)
18668b8d8c90f2d8ac651d14b57f116d20b3c911ac7fTed Kremenek    AddStmt(CE->getArg(I-1));
186728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getCallee());
186828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(CE->getArg(0));
186928a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1870cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitCXXPseudoDestructorExpr(CXXPseudoDestructorExpr *E) {
1871cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the name of the type being destroyed.
1872cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getDestroyedTypeInfo());
1873cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the scope type that looks disturbingly like the nested-name-specifier
1874cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // but isn't.
1875cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getScopeTypeInfo());
1876cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the nested-name-specifier.
1877f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor  if (NestedNameSpecifierLoc QualifierLoc = E->getQualifierLoc())
1878f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor    AddNestedNameSpecifierLoc(QualifierLoc);
1879cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit base expression.
1880cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getBase());
1881cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
18826d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenekvoid EnqueueVisitor::VisitCXXScalarValueInitExpr(CXXScalarValueInitExpr *E) {
18836d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
18846d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
188573d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitCXXTemporaryObjectExpr(CXXTemporaryObjectExpr *E) {
188673d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  EnqueueChildren(E);
188773d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
188873d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
1889b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenekvoid EnqueueVisitor::VisitCXXTypeidExpr(CXXTypeidExpr *E) {
1890b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  EnqueueChildren(E);
1891b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek  if (E->isTypeOperand())
1892b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
1893b8dd1cad52be63b18092bd9d9335cbeee3f5de9fTed Kremenek}
189455b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek
189555b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenekvoid EnqueueVisitor::VisitCXXUnresolvedConstructExpr(CXXUnresolvedConstructExpr
189655b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek                                                     *E) {
189755b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  EnqueueChildren(E);
189855b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
189955b933ae3b3efc29987b8bd92714257f00c4b53aTed Kremenek}
19001e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenekvoid EnqueueVisitor::VisitCXXUuidofExpr(CXXUuidofExpr *E) {
19011e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  EnqueueChildren(E);
19021e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek  if (E->isTypeOperand())
19031e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek    AddTypeLoc(E->getTypeOperandSourceInfo());
19041e7e877091187556bb6d644ab2b7c00a628121ebTed Kremenek}
1905e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenekvoid EnqueueVisitor::VisitDeclRefExpr(DeclRefExpr *DR) {
190660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  if (DR->hasExplicitTemplateArgs()) {
190760608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek    AddExplicitTemplateArgs(&DR->getExplicitTemplateArgs());
190860608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  }
1909e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek  WL.push_back(DeclRefExprParts(DR, Parent));
1910e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek}
1911f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenekvoid EnqueueVisitor::VisitDependentScopeDeclRefExpr(DependentScopeDeclRefExpr *E) {
1912f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
1913f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek  AddDeclarationNameInfo(E);
191400cf3cc2718671aa48e8da264a523b0058a8591eDouglas Gregor  AddNestedNameSpecifierLoc(E->getQualifierLoc());
1915f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek}
1916035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenekvoid EnqueueVisitor::VisitDeclStmt(DeclStmt *S) {
1917035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  unsigned size = WL.size();
1918035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  bool isFirst = true;
1919035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  for (DeclStmt::decl_iterator D = S->decl_begin(), DEnd = S->decl_end();
1920035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek       D != DEnd; ++D) {
1921035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    AddDecl(*D, isFirst);
1922035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    isFirst = false;
1923035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  }
1924035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  if (size == WL.size())
1925035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek    return;
1926035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // Now reverse the entries we just added.  This will match the DFS
1927035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  // ordering performed by the worklist.
1928035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  VisitorWorkList::iterator I = WL.begin() + size, E = WL.end();
1929035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek  std::reverse(I, E);
1930035dc41b509fcc470ceb6764aa64837505a2ece3Ted Kremenek}
1931cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitDesignatedInitExpr(DesignatedInitExpr *E) {
1932cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddStmt(E->getInit());
1933cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  typedef DesignatedInitExpr::Designator Designator;
1934cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (DesignatedInitExpr::reverse_designators_iterator
1935cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D = E->designators_rbegin(), DEnd = E->designators_rend();
1936cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek         D != DEnd; ++D) {
1937cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isFieldDesignator()) {
1938cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      if (FieldDecl *Field = D->getField())
1939cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        AddMemberRef(Field, D->getFieldLoc());
1940cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1941cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1942cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    if (D->isArrayDesignator()) {
1943cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getArrayIndex(*D));
1944cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
1945cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
1946cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    assert(D->isArrayRangeDesignator() && "Unknown designator kind");
1947cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeEnd(*D));
1948cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    AddStmt(E->getArrayRangeStart(*D));
1949cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
1950cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
195128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitExplicitCastExpr(ExplicitCastExpr *E) {
195228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(E);
195328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(E->getTypeInfoAsWritten());
195428a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
195528a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitForStmt(ForStmt *FS) {
195628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getBody());
195728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInc());
195828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getCond());
195928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(FS->getConditionVariable());
196028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(FS->getInit());
196128a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1962ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenekvoid EnqueueVisitor::VisitGotoStmt(GotoStmt *GS) {
1963ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek  WL.push_back(LabelRefVisit(GS->getLabel(), GS->getLabelLoc(), Parent));
1964ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek}
196528a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitIfStmt(IfStmt *If) {
196628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getElse());
196728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getThen());
196828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(If->getCond());
196928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(If->getConditionVariable());
197028a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
197128a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitInitListExpr(InitListExpr *IE) {
197228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  // We care about the syntactic form of the initializer list, only.
197328a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (InitListExpr *Syntactic = IE->getSyntacticForm())
197428a719433411ef782b582946823bc648ddcc4533Ted Kremenek    IE = Syntactic;
197528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(IE);
197628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
197728a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitMemberExpr(MemberExpr *M) {
197889629a746019a42797495b091711a1d68467e88aDouglas Gregor  WL.push_back(MemberExprParts(M, Parent));
197989629a746019a42797495b091711a1d68467e88aDouglas Gregor
198089629a746019a42797495b091711a1d68467e88aDouglas Gregor  // If the base of the member access expression is an implicit 'this', don't
198189629a746019a42797495b091711a1d68467e88aDouglas Gregor  // visit it.
198289629a746019a42797495b091711a1d68467e88aDouglas Gregor  // FIXME: If we ever want to show these implicit accesses, this will be
198389629a746019a42797495b091711a1d68467e88aDouglas Gregor  // unfortunate. However, clang_getCursor() relies on this behavior.
198475e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor  if (!M->isImplicitAccess())
198575e85048e73fcde2ce9d8a48dfdb1220e132eb59Douglas Gregor    AddStmt(M->getBase());
198628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
198773d15c452e675b684b7eee4f2096e386e59397aaTed Kremenekvoid EnqueueVisitor::VisitObjCEncodeExpr(ObjCEncodeExpr *E) {
198873d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek  AddTypeLoc(E->getEncodedTypeSourceInfo());
198973d15c452e675b684b7eee4f2096e386e59397aaTed Kremenek}
199028a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitObjCMessageExpr(ObjCMessageExpr *M) {
199128a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(M);
199228a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddTypeLoc(M->getClassReceiverTypeInfo());
199328a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
1994cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekvoid EnqueueVisitor::VisitOffsetOfExpr(OffsetOfExpr *E) {
1995cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the components of the offsetof expression.
1996cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  for (unsigned N = E->getNumComponents(), I = N; I > 0; --I) {
1997cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    typedef OffsetOfExpr::OffsetOfNode OffsetOfNode;
1998cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    const OffsetOfNode &Node = E->getComponent(I-1);
1999cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    switch (Node.getKind()) {
2000cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Array:
2001cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      AddStmt(E->getIndexExpr(Node.getArrayExprIndex()));
2002cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2003cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Field:
200406dec892b5300b43263d25c5476b506c9d6cfbadAbramo Bagnara      AddMemberRef(Node.getField(), Node.getSourceRange().getEnd());
2005cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      break;
2006cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Identifier:
2007cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    case OffsetOfNode::Base:
2008cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      continue;
2009cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek    }
2010cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  }
2011cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  // Visit the type into which we're computing the offset.
2012cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek  AddTypeLoc(E->getTypeSourceInfo());
2013cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek}
201428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitOverloadExpr(OverloadExpr *E) {
201560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek  AddExplicitTemplateArgs(E->getOptionalExplicitTemplateArgs());
20166045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek  WL.push_back(OverloadExprParts(E, Parent));
20176045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek}
2018f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbournevoid EnqueueVisitor::VisitUnaryExprOrTypeTraitExpr(
2019f4e3cfbe8abd124be6341ef5d714819b4fbd9082Peter Collingbourne                                              UnaryExprOrTypeTraitExpr *E) {
20206d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  EnqueueChildren(E);
20216d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek  if (E->isArgumentType())
20226d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek    AddTypeLoc(E->getArgumentTypeInfo());
20236d0a00d9b02499d6c1253ea03d4dc7c32f8f289eTed Kremenek}
202428a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitStmt(Stmt *S) {
202528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueChildren(S);
202628a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
202728a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitSwitchStmt(SwitchStmt *S) {
202828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getBody());
202928a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(S->getCond());
203028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(S->getConditionVariable());
203128a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
2032fafa75aebadef8d6b44a920e3f40529f150a5574Ted Kremenek
203328a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitWhileStmt(WhileStmt *W) {
203428a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getBody());
203528a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddStmt(W->getCond());
203628a719433411ef782b582946823bc648ddcc4533Ted Kremenek  AddDecl(W->getConditionVariable());
203728a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
203821ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
20392939b6f356161f572712d4d6310b65f9599e3675Ted Kremenekvoid EnqueueVisitor::VisitUnaryTypeTraitExpr(UnaryTypeTraitExpr *E) {
20402939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek  AddTypeLoc(E->getQueriedTypeSourceInfo());
20412939b6f356161f572712d4d6310b65f9599e3675Ted Kremenek}
20426ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
20436ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichetvoid EnqueueVisitor::VisitBinaryTypeTraitExpr(BinaryTypeTraitExpr *E) {
20446ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet  AddTypeLoc(E->getRhsTypeSourceInfo());
20450a03a3f98b14006a54bcac9e8908a7c9f50e519fFrancois Pichet  AddTypeLoc(E->getLhsTypeSourceInfo());
20466ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet}
20476ad6f2848d7652ab2991286eb48be440d3493b28Francois Pichet
204821ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegleyvoid EnqueueVisitor::VisitArrayTypeTraitExpr(ArrayTypeTraitExpr *E) {
204921ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley  AddTypeLoc(E->getQueriedTypeSourceInfo());
205021ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley}
205121ff2e516b0e0bc8c1dbf965cb3d44bac3c64330John Wiegley
2052552622067dc45013d240f73952fece703f5e63bdJohn Wiegleyvoid EnqueueVisitor::VisitExpressionTraitExpr(ExpressionTraitExpr *E) {
2053552622067dc45013d240f73952fece703f5e63bdJohn Wiegley  EnqueueChildren(E);
2054552622067dc45013d240f73952fece703f5e63bdJohn Wiegley}
2055552622067dc45013d240f73952fece703f5e63bdJohn Wiegley
205628a719433411ef782b582946823bc648ddcc4533Ted Kremenekvoid EnqueueVisitor::VisitUnresolvedMemberExpr(UnresolvedMemberExpr *U) {
205728a719433411ef782b582946823bc648ddcc4533Ted Kremenek  VisitOverloadExpr(U);
205828a719433411ef782b582946823bc648ddcc4533Ted Kremenek  if (!U->isImplicitAccess())
205928a719433411ef782b582946823bc648ddcc4533Ted Kremenek    AddStmt(U->getBase());
206028a719433411ef782b582946823bc648ddcc4533Ted Kremenek}
20619d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenekvoid EnqueueVisitor::VisitVAArgExpr(VAArgExpr *E) {
20629d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddStmt(E->getSubExpr());
20639d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek  AddTypeLoc(E->getWrittenTypeInfo());
20649d3bf79d94c961af95144ef63bbd6f9f3f32c59aTed Kremenek}
206594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregorvoid EnqueueVisitor::VisitSizeOfPackExpr(SizeOfPackExpr *E) {
206694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  WL.push_back(SizeOfPackExprParts(E, Parent));
206794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor}
20686045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek
2069c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekvoid CursorVisitor::EnqueueWorkList(VisitorWorkList &WL, Stmt *S) {
207028a719433411ef782b582946823bc648ddcc4533Ted Kremenek  EnqueueVisitor(WL, MakeCXCursor(S, StmtParent, TU)).Visit(S);
2071c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2072c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2073c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::IsInRegionOfInterest(CXCursor C) {
2074c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  if (RegionOfInterest.isValid()) {
2075c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SourceRange Range = getRawCursorExtent(C);
2076c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    if (Range.isInvalid() || CompareRegionOfInterest(Range))
2077c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      return false;
2078c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2079c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return true;
2080c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2081c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2082c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenekbool CursorVisitor::RunVisitorWorkList(VisitorWorkList &WL) {
2083c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  while (!WL.empty()) {
2084c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Dequeue the worklist item.
208582f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    VisitorJob LI = WL.back();
208682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek    WL.pop_back();
208782f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek
2088c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    // Set the Parent field, then back to its old value once we're done.
2089c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    SetParentRAII SetParent(Parent, StmtParent, LI.getParent());
2090c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2091c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    switch (LI.getKind()) {
2092f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      case VisitorJob::DeclVisitKind: {
209382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Decl *D = cast<DeclVisit>(&LI)->get();
2094f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        if (!D)
2095f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek          continue;
2096f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2097f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // For now, perform default visitation for Decls.
209882f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        if (Visit(MakeCXCursor(D, TU, cast<DeclVisit>(&LI)->isFirst())))
2099f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek            return true;
2100f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek
2101f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        continue;
2102f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek      }
210360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      case VisitorJob::ExplicitTemplateArgsVisitKind: {
210460608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        const ExplicitTemplateArgumentList *ArgList =
210560608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          cast<ExplicitTemplateArgsVisit>(&LI)->get();
210660608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        for (const TemplateArgumentLoc *Arg = ArgList->getTemplateArgs(),
210760608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               *ArgEnd = Arg + ArgList->NumTemplateArgs;
210860608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek               Arg != ArgEnd; ++Arg) {
210960608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek          if (VisitTemplateArgumentLoc(*Arg))
211060608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek            return true;
211160608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        }
211260608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek        continue;
211360608ec12d17168a3d1f415409a6a6eaf6d94508Ted Kremenek      }
2114cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      case VisitorJob::TypeLocVisitKind: {
2115cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        // Perform default visitation for TypeLocs.
211682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        if (Visit(cast<TypeLocVisit>(&LI)->get()))
2117cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek          return true;
2118cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek        continue;
2119cdb4caf6bd8beb562e169b4d3f6c604c4e4528f8Ted Kremenek      }
2120ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      case VisitorJob::LabelRefVisitKind: {
2121ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner        LabelDecl *LS = cast<LabelRefVisit>(&LI)->get();
2122e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        if (LabelStmt *stmt = LS->getStmt()) {
2123e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          if (Visit(MakeCursorLabelRef(stmt, cast<LabelRefVisit>(&LI)->getLoc(),
2124e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek                                       TU))) {
2125e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek            return true;
2126e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek          }
2127e7455016e5fae3db7e2d88a61633f76ab05fc9d3Ted Kremenek        }
2128ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek        continue;
2129ae1fd6fd47ce3fbb878155af6f517adaeae15a51Ted Kremenek      }
213047695c8ad8424851f62e0d4a983b45b15daee1c5Ted Kremenek
2131f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      case VisitorJob::NestedNameSpecifierLocVisitKind: {
2132f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        NestedNameSpecifierLocVisit *V = cast<NestedNameSpecifierLocVisit>(&LI);
2133f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        if (VisitNestedNameSpecifierLoc(V->get()))
2134f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor          return true;
2135f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor        continue;
2136f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor      }
2137f3db29fff6a583ecda823cf909ab7737d8d30129Douglas Gregor
2138f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      case VisitorJob::DeclarationNameInfoVisitKind: {
2139f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        if (VisitDeclarationNameInfo(cast<DeclarationNameInfoVisit>(&LI)
2140f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek                                     ->get()))
2141f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek          return true;
2142f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek        continue;
2143f64d80306144f978148ba92f36f7cea7b671dd34Ted Kremenek      }
2144cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      case VisitorJob::MemberRefVisitKind: {
2145cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        MemberRefVisit *V = cast<MemberRefVisit>(&LI);
2146cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (Visit(MakeCursorMemberRef(V->get(), V->getLoc(), TU)))
2147cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          return true;
2148cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        continue;
2149cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek      }
2150c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::StmtVisitKind: {
215182f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        Stmt *S = cast<StmtVisit>(&LI)->get();
21528c269ac75569454a049385b1246140db5f2b6faaTed Kremenek        if (!S)
21538c269ac75569454a049385b1246140db5f2b6faaTed Kremenek          continue;
21548c269ac75569454a049385b1246140db5f2b6faaTed Kremenek
2155f1107457c7d5cb57718725c51dfed4bf2ba6ee9eTed Kremenek        // Update the current cursor.
2156c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        CXCursor Cursor = MakeCXCursor(S, StmtParent, TU);
2157cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        if (!IsInRegionOfInterest(Cursor))
2158cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          continue;
2159cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek        switch (Visitor(Cursor, Parent, ClientData)) {
2160cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Break: return true;
2161cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Continue: break;
2162cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek          case CXChildVisit_Recurse:
2163cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenek            EnqueueWorkList(WL, S);
216482f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek            break;
2165c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
216682f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        continue;
2167c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2168c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      case VisitorJob::MemberExprPartsKind: {
2169c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Handle the other pieces in the MemberExpr besides the base.
217082f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        MemberExpr *M = cast<MemberExprParts>(&LI)->get();
2171c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2172c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the nested-name-specifier
217340d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = M->getQualifierLoc())
217440d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2175c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            return true;
2176c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2177c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the declaration name.
2178c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (VisitDeclarationNameInfo(M->getMemberNameInfo()))
2179c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          return true;
2180c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2181c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        // Visit the explicitly-specified template arguments, if any.
2182c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        if (M->hasExplicitTemplateArgs()) {
2183c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          for (const TemplateArgumentLoc *Arg = M->getTemplateArgs(),
2184c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               *ArgEnd = Arg + M->getNumTemplateArgs();
2185c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek               Arg != ArgEnd; ++Arg) {
2186c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek            if (VisitTemplateArgumentLoc(*Arg))
2187c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek              return true;
2188c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek          }
2189c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        }
2190c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek        continue;
2191c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek      }
2192e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      case VisitorJob::DeclRefExprPartsKind: {
219382f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        DeclRefExpr *DR = cast<DeclRefExprParts>(&LI)->get();
2194e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit nested-name-specifier, if present.
219540d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = DR->getQualifierLoc())
219640d96a69c0e1e8c10f92d450c305a7aae696ca9cDouglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
2197e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek            return true;
2198e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        // Visit declaration name.
2199e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        if (VisitDeclarationNameInfo(DR->getNameInfo()))
2200e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek          return true;
2201e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek        continue;
2202e4979ccb5960608edce73f3b274eb7c2de15dac5Ted Kremenek      }
22036045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      case VisitorJob::OverloadExprPartsKind: {
220482f3c50fa163f99d1407849e556d3859a09afd78Ted Kremenek        OverloadExpr *O = cast<OverloadExprParts>(&LI)->get();
22056045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the nested-name-specifier.
22064c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor        if (NestedNameSpecifierLoc QualifierLoc = O->getQualifierLoc())
22074c9be89bb615ec07eb3ed507c8fa9d0baa8a5ad7Douglas Gregor          if (VisitNestedNameSpecifierLoc(QualifierLoc))
22086045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek            return true;
22096045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the declaration name.
22106045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (VisitDeclarationNameInfo(O->getNameInfo()))
22116045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22126045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        // Visit the overloaded declaration reference.
22136045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        if (Visit(MakeCursorOverloadedDeclRef(O, TU)))
22146045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek          return true;
22156045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek        continue;
22166045878f1fa7fad19d826634617991db99d3bd22Ted Kremenek      }
221794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      case VisitorJob::SizeOfPackExprPartsKind: {
221894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        SizeOfPackExpr *E = cast<SizeOfPackExprParts>(&LI)->get();
221994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        NamedDecl *Pack = E->getPack();
222094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTypeParmDecl>(Pack)) {
222194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTypeRef(cast<TemplateTypeParmDecl>(Pack),
222294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                      E->getPackLoc(), TU)))
222394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
222494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
222594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
222694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
222794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
222894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        if (isa<TemplateTemplateParmDecl>(Pack)) {
222994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          if (Visit(MakeCursorTemplateRef(cast<TemplateTemplateParmDecl>(Pack),
223094d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor                                          E->getPackLoc(), TU)))
223194d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor            return true;
223294d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
223394d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor          continue;
223494d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        }
223594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
223694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // Non-type template parameter packs and function parameter packs are
223794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        // treated like DeclRefExpr cursors.
223894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        continue;
223994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      }
2240c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek    }
2241c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  }
2242c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek  return false;
2243c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2244c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
2245cdba6595a61a7bd31f504260abf63c900a759d0fTed Kremenekbool CursorVisitor::Visit(Stmt *S) {
2246d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  VisitorWorkList *WL = 0;
2247d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  if (!WorkListFreeList.empty()) {
2248d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = WorkListFreeList.back();
2249d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL->clear();
2250d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListFreeList.pop_back();
2251d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2252d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  else {
2253d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WL = new VisitorWorkList();
2254d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek    WorkListCache.push_back(WL);
2255d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  }
2256d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  EnqueueWorkList(*WL, S);
2257d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  bool result = RunVisitorWorkList(*WL);
2258d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  WorkListFreeList.push_back(WL);
2259d1ded66c4eda8d170222071dec7ebba78bd86ea4Ted Kremenek  return result;
2260c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek}
2261c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
226248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichetnamespace {
226348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichettypedef llvm::SmallVector<SourceRange, 4> RefNamePieces;
226448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois PichetRefNamePieces buildPieces(unsigned NameFlags, bool IsMemberRefExpr,
226548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const DeclarationNameInfo &NI,
226648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const SourceRange &QLoc,
226748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                          const ExplicitTemplateArgumentList *TemplateArgs = 0){
226848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantQualifier = NameFlags & CXNameRange_WantQualifier;
226948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantTemplateArgs = NameFlags & CXNameRange_WantTemplateArgs;
227048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const bool WantSinglePiece = NameFlags & CXNameRange_WantSinglePiece;
227148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
227248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  const DeclarationName::NameKind Kind = NI.getName().getNameKind();
227348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
227448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  RefNamePieces Pieces;
227548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
227648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantQualifier && QLoc.isValid())
227748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(QLoc);
227848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
227948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind != DeclarationName::CXXOperatorName || IsMemberRefExpr)
228048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(NI.getLoc());
228148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
228248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantTemplateArgs && TemplateArgs)
228348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceRange(TemplateArgs->LAngleLoc,
228448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                                 TemplateArgs->RAngleLoc));
228548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
228648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (Kind == DeclarationName::CXXOperatorName) {
228748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
228848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.BeginOpNameLoc));
228948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(SourceLocation::getFromRawEncoding(
229048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet                       NI.getInfo().CXXOperatorName.EndOpNameLoc));
229148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
229248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229348a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  if (WantSinglePiece) {
229448a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    SourceRange R(Pieces.front().getBegin(), Pieces.back().getEnd());
229548a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.clear();
229648a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet    Pieces.push_back(R);
229748a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  }
229848a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
229948a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet  return Pieces;
230048a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
230148a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet}
230248a8d14fc6f064a5297024c2b34733a4080b2efeFrancois Pichet
2303c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2304c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek// Misc. API hooks.
2305c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek//===----------------------------------------------------------------------===//
2306c0e1d923231d42fb0ac86e56a29ffc5c754657caTed Kremenek
23078c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic llvm::sys::Mutex EnableMultithreadingMutex;
23088c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregorstatic bool EnabledMultithreading;
23098c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
23105e4bc590b0ea010e38372d0b4a0aab578a746fe6Benjamin Kramerextern "C" {
23110a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas GregorCXIndex clang_createIndex(int excludeDeclarationsFromPCH,
23120a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor                          int displayDiagnostics) {
231348615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // Disable pretty stack trace functionality, which will otherwise be a very
231448615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  // poor citizen of the world and set up all sorts of signal handlers.
231548615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar  llvm::DisablePrettyStackTrace = true;
231648615ffe41e41e0cc232dfb61289b707ece37ea1Daniel Dunbar
2317c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // We use crash recovery to make some of our APIs more reliable, implicitly
2318c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  // enable it.
2319c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar  llvm::CrashRecoveryContext::Enable();
2320c7df4f344d78fe0d7591be3756712e777b3d2e8dDaniel Dunbar
23218c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  // Enable support for multithreading in LLVM.
23228c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  {
23238c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    llvm::sys::ScopedLock L(EnableMultithreadingMutex);
23248c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    if (!EnabledMultithreading) {
23258c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      llvm::llvm_start_multithreaded();
23268c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor      EnabledMultithreading = true;
23278c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor    }
23288c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor  }
23298c8d5412cddcc1c45beb0353d91d7894db74e585Douglas Gregor
2330a030b7cf5e6aad5889b1b662b6979840bc75f87fDouglas Gregor  CIndexer *CIdxr = new CIndexer();
2331e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  if (excludeDeclarationsFromPCH)
2332e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    CIdxr->setOnlyLocalDecls();
23330a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  if (displayDiagnostics)
23340a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor    CIdxr->setDisplayDiagnostics();
2335e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  return CIdxr;
2336600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2337600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
23389ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeIndex(CXIndex CIdx) {
23392b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (CIdx)
23402b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    delete static_cast<CIndexer *>(CIdx);
23412bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
23422bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff
2343d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenekvoid clang_toggleCrashRecovery(unsigned isEnabled) {
2344d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  if (isEnabled)
2345d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Enable();
2346d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek  else
2347d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek    llvm::CrashRecoveryContext::Disable();
2348d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek}
2349d2427ddf00aacdc288c386f3882e0821ca9bd27bTed Kremenek
23509ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit clang_createTranslationUnit(CXIndex CIdx,
2351a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                              const char *ast_filename) {
23522b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
23532b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor    return 0;
2354f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
23557d1d49d2971b20a97b3c2a301470b9eaaa130137Douglas Gregor  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2356389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOptions FileSystemOpts;
2357389db16c63eec6ecfa9b235155252d8da766e94eArgyrios Kyrtzidis  FileSystemOpts.WorkingDir = CXXIdx->getWorkingDirectory();
23580d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
235928019772db70d4547be05a042eb950bc910f134fDouglas Gregor  llvm::IntrusiveRefCntPtr<Diagnostic> Diags;
2360a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *TU = ASTUnit::LoadFromASTFile(ast_filename, Diags, FileSystemOpts,
2361a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                  CXXIdx->getOnlyLocalDecls(),
2362a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                  0, 0, true);
2363a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return MakeCXTranslationUnit(TU);
2364600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
2365600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2366b1c031be513705d924038f497279b9b599868ba1Douglas Gregorunsigned clang_defaultEditingTranslationUnitOptions() {
23672a2c50b330e7754499f42173616a36865b5f313bDouglas Gregor  return CXTranslationUnit_PrecompiledPreamble |
2368b5af843a20e237ad1a13ad66a867e200695b8c8eDouglas Gregor         CXTranslationUnit_CacheCompletionResults;
2369b1c031be513705d924038f497279b9b599868ba1Douglas Gregor}
2370b1c031be513705d924038f497279b9b599868ba1Douglas Gregor
23719ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXTranslationUnit
23729ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarclang_createTranslationUnitFromSourceFile(CXIndex CIdx,
23739ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          const char *source_filename,
23749ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          int num_command_line_args,
23752ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                          const char * const *command_line_args,
23764db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor                                          unsigned num_unsaved_files,
2377a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor                                          struct CXUnsavedFile *unsaved_files) {
2378dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor  unsigned Options = CXTranslationUnit_DetailedPreprocessingRecord |
2379ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth                     CXTranslationUnit_NestedMacroExpansions;
23805a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor  return clang_parseTranslationUnit(CIdx, source_filename,
23815a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    command_line_args, num_command_line_args,
23825a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor                                    unsaved_files, num_unsaved_files,
2383dca8ee8b7bc86076916a3a80f553f7a4e98c14afDouglas Gregor                                    Options);
23845a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor}
238519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
238619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbarstruct ParseTranslationUnitInfo {
238719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx;
238819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename;
23892ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char *const *command_line_args;
239019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args;
239119ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
239219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files;
239319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options;
239419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXTranslationUnit result;
239519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar};
2396b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_parseTranslationUnit_Impl(void *UserData) {
239719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo *PTUI =
239819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    static_cast<ParseTranslationUnitInfo*>(UserData);
239919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  CXIndex CIdx = PTUI->CIdx;
240019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  const char *source_filename = PTUI->source_filename;
24012ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor  const char * const *command_line_args = PTUI->command_line_args;
240219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  int num_command_line_args = PTUI->num_command_line_args;
240319ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  struct CXUnsavedFile *unsaved_files = PTUI->unsaved_files;
240419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned num_unsaved_files = PTUI->num_unsaved_files;
240519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  unsigned options = PTUI->options;
240619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  PTUI->result = 0;
24075a43021ac491bf091494167127772a20d9a9bb48Douglas Gregor
24082b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CIdx)
240919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return;
2410f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2411e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff  CIndexer *CXXIdx = static_cast<CIndexer *>(CIdx);
2412e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff
241344c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  bool PrecompilePreamble = options & CXTranslationUnit_PrecompiledPreamble;
2414467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  // FIXME: Add a flag for modules.
2415467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor  TranslationUnitKind TUKind
2416467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor    = (options & CXTranslationUnit_Incomplete)? TU_Prefix : TU_Complete;
241787c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor  bool CacheCodeCompetionResults
241887c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor    = options & CXTranslationUnit_CacheCompletionResults;
241987c08a5d6b9e1e44ae6f554df40139d3a6f60b33Douglas Gregor
24205352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  // Configure the diagnostics.
24215352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  DiagnosticOptions DiagOpts;
242225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::IntrusiveRefCntPtr<Diagnostic>
242325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Diags(CompilerInstance::createDiagnostics(DiagOpts, num_command_line_args,
242425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                                command_line_args));
242525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
242625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
242725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<Diagnostic,
242825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    llvm::CrashRecoveryContextReleaseRefCleanup<Diagnostic> >
242925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    DiagCleanup(Diags.getPtr());
243025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
243125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<ASTUnit::RemappedFile> >
243225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
243325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
243425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
243525a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
243625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
2437f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
24384db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
24395f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2440f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    const llvm::MemoryBuffer *Buffer
2441a0a270c0f1c0a4e3482438bdc5f4a7bd3d25f0a6Chris Lattner      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
244225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
244325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
24444db64a461cb3442934afe43c83ed3f17f7c11c1dDouglas Gregor  }
2445f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
244625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<const char *> >
244725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args(new std::vector<const char*>());
244825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
244925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this method.
245025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<std::vector<const char*> >
245125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    ArgsCleanup(Args.get());
245225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
245352ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // Since the Clang C library is primarily used by batch tools dealing with
245452ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // (often very broken) source code, where spell-checking can have a
245552ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // significant negative impact on performance (particularly when
245652ddc5df59a26570fbca47d269f82954ae3397d1Douglas Gregor  // precompiled headers are involved), we disable it by default.
2457b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  // Only do this if we haven't found a spell-checking-related argument.
2458b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  bool FoundSpellCheckingArgument = false;
2459b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  for (int I = 0; I != num_command_line_args; ++I) {
2460b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (strcmp(command_line_args[I], "-fno-spell-checking") == 0 ||
2461b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        strcmp(command_line_args[I], "-fspell-checking") == 0) {
2462b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      FoundSpellCheckingArgument = true;
2463b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      break;
2464e56b4baeba5097852e04bc41ca2e0396cf729955Steve Naroff    }
2465b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  }
2466b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  if (!FoundSpellCheckingArgument)
246725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-fno-spell-checking");
2468b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
246925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  Args->insert(Args->end(), command_line_args,
247025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek               command_line_args + num_command_line_args);
2471d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2472c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // The 'source_filename' argument is optional.  If the caller does not
2473c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // specify it then it is assumed that the source file is specified
2474c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // in the actual argument list.
2475c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // Put the source file after command_line_args otherwise if '-x' flag is
2476c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  // present it will be unused.
2477c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis  if (source_filename)
247825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back(source_filename);
2479c842955140adf7b0a7c73c61390bf580c50a182dArgyrios Kyrtzidis
248044c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  // Do we need the detailed preprocessing record?
2481ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth  bool NestedMacroExpansions = false;
248244c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  if (options & CXTranslationUnit_DetailedPreprocessingRecord) {
248325a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-Xclang");
248425a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    Args->push_back("-detailed-preprocessing-record");
2485ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth    NestedMacroExpansions
2486ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth      = (options & CXTranslationUnit_NestedMacroExpansions);
248744c181aec37789f25f6c15543c164416f72e562aDouglas Gregor  }
248844c181aec37789f25f6c15543c164416f72e562aDouglas Gregor
2489026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  unsigned NumErrors = Diags->getClient()->getNumErrors();
2490b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor  llvm::OwningPtr<ASTUnit> Unit(
24914ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek    ASTUnit::LoadFromCommandLine(Args->size() ? &(*Args)[0] : 0
24924ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 /* vector::data() not portable */,
24934ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 Args->size() ? (&(*Args)[0] + Args->size()) :0,
2494b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 Diags,
2495b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getClangResourcesPath(),
2496b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 CXXIdx->getOnlyLocalDecls(),
2497e47be3e9682e82da15059006f43c7f3c021e4fffDouglas Gregor                                 /*CaptureDiagnostics=*/true,
24984ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                                 RemappedFiles->size() ? &(*RemappedFiles)[0]:0,
249925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                 RemappedFiles->size(),
2500299a4a967b02c9f0d0d94ad8560e3ced893f9116Argyrios Kyrtzidis                                 /*RemappedFilesKeepOriginalName=*/true,
2501b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                 PrecompilePreamble,
2502467dc88512b4ba4bb16e274ea3771dc1415d31daDouglas Gregor                                 TUKind,
250399ba202f659e1885fa5ee114f97c97cf6a857491Douglas Gregor                                 CacheCodeCompetionResults,
2504ba7537febdf1bc1cc617e1f1746f2644feba6274Chandler Carruth                                 NestedMacroExpansions));
2505b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor
2506026f6911bb985c800a54446de9f6da8745ae025aArgyrios Kyrtzidis  if (NumErrors != Diags->getClient()->getNumErrors()) {
2507b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    // Make sure to check that 'Unit' is non-NULL.
2508b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    if (CXXIdx->getDisplayDiagnostics() && Unit.get()) {
2509b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      for (ASTUnit::stored_diag_iterator D = Unit->stored_diag_begin(),
2510b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                      DEnd = Unit->stored_diag_end();
2511b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor           D != DEnd; ++D) {
2512b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        CXStoredDiagnostic Diag(*D, Unit->getASTContext().getLangOptions());
2513b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        CXString Msg = clang_formatDiagnostic(&Diag,
2514b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor                                    clang_defaultDiagnosticDisplayOptions());
2515b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        fprintf(stderr, "%s\n", clang_getCString(Msg));
2516b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor        clang_disposeString(Msg);
2517b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      }
2518274f1906f12ebf8fcc179701deeda6d3271120c1Douglas Gregor#ifdef LLVM_ON_WIN32
2519b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // On Windows, force a flush, since there may be multiple copies of
2520b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // stderr and stdout in the file system, all with different buffers
2521b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      // but writing to the same device.
2522b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor      fflush(stderr);
2523b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor#endif
2524b10daedb8fc4c6f70a66dbc1a6eea01684bb4e77Douglas Gregor    }
2525a88084b78fd4ca5d3d858c14b02414f8cc399f02Douglas Gregor  }
2526d93256e55673a17d18543397ec462416acb13792Douglas Gregor
2527a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  PTUI->result = MakeCXTranslationUnit(Unit.take());
252819ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar}
252919ffd492a31a25fb691098bf79f317e5f3edf177Daniel DunbarCXTranslationUnit clang_parseTranslationUnit(CXIndex CIdx,
253019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             const char *source_filename,
25312ef6944d529c94824f5bf96f65665f5bee30f5a2Douglas Gregor                                         const char * const *command_line_args,
253219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             int num_command_line_args,
25339e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                            struct CXUnsavedFile *unsaved_files,
253419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned num_unsaved_files,
253519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar                                             unsigned options) {
253619ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  ParseTranslationUnitInfo PTUI = { CIdx, source_filename, command_line_args,
25379e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_command_line_args, unsaved_files,
25389e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                    num_unsaved_files, options, 0 };
253919ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  llvm::CrashRecoveryContext CRC;
254019ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
2541bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_parseTranslationUnit_Impl, &PTUI)) {
254260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "libclang: crash detected during parsing: {\n");
254360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'source_filename' : '%s'\n", source_filename);
254460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'command_line_args' : [");
254560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (int i = 0; i != num_command_line_args; ++i) {
254660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
254760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
254860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "'%s'", command_line_args[i]);
254960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
255060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
255160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'unsaved_files' : [");
255260a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    for (unsigned i = 0; i != num_unsaved_files; ++i) {
255360a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      if (i)
255460a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar        fprintf(stderr, ", ");
255560a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar      fprintf(stderr, "('%s', '...', %ld)", unsaved_files[i].Filename,
255660a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar              unsaved_files[i].Length);
255760a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    }
255860a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "],\n");
255960a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "  'options' : %d,\n", options);
256060a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar    fprintf(stderr, "}\n");
256160a4543d42b6b8564a5650345de7beb7e99c60f3Daniel Dunbar
256219ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar    return 0;
25636df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE")) {
25646df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(PTUI.result);
256519ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  }
25666df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
256719ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar  return PTUI.result;
25685b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff}
25695b7d8e254f6c2855b37b5521c0aee0a560dab237Steve Naroff
25701999844e7a18786e61e619e1dc6c789827541863Douglas Gregorunsigned clang_defaultSaveOptions(CXTranslationUnit TU) {
25711999844e7a18786e61e619e1dc6c789827541863Douglas Gregor  return CXSaveTranslationUnit_None;
25721999844e7a18786e61e619e1dc6c789827541863Douglas Gregor}
25731999844e7a18786e61e619e1dc6c789827541863Douglas Gregor
25741999844e7a18786e61e619e1dc6c789827541863Douglas Gregorint clang_saveTranslationUnit(CXTranslationUnit TU, const char *FileName,
25751999844e7a18786e61e619e1dc6c789827541863Douglas Gregor                              unsigned options) {
25767ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor  if (!TU)
257739c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor    return CXSaveError_InvalidTU;
25787ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor
257939c411fa229b2a6747b92f945d1702ee674d3470Douglas Gregor  CXSaveError result = static_cast<ASTUnit *>(TU->TUData)->Save(FileName);
25806df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  if (getenv("LIBCLANG_RESOURCE_USAGE"))
25816df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
25826df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  return result;
25837ae2faafd30524ef5f863bb3b8701977888839bbDouglas Gregor}
258419ffd492a31a25fb691098bf79f317e5f3edf177Daniel Dunbar
25859ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarvoid clang_disposeTranslationUnit(CXTranslationUnit CTUnit) {
2586ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  if (CTUnit) {
2587ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // If the translation unit has been marked as unsafe to free, just discard
2588ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    // it.
2589a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    if (static_cast<ASTUnit *>(CTUnit->TUData)->isUnsafeToFree())
2590ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar      return;
2591ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2592a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete static_cast<ASTUnit *>(CTUnit->TUData);
2593a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    disposeCXStringPool(CTUnit->StringPool);
2594a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    delete CTUnit;
2595ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  }
25962bd6b9f298afb16a2aec035ebd7b29af7c5c3da8Steve Naroff}
25970d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
2598e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregorunsigned clang_defaultReparseOptions(CXTranslationUnit TU) {
2599e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor  return CXReparse_None;
2600e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor}
2601e1e13bf568a7e37c95eda6fcfa626659a06e67b1Douglas Gregor
2602ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarstruct ReparseTranslationUnitInfo {
2603ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU;
2604ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files;
2605ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files;
2606ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options;
2607ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  int result;
2608ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar};
2609593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2610b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbarstatic void clang_reparseTranslationUnit_Impl(void *UserData) {
2611ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo *RTUI =
2612ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    static_cast<ReparseTranslationUnitInfo*>(UserData);
2613ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  CXTranslationUnit TU = RTUI->TU;
2614ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned num_unsaved_files = RTUI->num_unsaved_files;
2615ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  struct CXUnsavedFile *unsaved_files = RTUI->unsaved_files;
2616ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  unsigned options = RTUI->options;
2617ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  (void) options;
2618ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  RTUI->result = 1;
2619ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2620abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  if (!TU)
2621ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return;
2622593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2623a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
2624593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
2625abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
262625a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::OwningPtr<std::vector<ASTUnit::RemappedFile> >
262725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles(new std::vector<ASTUnit::RemappedFile>());
262825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
262925a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  // Recover resources if we crash before exiting this function.
263025a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek  llvm::CrashRecoveryContextCleanupRegistrar<
263125a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    std::vector<ASTUnit::RemappedFile> > RemappedCleanup(RemappedFiles.get());
263225a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek
2633abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  for (unsigned I = 0; I != num_unsaved_files; ++I) {
26345f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    StringRef Data(unsaved_files[I].Contents, unsaved_files[I].Length);
2635abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor    const llvm::MemoryBuffer *Buffer
26361abc6bca3d7fb0e7b1e40fbcad6cfb5e10594548Douglas Gregor      = llvm::MemoryBuffer::getMemBufferCopy(Data, unsaved_files[I].Filename);
263725a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek    RemappedFiles->push_back(std::make_pair(unsaved_files[I].Filename,
263825a11e1c5fad62dbad25a265e334720157e3fbc1Ted Kremenek                                            Buffer));
2639abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor  }
2640abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor
26414ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek  if (!CXXUnit->Reparse(RemappedFiles->size() ? &(*RemappedFiles)[0] : 0,
26424ee9926671913ea6189ef9840a244d7c4385a7d5Ted Kremenek                        RemappedFiles->size()))
2643593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor    RTUI->result = 0;
2644abc563f554951259bbe0315055cad92ee14d87e4Douglas Gregor}
2645593b0c1047f1323ebbda78ae38e96f976241c663Douglas Gregor
2646ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbarint clang_reparseTranslationUnit(CXTranslationUnit TU,
2647ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned num_unsaved_files,
2648ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 struct CXUnsavedFile *unsaved_files,
2649ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                 unsigned options) {
2650ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  ReparseTranslationUnitInfo RTUI = { TU, num_unsaved_files, unsaved_files,
2651ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar                                      options, 0 };
2652ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  llvm::CrashRecoveryContext CRC;
2653ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2654bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  if (!RunSafely(CRC, clang_reparseTranslationUnit_Impl, &RTUI)) {
2655b1fd3458680bc9c8988dee8967e9c0709fef3945Daniel Dunbar    fprintf(stderr, "libclang: crash detected during reparsing\n");
2656a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    static_cast<ASTUnit *>(TU->TUData)->setUnsafeToFree(true);
2657ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar    return 1;
26586df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  } else if (getenv("LIBCLANG_RESOURCE_USAGE"))
26596df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    PrintLibclangResourceUsage(TU);
26601dfb26af4d6aa4f7818e256659a79f1ec2cba784Ted Kremenek
2661ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar  return RTUI.result;
2662ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar}
2663ea94bbc4769697143e717df9b0310f874102b6c1Daniel Dunbar
2664df95a13ec73d2cdaea79555cb412d767f4963120Douglas Gregor
26659ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getTranslationUnitSpelling(CXTranslationUnit CTUnit) {
26662b37c9e6cabaf3317922af13e6d12061242f04efDouglas Gregor  if (!CTUnit)
2667ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
2668f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2669a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(CTUnit->TUData);
2670ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(CXXUnit->getOriginalSourceFileName(), true);
2671af08ddc8f1c53fed8d8d0ad82aa2a0bb7d654bd1Steve Naroff}
26721eb79b58e56b99cf557d5d353586a10c5360364dDaniel Dunbar
26737eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas GregorCXCursor clang_getTranslationUnitCursor(CXTranslationUnit TU) {
2674b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  CXCursor Result = { CXCursor_TranslationUnit, { 0, 0, TU } };
26757eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return Result;
26767eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
26777eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
2678fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2679600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff
2680fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
26811db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor// CXSourceLocation and CXSourceRange Operations.
26821db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor//===----------------------------------------------------------------------===//
26831db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
2684b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregorextern "C" {
2685b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXSourceLocation clang_getNullLocation() {
26865352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXSourceLocation Result = { { 0, 0 }, 0 };
2687b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  return Result;
2688b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2689b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor
2690b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregorunsigned clang_equalLocations(CXSourceLocation loc1, CXSourceLocation loc2) {
269190a6b9e1e4d4d1995ada044e319d6e722b07a6b4Daniel Dunbar  return (loc1.ptr_data[0] == loc2.ptr_data[0] &&
269290a6b9e1e4d4d1995ada044e319d6e722b07a6b4Daniel Dunbar          loc1.ptr_data[1] == loc2.ptr_data[1] &&
269390a6b9e1e4d4d1995ada044e319d6e722b07a6b4Daniel Dunbar          loc1.int_data == loc2.int_data);
2694b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2695b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor
2696b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXSourceLocation clang_getLocation(CXTranslationUnit tu,
2697b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor                                   CXFile file,
2698b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor                                   unsigned line,
2699b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor                                   unsigned column) {
270042748ec5cb2d75fe0dbb3a6db5aee6c11b5dc190Douglas Gregor  if (!tu || !file)
2701b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor    return clang_getNullLocation();
270242748ec5cb2d75fe0dbb3a6db5aee6c11b5dc190Douglas Gregor
270386a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor  bool Logging = ::getenv("LIBCLANG_LOGGING");
2704a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
270586a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor  const FileEntry *File = static_cast<const FileEntry *>(file);
2706507097ec40105ed927cb5a744fad98f5875aacacArgyrios Kyrtzidis  SourceLocation SLoc = CXXUnit->getLocation(File, line, column);
270786a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor  if (SLoc.isInvalid()) {
270886a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor    if (Logging)
270986a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor      llvm::errs() << "clang_getLocation(\"" << File->getName()
271086a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor                   << "\", " << line << ", " << column << ") = invalid\n";
271186a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor    return clang_getNullLocation();
271286a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor  }
271386a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor
271486a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor  if (Logging)
271586a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor    llvm::errs() << "clang_getLocation(\"" << File->getName()
271686a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor                 << "\", " << line << ", " << column << ") = "
271786a4d0dd6a630639aab7715323ed068940e650afDouglas Gregor                 << SLoc.getRawEncoding() << "\n";
271883889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall
271983889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall  return cxloc::translateSourceLocation(CXXUnit->getASTContext(), SLoc);
272083889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall}
272183889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall
272283889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid ChisnallCXSourceLocation clang_getLocationForOffset(CXTranslationUnit tu,
272383889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall                                            CXFile file,
272483889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall                                            unsigned offset) {
272583889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall  if (!tu || !file)
272683889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall    return clang_getNullLocation();
272783889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall
2728a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2729507097ec40105ed927cb5a744fad98f5875aacacArgyrios Kyrtzidis  SourceLocation SLoc
2730507097ec40105ed927cb5a744fad98f5875aacacArgyrios Kyrtzidis    = CXXUnit->getLocation(static_cast<const FileEntry *>(file), offset);
273183889a7f1f338e343ef72aeeef9c27f7b62c0f0fDavid Chisnall  if (SLoc.isInvalid()) return clang_getNullLocation();
2732f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
27331a9a0bc472ee4fec72ee8be8b575fb66ca600d1bTed Kremenek  return cxloc::translateSourceLocation(CXXUnit->getASTContext(), SLoc);
2734b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2735b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor
27365352ac06d8f6194825bb2a99ffa009b61bafb503Douglas GregorCXSourceRange clang_getNullRange() {
27375352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  CXSourceRange Result = { { 0, 0 }, 0, 0 };
27385352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  return Result;
27395352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor}
2740d52864bd33c66aacc84133460d8c9c0dfcdd5c18Daniel Dunbar
2741b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXSourceRange clang_getRange(CXSourceLocation begin, CXSourceLocation end) {
27425352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  if (begin.ptr_data[0] != end.ptr_data[0] ||
27435352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor      begin.ptr_data[1] != end.ptr_data[1])
27445352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
2745f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2746f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  CXSourceRange Result = { { begin.ptr_data[0], begin.ptr_data[1] },
27475352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor                           begin.int_data, end.int_data };
2748b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  return Result;
2749b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2750ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor
2751ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregorunsigned clang_equalRanges(CXSourceRange range1, CXSourceRange range2)
2752ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor{
2753ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor  return range1.ptr_data[0] == range2.ptr_data[0]
2754ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor      && range1.ptr_data[1] == range2.ptr_data[1]
2755ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor      && range1.begin_int_data == range2.begin_int_data
2756ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor      && range1.end_int_data == range2.end_int_data;
2757ab4e83b904d81d8ab1f8c594655822a023cad87dDouglas Gregor}
27589d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek} // end: extern "C"
2759b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor
27609d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenekstatic void createNullLocation(CXFile *file, unsigned *line,
27619d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek                               unsigned *column, unsigned *offset) {
27629d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  if (file)
27639d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek   *file = 0;
27649d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  if (line)
27659d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek   *line = 0;
27669d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  if (column)
27679d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek   *column = 0;
27689d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  if (offset)
27699d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek   *offset = 0;
27709d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  return;
27719d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek}
27729d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek
27739d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenekextern "C" {
277420174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruthvoid clang_getExpansionLocation(CXSourceLocation location,
277520174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                CXFile *file,
277620174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                unsigned *line,
277720174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                unsigned *column,
277820174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                unsigned *offset) {
27791db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  SourceLocation Loc = SourceLocation::getFromRawEncoding(location.int_data);
27801db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
2781bb4a61a121ba1ee91eb5725881d98249704bb0aaDaniel Dunbar  if (!location.ptr_data[0] || Loc.isInvalid()) {
27829d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek    createNullLocation(file, line, column, offset);
278346766dc31c09d89024de5aba9e22112a56eadbdfDouglas Gregor    return;
278446766dc31c09d89024de5aba9e22112a56eadbdfDouglas Gregor  }
278546766dc31c09d89024de5aba9e22112a56eadbdfDouglas Gregor
2786bb4a61a121ba1ee91eb5725881d98249704bb0aaDaniel Dunbar  const SourceManager &SM =
2787bb4a61a121ba1ee91eb5725881d98249704bb0aaDaniel Dunbar    *static_cast<const SourceManager*>(location.ptr_data[0]);
278820174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth  SourceLocation ExpansionLoc = SM.getExpansionLoc(Loc);
27891db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
2790cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth  // Check that the FileID is invalid on the expansion location.
27919d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  // This can manifest in invalid code.
279220174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth  FileID fileID = SM.getFileID(ExpansionLoc);
2793e23ac65af568ffe611b0990818ac3a57c856a4d8Douglas Gregor  bool Invalid = false;
2794e23ac65af568ffe611b0990818ac3a57c856a4d8Douglas Gregor  const SrcMgr::SLocEntry &sloc = SM.getSLocEntry(fileID, &Invalid);
2795e23ac65af568ffe611b0990818ac3a57c856a4d8Douglas Gregor  if (!sloc.isFile() || Invalid) {
27969d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek    createNullLocation(file, line, column, offset);
27979d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek    return;
27989d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek  }
27999d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek
28001db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  if (file)
28019d5a165d301cc9df68631e624322dd2a962f65b3Ted Kremenek    *file = (void *)SM.getFileEntryForSLocEntry(sloc);
28021db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  if (line)
280320174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    *line = SM.getExpansionLineNumber(ExpansionLoc);
28041db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  if (column)
280520174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    *column = SM.getExpansionColumnNumber(ExpansionLoc);
2806e69517ce61638f12c9abe4605753a45275ac4e37Douglas Gregor  if (offset)
280720174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    *offset = SM.getDecomposedLoc(ExpansionLoc).second;
280820174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth}
280920174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth
2810e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidisvoid clang_getPresumedLocation(CXSourceLocation location,
2811e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis                               CXString *filename,
2812e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis                               unsigned *line,
2813e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis                               unsigned *column) {
2814e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis  SourceLocation Loc = SourceLocation::getFromRawEncoding(location.int_data);
2815e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis
2816e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis  if (!location.ptr_data[0] || Loc.isInvalid()) {
2817e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (filename)
2818e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *filename = createCXString("");
2819e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (line)
2820e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *line = 0;
2821e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (column)
2822e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *column = 0;
2823e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis  }
2824e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis  else {
2825e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis	const SourceManager &SM =
2826e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis        *static_cast<const SourceManager*>(location.ptr_data[0]);
2827e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    PresumedLoc PreLoc = SM.getPresumedLoc(Loc);
2828e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis
2829e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (filename)
2830e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *filename = createCXString(PreLoc.getFilename());
2831e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (line)
2832e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *line = PreLoc.getLine();
2833e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis    if (column)
2834e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis      *column = PreLoc.getColumn();
2835e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis  }
2836e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis}
2837e6be34d8f77312edf9ed38034e52cb4d22c8e1c1Argyrios Kyrtzidis
283820174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruthvoid clang_getInstantiationLocation(CXSourceLocation location,
283920174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                    CXFile *file,
284020174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                    unsigned *line,
284120174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                    unsigned *column,
284220174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                    unsigned *offset) {
284320174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth  // Redirect to new API.
284420174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth  clang_getExpansionLocation(location, file, line, column, offset);
2845e69517ce61638f12c9abe4605753a45275ac4e37Douglas Gregor}
2846e69517ce61638f12c9abe4605753a45275ac4e37Douglas Gregor
2847a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregorvoid clang_getSpellingLocation(CXSourceLocation location,
2848a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                               CXFile *file,
2849a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                               unsigned *line,
2850a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                               unsigned *column,
2851a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor                               unsigned *offset) {
2852a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  SourceLocation Loc = SourceLocation::getFromRawEncoding(location.int_data);
2853a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor
28545adc0515aaacb6c4d4f0c9626d86c1e5c177467cArgyrios Kyrtzidis  if (!location.ptr_data[0] || Loc.isInvalid())
28555adc0515aaacb6c4d4f0c9626d86c1e5c177467cArgyrios Kyrtzidis    return createNullLocation(file, line, column, offset);
2856a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor
2857a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  const SourceManager &SM =
2858a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    *static_cast<const SourceManager*>(location.ptr_data[0]);
2859a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  SourceLocation SpellLoc = Loc;
2860a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (SpellLoc.isMacroID()) {
2861a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    SourceLocation SimpleSpellingLoc = SM.getImmediateSpellingLoc(SpellLoc);
2862a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    if (SimpleSpellingLoc.isFileID() &&
2863a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor        SM.getFileEntryForID(SM.getDecomposedLoc(SimpleSpellingLoc).first))
2864a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor      SpellLoc = SimpleSpellingLoc;
2865a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    else
2866402785357ab053dd53f4fdd858b9630a5e0f8badChandler Carruth      SpellLoc = SM.getExpansionLoc(SpellLoc);
2867a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  }
2868a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor
2869a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  std::pair<FileID, unsigned> LocInfo = SM.getDecomposedLoc(SpellLoc);
2870a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  FileID FID = LocInfo.first;
2871a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  unsigned FileOffset = LocInfo.second;
2872a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor
28735adc0515aaacb6c4d4f0c9626d86c1e5c177467cArgyrios Kyrtzidis  if (FID.isInvalid())
28745adc0515aaacb6c4d4f0c9626d86c1e5c177467cArgyrios Kyrtzidis    return createNullLocation(file, line, column, offset);
28755adc0515aaacb6c4d4f0c9626d86c1e5c177467cArgyrios Kyrtzidis
2876a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (file)
2877a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    *file = (void *)SM.getFileEntryForID(FID);
2878a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (line)
2879a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    *line = SM.getLineNumber(FID, FileOffset);
2880a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (column)
2881a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    *column = SM.getColumnNumber(FID, FileOffset);
2882a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor  if (offset)
2883a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor    *offset = FileOffset;
2884a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor}
2885a9b06d4c246d6c301e3dd1844f5dba669ed9c631Douglas Gregor
28861db19dea8d221f27be46332d668d1e2decb7f1abDouglas GregorCXSourceLocation clang_getRangeStart(CXSourceRange range) {
2887f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  CXSourceLocation Result = { { range.ptr_data[0], range.ptr_data[1] },
28885352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor                              range.begin_int_data };
28891db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  return Result;
28901db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor}
28911db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
28921db19dea8d221f27be46332d668d1e2decb7f1abDouglas GregorCXSourceLocation clang_getRangeEnd(CXSourceRange range) {
2893bb4a61a121ba1ee91eb5725881d98249704bb0aaDaniel Dunbar  CXSourceLocation Result = { { range.ptr_data[0], range.ptr_data[1] },
28945352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor                              range.end_int_data };
28951db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor  return Result;
28961db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor}
28971db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor
2898b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor} // end: extern "C"
2899b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor
29001db19dea8d221f27be46332d668d1e2decb7f1abDouglas Gregor//===----------------------------------------------------------------------===//
2901fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXFile Operations.
2902fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2903fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2904fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
290574844072411bae91d5dbb89955d200cbe1e0a1c8Ted KremenekCXString clang_getFileName(CXFile SFile) {
290698258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
2907a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return createCXString((const char*)NULL);
2908f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
290988145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
291074844072411bae91d5dbb89955d200cbe1e0a1c8Ted Kremenek  return createCXString(FEnt->getName());
291188145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
291288145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff
291388145034694ed5267fa6fa5febc54fadc02bd479Steve Narofftime_t clang_getFileTime(CXFile SFile) {
291498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (!SFile)
291598258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor    return 0;
2916f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
291788145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  FileEntry *FEnt = static_cast<FileEntry *>(SFile);
291888145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff  return FEnt->getModificationTime();
2919ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff}
2920f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2921b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXFile clang_getFile(CXTranslationUnit tu, const char *file_name) {
2922b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!tu)
2923b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor    return 0;
2924f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2925a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2926f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2927b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  FileManager &FMgr = CXXUnit->getFileManager();
292839b49bcaaddb1049234fca9500c0ac02c088e23dChris Lattner  return const_cast<FileEntry *>(FMgr.getFile(file_name));
2929b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor}
2930f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2931dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregorunsigned clang_isFileMultipleIncludeGuarded(CXTranslationUnit tu, CXFile file) {
2932dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  if (!tu || !file)
2933dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor    return 0;
2934dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2935dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  ASTUnit *CXXUnit = static_cast<ASTUnit *>(tu->TUData);
2936dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  FileEntry *FEnt = static_cast<FileEntry *>(file);
2937dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor  return CXXUnit->getPreprocessor().getHeaderSearchInfo()
2938dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor                                          .isFileMultipleIncludeGuarded(FEnt);
2939dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor}
2940dd3e5549e3c11e217078938aacf72f042eea5343Douglas Gregor
2941fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
2942fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2943fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2944fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek// CXCursor Operations.
2945fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
2946fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
2947fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekstatic Decl *getDeclFromExpr(Stmt *E) {
2948c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2949db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return getDeclFromExpr(CE->getSubExpr());
2950db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2951fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (DeclRefExpr *RefExpr = dyn_cast<DeclRefExpr>(E))
2952fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RefExpr->getDecl();
295338f28c1189142429384996409fffbc57f67b5c60Douglas Gregor  if (BlockDeclRefExpr *RefExpr = dyn_cast<BlockDeclRefExpr>(E))
295438f28c1189142429384996409fffbc57f67b5c60Douglas Gregor    return RefExpr->getDecl();
2955fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (MemberExpr *ME = dyn_cast<MemberExpr>(E))
2956fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return ME->getMemberDecl();
2957fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCIvarRefExpr *RE = dyn_cast<ObjCIvarRefExpr>(E))
2958fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return RE->getDecl();
2959db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCPropertyRefExpr *PRE = dyn_cast<ObjCPropertyRefExpr>(E))
296012f78a6741a4cb3d904340f8d3d2714568b50e7aJohn McCall    return PRE->isExplicitProperty() ? PRE->getExplicitProperty() : 0;
2961db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2962fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (CallExpr *CE = dyn_cast<CallExpr>(E))
2963fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return getDeclFromExpr(CE->getCallee());
29645f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  if (CXXConstructExpr *CE = dyn_cast<CXXConstructExpr>(E))
296593798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    if (!CE->isElidable())
296693798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CE->getConstructor();
2967fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  if (ObjCMessageExpr *OME = dyn_cast<ObjCMessageExpr>(E))
2968fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek    return OME->getMethodDecl();
2969f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
2970db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor  if (ObjCProtocolExpr *PE = dyn_cast<ObjCProtocolExpr>(E))
2971db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor    return PE->getProtocol();
2972c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor  if (SubstNonTypeTemplateParmPackExpr *NTTP
2973c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor                              = dyn_cast<SubstNonTypeTemplateParmPackExpr>(E))
2974c7793c73ba8a343de3f2552d984851985a46f159Douglas Gregor    return NTTP->getParameterPack();
297594d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
297694d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    if (isa<NonTypeTemplateParmDecl>(SizeOfPack->getPack()) ||
297794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor        isa<ParmVarDecl>(SizeOfPack->getPack()))
297894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor      return SizeOfPack->getPack();
2979db1314e3ef76f38de07f1b3c7cdc1100a0678931Douglas Gregor
2980fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek  return 0;
2981fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek}
2982ee9405e807d7c447c0143c2bd865b759192e97b3Steve Naroff
2983c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbarstatic SourceLocation getLocationFromExpr(Expr *E) {
2984c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis  if (ImplicitCastExpr *CE = dyn_cast<ImplicitCastExpr>(E))
2985c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis    return getLocationFromExpr(CE->getSubExpr());
2986c2954616fbd11f5a6117236f58420029b773a639Argyrios Kyrtzidis
2987c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCMessageExpr *Msg = dyn_cast<ObjCMessageExpr>(E))
2988c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return /*FIXME:*/Msg->getLeftLoc();
2989c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E))
2990c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return DRE->getLocation();
299138f28c1189142429384996409fffbc57f67b5c60Douglas Gregor  if (BlockDeclRefExpr *RefExpr = dyn_cast<BlockDeclRefExpr>(E))
299238f28c1189142429384996409fffbc57f67b5c60Douglas Gregor    return RefExpr->getLocation();
2993c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (MemberExpr *Member = dyn_cast<MemberExpr>(E))
2994c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Member->getMemberLoc();
2995c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  if (ObjCIvarRefExpr *Ivar = dyn_cast<ObjCIvarRefExpr>(E))
2996c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar    return Ivar->getLocation();
299794d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor  if (SizeOfPackExpr *SizeOfPack = dyn_cast<SizeOfPackExpr>(E))
299894d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor    return SizeOfPack->getPackLoc();
299994d96291cd041adc5731a2294828a9c20e450b74Douglas Gregor
3000c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar  return E->getLocStart();
3001c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar}
3002c29f4c396019f51e09511e385d6c07c4ea2a1464Daniel Dunbar
3003fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenekextern "C" {
3004f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3005f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenekunsigned clang_visitChildren(CXCursor parent,
3006b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXCursorVisitor visitor,
3007b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor                             CXClientData client_data) {
3008a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CursorVisitor CursorVis(getCursorTU(parent), visitor, client_data,
300904a9eb3f0dfcc42b317057e236f73e9b196a2ff8Douglas Gregor                          false);
3010b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor  return CursorVis.VisitChildren(parent);
3011b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor}
3012b1373d03e4e38620647311bb87e56c0cf7cffeb7Douglas Gregor
30133387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#ifndef __has_feature
30143387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#define __has_feature(x) 0
30153387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
30163387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#if __has_feature(blocks)
30173387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef enum CXChildVisitResult
30183387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall     (^CXCursorVisitorBlock)(CXCursor cursor, CXCursor parent);
30193387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
30203387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
30213387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
30223387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
30233387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block(cursor, parent);
30243387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
30253387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#else
30263387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// If we are compiled with a compiler that doesn't have native blocks support,
30273387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall// define and call the block manually, so the
30283387c65a094a02b2a94c05111d035a97d3d5c794David Chisnalltypedef struct _CXChildVisitResult
30293387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall{
30303387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	void *isa;
30313387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int flags;
30323387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall	int reserved;
30339e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar	enum CXChildVisitResult(*invoke)(struct _CXChildVisitResult*, CXCursor,
30349e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                         CXCursor);
30353387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall} *CXCursorVisitorBlock;
30363387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
30373387c65a094a02b2a94c05111d035a97d3d5c794David Chisnallstatic enum CXChildVisitResult visitWithBlock(CXCursor cursor, CXCursor parent,
30383387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall    CXClientData client_data) {
30393387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  CXCursorVisitorBlock block = (CXCursorVisitorBlock)client_data;
30403387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return block->invoke(block, cursor, parent);
30413387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
30423387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall#endif
30433387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
30443387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
30459e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbarunsigned clang_visitChildrenWithBlock(CXCursor parent,
30469e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar                                      CXCursorVisitorBlock block) {
30473387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall  return clang_visitChildren(parent, visitWithBlock, block);
30483387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall}
30493387c65a094a02b2a94c05111d035a97d3d5c794David Chisnall
305078205d4bada39d95097e766af9eb30cdd0159461Douglas Gregorstatic CXString getDeclSpelling(Decl *D) {
305178205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  NamedDecl *ND = dyn_cast_or_null<NamedDecl>(D);
3052e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  if (!ND) {
30535f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
3054e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
3055e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return createCXString(Property->getIdentifier()->getName());
3056e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
3057ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
3058e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor  }
3059e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
306078205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCMethodDecl *OMD = dyn_cast<ObjCMethodDecl>(ND))
3061ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(OMD->getSelector().getAsString());
3062f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
306378205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor  if (ObjCCategoryImplDecl *CIMP = dyn_cast<ObjCCategoryImplDecl>(ND))
306478205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // No, this isn't the same as the code below. getIdentifier() is non-virtual
306578205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // and returns different names. NamedDecl returns the class name and
306678205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor    // ObjCCategoryImplDecl returns the category name.
3067ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(CIMP->getIdentifier()->getNameStart());
3068f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
30690a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  if (isa<UsingDirectiveDecl>(D))
30700a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("");
30710a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor
307250aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  llvm::SmallString<1024> S;
307350aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  llvm::raw_svector_ostream os(S);
307450aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  ND->printName(os);
307550aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek
307650aa6acd0b8d40c8956372a69e0a73f0802a5494Ted Kremenek  return createCXString(os.str());
307778205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor}
3078f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
30799ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXString clang_getCursorSpelling(CXCursor C) {
30807eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  if (clang_isTranslationUnit(C.kind))
3081a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return clang_getTranslationUnitSpelling(
3082a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                            static_cast<CXTranslationUnit>(C.data[2]));
30837eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
3084f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  if (clang_isReference(C.kind)) {
3085f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    switch (C.kind) {
3086acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCSuperClassRef: {
30872e331b938b38057e333fab0ba841130ea8467794Douglas Gregor      ObjCInterfaceDecl *Super = getCursorObjCSuperClassRef(C).first;
3088ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Super->getIdentifier()->getNameStart());
3089acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
3090acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCClassRef: {
30911adb082a709f7b588f03672999294e061234b2cfDouglas Gregor      ObjCInterfaceDecl *Class = getCursorObjCClassRef(C).first;
3092ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(Class->getIdentifier()->getNameStart());
3093acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
3094acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    case CXCursor_ObjCProtocolRef: {
309578db0cdd49ec24034a5b2a4210fcda03a0919a81Douglas Gregor      ObjCProtocolDecl *OID = getCursorObjCProtocolRef(C).first;
3096f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      assert(OID && "getCursorSpelling(): Missing protocol decl");
3097ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(OID->getIdentifier()->getNameStart());
3098acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    }
30993064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
31003064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = getCursorCXXBaseSpecifier(C);
31013064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return createCXString(B->getType().getAsString());
31023064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
31037d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    case CXCursor_TypeRef: {
31047d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      TypeDecl *Type = getCursorTypeRef(C).first;
31057d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      assert(Type && "Missing type decl");
31067d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3107ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString(getCursorContext(C).getTypeDeclType(Type).
3108ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                              getAsString());
31097d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
31100b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
31110b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      TemplateDecl *Template = getCursorTemplateRef(C).first;
31126931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(Template && "Missing template decl");
31130b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
31140b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString(Template->getNameAsString());
31150b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
31166931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
31176931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
31186931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      NamedDecl *NS = getCursorNamespaceRef(C).first;
31196931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      assert(NS && "Missing namespace decl");
31206931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
31216931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return createCXString(NS->getNameAsString());
31226931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
31237d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3124a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
3125a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      FieldDecl *Field = getCursorMemberRef(C).first;
3126a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      assert(Field && "Missing member decl");
3127a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3128a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return createCXString(Field->getNameAsString());
3129a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
3130a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
313136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
313236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      LabelStmt *Label = getCursorLabelRef(C).first;
313336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      assert(Label && "Missing label");
313436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
3135ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
313636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
313736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
31381f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef: {
31391f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
31401f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Decl *D = Storage.dyn_cast<Decl *>()) {
31411f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        if (NamedDecl *ND = dyn_cast<NamedDecl>(D))
31421f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor          return createCXString(ND->getNameAsString());
31431f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
31441f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      }
31451f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
31461f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString(E->getName().getAsString());
31471f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      OverloadedTemplateStorage *Ovl
31481f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        = Storage.get<OverloadedTemplateStorage*>();
31491f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      if (Ovl->size() == 0)
31501f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor        return createCXString("");
31511f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return createCXString((*Ovl->begin())->getNameAsString());
31521f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    }
31531f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3154acca725aeff400274fab552fc83d87af8a11995dDaniel Dunbar    default:
3155ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek      return createCXString("<not implemented>");
3156f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff    }
3157f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  }
315897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
315997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
316097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    Decl *D = getDeclFromExpr(getCursorExpr(C));
316197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    if (D)
316278205d4bada39d95097e766af9eb30cdd0159461Douglas Gregor      return getDeclSpelling(D);
3163ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
316497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
316597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
316636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
316736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
316836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (LabelStmt *Label = dyn_cast_or_null<LabelStmt>(S))
3169ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner      return createCXString(Label->getName());
317036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
317136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("");
317236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
317336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
31749b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
31759e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    return createCXString(getCursorMacroExpansion(C)->getName()
31764ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor                                                           ->getNameStart());
31774ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor
3178572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition)
3179572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString(getCursorMacroDefinition(C)->getName()
3180572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor                                                           ->getNameStart());
3181572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3182ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective)
3183ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString(getCursorInclusionDirective(C)->getFileName());
3184ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
318560cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor  if (clang_isDeclaration(C.kind))
318660cbfacd947590f83257a4191566dda92fbbde69Douglas Gregor    return getDeclSpelling(getCursorDecl(C));
3187e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3188ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString("");
3189f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3190f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
3191358559d8d7b458c5f64941842383a16e61f0828dDouglas GregorCXString clang_getCursorDisplayName(CXCursor C) {
3192358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!clang_isDeclaration(C.kind))
3193358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return clang_getCursorSpelling(C);
3194358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3195358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  Decl *D = getCursorDecl(C);
3196358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (!D)
3197358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString("");
3198358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3199358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  PrintingPolicy &Policy = getCursorContext(C).PrintingPolicy;
3200358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast<FunctionTemplateDecl>(D))
3201358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    D = FunTmpl->getTemplatedDecl();
3202358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3203358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (FunctionDecl *Function = dyn_cast<FunctionDecl>(D)) {
3204358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3205358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3206358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << Function->getNameAsString();
3207358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->getPrimaryTemplate())
3208358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "<>";
3209358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "(";
3210358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Function->getNumParams(); I != N; ++I) {
3211358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3212358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3213358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << Function->getParamDecl(I)->getType().getAsString(Policy);
3214358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3215358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3216358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (Function->isVariadic()) {
3217358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Function->getNumParams())
3218358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3219358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      OS << "...";
3220358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3221358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ")";
3222358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3223358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3224358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3225358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateDecl *ClassTemplate = dyn_cast<ClassTemplateDecl>(D)) {
3226358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3227358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3228358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ClassTemplate->getNameAsString();
3229358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << "<";
3230358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    TemplateParameterList *Params = ClassTemplate->getTemplateParameters();
3231358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    for (unsigned I = 0, N = Params->size(); I != N; ++I) {
3232358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (I)
3233358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << ", ";
3234358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3235358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      NamedDecl *Param = Params->getParam(I);
3236358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (Param->getIdentifier()) {
3237358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << Param->getIdentifier()->getName();
3238358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        continue;
3239358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      }
3240358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3241358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // There is no parameter name, which makes this tricky. Try to come up
3242358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      // with something useful that isn't too long.
3243358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      if (TemplateTypeParmDecl *TTP = dyn_cast<TemplateTypeParmDecl>(Param))
3244358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << (TTP->wasDeclaredWithTypename()? "typename" : "class");
3245358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else if (NonTypeTemplateParmDecl *NTTP
3246358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                    = dyn_cast<NonTypeTemplateParmDecl>(Param))
3247358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << NTTP->getType().getAsString(Policy);
3248358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      else
3249358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor        OS << "template<...> class";
3250358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    }
3251358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3252358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ">";
3253358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3254358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3255358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3256358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  if (ClassTemplateSpecializationDecl *ClassSpec
3257358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                              = dyn_cast<ClassTemplateSpecializationDecl>(D)) {
3258358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    // If the type was explicitly written, use that.
3259358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    if (TypeSourceInfo *TSInfo = ClassSpec->getTypeAsWritten())
3260358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor      return createCXString(TSInfo->getType().getAsString(Policy));
3261358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3262358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::SmallString<64> Str;
3263358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    llvm::raw_svector_ostream OS(Str);
3264358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << ClassSpec->getNameAsString();
3265358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    OS << TemplateSpecializationType::PrintTemplateArgumentList(
3266910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().data(),
3267910f8008fea79120489a53593fe971b0b8a4a740Douglas Gregor                                      ClassSpec->getTemplateArgs().size(),
3268358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor                                                                Policy);
3269358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor    return createCXString(OS.str());
3270358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  }
3271358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3272358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor  return clang_getCursorSpelling(C);
3273358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor}
3274358559d8d7b458c5f64941842383a16e61f0828dDouglas Gregor
3275e68fff6fc083c6270d835216a3de0b82c6ef0310Ted KremenekCXString clang_getCursorKindSpelling(enum CXCursorKind Kind) {
327689922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  switch (Kind) {
3277e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FunctionDecl:
3278e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FunctionDecl");
3279e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypedefDecl:
3280e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypedefDecl");
3281e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumDecl:
3282e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumDecl");
3283e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_EnumConstantDecl:
3284e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("EnumConstantDecl");
3285e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_StructDecl:
3286e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("StructDecl");
3287e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnionDecl:
3288e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnionDecl");
3289e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ClassDecl:
3290e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ClassDecl");
3291e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_FieldDecl:
3292e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("FieldDecl");
3293e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_VarDecl:
3294e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("VarDecl");
3295e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ParmDecl:
3296e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ParmDecl");
3297e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInterfaceDecl:
3298e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInterfaceDecl");
3299e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryDecl:
3300e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryDecl");
3301e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolDecl:
3302e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolDecl");
3303e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCPropertyDecl:
3304e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCPropertyDecl");
3305e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCIvarDecl:
3306e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCIvarDecl");
3307e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCInstanceMethodDecl:
3308e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCInstanceMethodDecl");
3309e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassMethodDecl:
3310e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassMethodDecl");
3311e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCImplementationDecl:
3312e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCImplementationDecl");
3313e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCCategoryImplDecl:
3314e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCCategoryImplDecl");
33158bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek  case CXCursor_CXXMethod:
33168bd5a69999cfd06b6b5a58fdd04e4f802b2df5a4Ted Kremenek      return createCXString("CXXMethod");
3317e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedDecl:
3318e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedDecl");
3319e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCSuperClassRef:
3320e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCSuperClassRef");
3321e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCProtocolRef:
3322e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCProtocolRef");
3323e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCClassRef:
3324e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCClassRef");
3325e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TypeRef:
3326e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TypeRef");
33270b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor  case CXCursor_TemplateRef:
33280b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return createCXString("TemplateRef");
33296931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceRef:
33306931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceRef");
3331a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor  case CXCursor_MemberRef:
3332a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    return createCXString("MemberRef");
333336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelRef:
333436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return createCXString("LabelRef");
33351f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case CXCursor_OverloadedDeclRef:
33361f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return createCXString("OverloadedDeclRef");
3337e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedExpr:
3338e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedExpr");
33391ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek  case CXCursor_BlockExpr:
33401ee6cad59f017601ea54fbb4f62a6e8d69897e3eTed Kremenek      return createCXString("BlockExpr");
3341e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_DeclRefExpr:
3342e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("DeclRefExpr");
3343e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_MemberRefExpr:
3344e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("MemberRefExpr");
3345e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_CallExpr:
3346e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("CallExpr");
3347e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_ObjCMessageExpr:
3348e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("ObjCMessageExpr");
3349e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_UnexposedStmt:
3350e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("UnexposedStmt");
335136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  case CXCursor_LabelStmt:
335236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return createCXString("LabelStmt");
3353e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_InvalidFile:
3354e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("InvalidFile");
3355292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek  case CXCursor_InvalidCode:
3356292db6401f040795db3ea4e00fc02622d6c3ba1dTed Kremenek    return createCXString("InvalidCode");
3357e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NoDeclFound:
3358e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NoDeclFound");
3359e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_NotImplemented:
3360e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("NotImplemented");
3361e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  case CXCursor_TranslationUnit:
3362e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek      return createCXString("TranslationUnit");
3363e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_UnexposedAttr:
3364e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("UnexposedAttr");
3365e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek  case CXCursor_IBActionAttr:
3366e77f443dbca8cdc23e5aa94a2653367e4a7cbe47Ted Kremenek      return createCXString("attribute(ibaction)");
33679f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_IBOutletAttr:
33689f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor     return createCXString("attribute(iboutlet)");
3369857e918a8a40deb128840308a318bf623d68295fTed Kremenek  case CXCursor_IBOutletCollectionAttr:
3370857e918a8a40deb128840308a318bf623d68295fTed Kremenek      return createCXString("attribute(iboutletcollection)");
33716639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXFinalAttr:
33726639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(final)");
33736639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  case CXCursor_CXXOverrideAttr:
33746639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      return createCXString("attribute(override)");
33759f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  case CXCursor_PreprocessingDirective:
33769f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return createCXString("preprocessing directive");
3377572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  case CXCursor_MacroDefinition:
3378572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return createCXString("macro definition");
33799b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  case CXCursor_MacroExpansion:
33809b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    return createCXString("macro expansion");
3381ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  case CXCursor_InclusionDirective:
3382ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return createCXString("inclusion directive");
33838f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek  case CXCursor_Namespace:
33848f06e0e9fec3ca501e5fb129f413adbfc88e82f8Ted Kremenek    return createCXString("Namespace");
3385a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek  case CXCursor_LinkageSpec:
3386a0536d8dd900bb48ea886bd68d777b03b061c068Ted Kremenek    return createCXString("LinkageSpec");
33873064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek  case CXCursor_CXXBaseSpecifier:
33883064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    return createCXString("C++ base class specifier");
338901829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Constructor:
339001829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConstructor");
339101829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_Destructor:
339201829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXDestructor");
339301829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor  case CXCursor_ConversionFunction:
339401829d3afafdfd355cbe93537bc408aeeed964c6Douglas Gregor    return createCXString("CXXConversion");
3395fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTypeParameter:
3396fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTypeParameter");
3397fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_NonTypeTemplateParameter:
3398fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("NonTypeTemplateParameter");
3399fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_TemplateTemplateParameter:
3400fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("TemplateTemplateParameter");
3401fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor  case CXCursor_FunctionTemplate:
3402fe72e9ceeae6cc8669cd8bb722425300190638eaDouglas Gregor    return createCXString("FunctionTemplate");
340339d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor  case CXCursor_ClassTemplate:
340439d6f07b056c31e1e6b5946165ed4b23e7887f22Douglas Gregor    return createCXString("ClassTemplate");
340574dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor  case CXCursor_ClassTemplatePartialSpecialization:
340674dbe640021d96a8dbb85c592471c04449ade81cDouglas Gregor    return createCXString("ClassTemplatePartialSpecialization");
34076931900f43cea558c6974075256c07728dbfecc6Douglas Gregor  case CXCursor_NamespaceAlias:
34086931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    return createCXString("NamespaceAlias");
34090a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor  case CXCursor_UsingDirective:
34100a35bceb7768fc0be62cb644a4e31d8bfd9fb44aDouglas Gregor    return createCXString("UsingDirective");
34117e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor  case CXCursor_UsingDeclaration:
34127e24256c95afb64b4d5abf201a0f9f0527cb4cf3Douglas Gregor    return createCXString("UsingDeclaration");
3413162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case CXCursor_TypeAliasDecl:
3414352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("TypeAliasDecl");
3415352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCSynthesizeDecl:
3416352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCSynthesizeDecl");
3417352697a87bca664356f21a838b162084013625eaDouglas Gregor  case CXCursor_ObjCDynamicDecl:
3418352697a87bca664356f21a838b162084013625eaDouglas Gregor    return createCXString("ObjCDynamicDecl");
341989922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  }
3420e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3421deb06bd3566e18f677e76bc435d478b033fe328bTed Kremenek  llvm_unreachable("Unhandled CXCursorKind");
3422a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  return createCXString((const char*) 0);
3423600866cc7d6d9ec2e27d4b6d6ec461f6463b5ab6Steve Naroff}
342489922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff
3425064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidisstruct GetCursorData {
3426064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  SourceLocation TokenBeginLoc;
34274b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  bool PointsAtMacroArgExpansion;
3428064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor &BestCursor;
3429064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
34304b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  GetCursorData(SourceManager &SM,
34314b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                SourceLocation tokenBegin, CXCursor &outputCursor)
34324b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    : TokenBeginLoc(tokenBegin), BestCursor(outputCursor) {
34334b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    PointsAtMacroArgExpansion = SM.isMacroArgExpansion(tokenBegin);
34344b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  }
3435064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis};
3436064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
34374b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidisstatic enum CXChildVisitResult GetCursorVisitor(CXCursor cursor,
34384b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXCursor parent,
34394b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis                                                CXClientData client_data) {
3440064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  GetCursorData *Data = static_cast<GetCursorData *>(client_data);
3441064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  CXCursor *BestCursor = &Data->BestCursor;
34424b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis
34434b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // If we point inside a macro argument we should provide info of what the
34444b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // token is so use the actual cursor, don't replace it with a macro expansion
34454b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  // cursor.
34464b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis  if (cursor.kind == CXCursor_MacroExpansion && Data->PointsAtMacroArgExpansion)
34474b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    return CXChildVisit_Recurse;
3448064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
3449064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  if (clang_isExpression(cursor.kind) &&
3450064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis      clang_isDeclaration(BestCursor->kind)) {
3451064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    Decl *D = getCursorDecl(*BestCursor);
3452064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
3453064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // Avoid having the cursor of an expression replace the declaration cursor
3454064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // when the expression source range overlaps the declaration range.
3455064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // This can happen for C++ constructor expressions whose range generally
3456064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    // include the variable declaration, e.g.:
3457064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    //  MyCXXClass foo; // Make sure pointing at 'foo' returns a VarDecl cursor.
3458064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    if (D->getLocation().isValid() && Data->TokenBeginLoc.isValid() &&
3459064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis        D->getLocation() == Data->TokenBeginLoc)
3460064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis      return CXChildVisit_Break;
3461064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis  }
3462064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis
346393798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // If our current best cursor is the construction of a temporary object,
346493798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // don't replace that cursor with a type reference, because we want
346593798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  // clang_getCursor() to point at the constructor.
346693798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor  if (clang_isExpression(BestCursor->kind) &&
346793798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor      isa<CXXTemporaryObjectExpr>(getCursorExpr(*BestCursor)) &&
346893798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor      cursor.kind == CXCursor_TypeRef)
346993798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor    return CXChildVisit_Recurse;
347093798e25d68e2d146cff9bd0355d4b7b1ca765f1Douglas Gregor
347185fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor  // Don't override a preprocessing cursor with another preprocessing
347285fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor  // cursor; we want the outermost preprocessing cursor.
347385fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor  if (clang_isPreprocessing(cursor.kind) &&
347485fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor      clang_isPreprocessing(BestCursor->kind))
347585fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor    return CXChildVisit_Recurse;
347685fe1560b061b5f93a52dbd07cddd6e808854710Douglas Gregor
347733e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  *BestCursor = cursor;
347833e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  return CXChildVisit_Recurse;
347933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor}
3480e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3481b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas GregorCXCursor clang_getCursor(CXTranslationUnit TU, CXSourceLocation Loc) {
3482b979034b100be14de2223f2b8f6cc7a3275cbe4fDouglas Gregor  if (!TU)
3483f462989fe8d6f59ab2d7d0fe2b4b96292ce706eaTed Kremenek    return clang_getNullCursor();
3484e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek
3485a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
3486bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
3487bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor
3488a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  // Translate the given source location to make it point at the beginning of
3489a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  // the token under the cursor.
3490a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek  SourceLocation SLoc = cxloc::translateSourceLocation(Loc);
3491a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek
3492a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek  // Guard against an invalid SourceLocation, or we may assert in one
3493a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek  // of the following calls.
3494a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek  if (SLoc.isInvalid())
3495a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek    return clang_getNullCursor();
3496a629ea42f6bc095190db2f3932b60a0be14f3d34Ted Kremenek
349740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  bool Logging = getenv("LIBCLANG_LOGGING");
3498a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  SLoc = Lexer::GetBeginningOfToken(SLoc, CXXUnit->getSourceManager(),
3499a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor                                    CXXUnit->getASTContext().getLangOptions());
3500a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
350133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  CXCursor Result = MakeCXCursorInvalid(CXCursor_NoDeclFound);
350233e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (SLoc.isValid()) {
350333e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    // FIXME: Would be great to have a "hint" cursor, then walk from that
350433e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    // hint cursor upward until we find a cursor whose source range encloses
350533e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    // the region of interest, rather than starting from the translation unit.
35064b43b305342ae2e49d473d0fa6152e5d0c343765Argyrios Kyrtzidis    GetCursorData ResultData(CXXUnit->getSourceManager(), SLoc, Result);
3507a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    CXCursor Parent = clang_getTranslationUnitCursor(TU);
3508064c44b8a17d4c426b87ad1c58de81d76a13fca3Argyrios Kyrtzidis    CursorVisitor CursorVis(TU, GetCursorVisitor, &ResultData,
350908e0bc16b3312c27e87d33be7dcf3d4fe5bdd2e2Douglas Gregor                            /*VisitPreprocessorLast=*/true,
351008e0bc16b3312c27e87d33be7dcf3d4fe5bdd2e2Douglas Gregor                            SourceLocation(SLoc));
351133e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor    CursorVis.VisitChildren(Parent);
351277128ddd3077fc045751a55bb3226802b15d5510Steve Naroff  }
351340749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
351440749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  if (Logging) {
351540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile SearchFile;
351640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned SearchLine, SearchColumn;
351740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXFile ResultFile;
351840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    unsigned ResultLine, ResultColumn;
35196653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    CXString SearchFileName, ResultFileName, KindSpelling, USR;
35206653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    const char *IsDef = clang_isCursorDefinition(Result)? " (Definition)" : "";
352140749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    CXSourceLocation ResultLoc = clang_getCursorLocation(Result);
352240749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
352320174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(Loc, &SearchFile, &SearchLine, &SearchColumn, 0);
352420174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth    clang_getExpansionLocation(ResultLoc, &ResultFile, &ResultLine,
352520174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                               &ResultColumn, 0);
352640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    SearchFileName = clang_getFileName(SearchFile);
352740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    ResultFileName = clang_getFileName(ResultFile);
352840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    KindSpelling = clang_getCursorKindSpelling(Result.kind);
35296653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    USR = clang_getCursorUSR(Result);
35306653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    fprintf(stderr, "clang_getCursor(%s:%d:%d) = %s(%s:%d:%d):%s%s\n",
353140749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(SearchFileName), SearchLine, SearchColumn,
353240749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor            clang_getCString(KindSpelling),
35336653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(ResultFileName), ResultLine, ResultColumn,
35346653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor            clang_getCString(USR), IsDef);
353540749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(SearchFileName);
353640749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(ResultFileName);
353740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor    clang_disposeString(KindSpelling);
35386653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    clang_disposeString(USR);
35390aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor
35400aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    CXCursor Definition = clang_getCursorDefinition(Result);
35410aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    if (!clang_equalCursors(Definition, clang_getNullCursor())) {
35420aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXSourceLocation DefinitionLoc = clang_getCursorLocation(Definition);
35430aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionKindSpelling
35440aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor                                = clang_getCursorKindSpelling(Definition.kind);
35450aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXFile DefinitionFile;
35460aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      unsigned DefinitionLine, DefinitionColumn;
354720174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth      clang_getExpansionLocation(DefinitionLoc, &DefinitionFile,
354820174221af145554b76a0b0f5e4eb3ac70d05945Chandler Carruth                                 &DefinitionLine, &DefinitionColumn, 0);
35490aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      CXString DefinitionFileName = clang_getFileName(DefinitionFile);
35500aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      fprintf(stderr, "  -> %s(%s:%d:%d)\n",
35510aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionKindSpelling),
35520aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              clang_getCString(DefinitionFileName),
35530aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor              DefinitionLine, DefinitionColumn);
35540aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionFileName);
35550aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor      clang_disposeString(DefinitionKindSpelling);
35560aefbd8d6a82fe8f70ec92871ea42fa5240a960fDouglas Gregor    }
355740749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor  }
355840749ee585abc84fbb3c8fdbd8aaac062f153062Douglas Gregor
3559e68fff6fc083c6270d835216a3de0b82c6ef0310Ted Kremenek  return Result;
356077128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
356177128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
3562738855554394a6afcf39cc8345fd22c3756b8dd0Ted KremenekCXCursor clang_getNullCursor(void) {
35635bfb8c128c2ac8eb4032afc180cdc400a0f953caDouglas Gregor  return MakeCXCursorInvalid(CXCursor_InvalidFile);
3564738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
3565738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek
3566738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenekunsigned clang_equalCursors(CXCursor X, CXCursor Y) {
3567283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  return X == Y;
3568738855554394a6afcf39cc8345fd22c3756b8dd0Ted Kremenek}
35690d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbar
35709ce5584553054d0cb934940586aca0186e87fa57Douglas Gregorunsigned clang_hashCursor(CXCursor C) {
35719ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  unsigned Index = 0;
35729ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  if (clang_isExpression(C.kind) || clang_isStatement(C.kind))
35739ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor    Index = 1;
35749ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35759ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor  return llvm::DenseMapInfo<std::pair<unsigned, void*> >::getHashValue(
35769ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor                                        std::make_pair(C.kind, C.data[Index]));
35779ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor}
35789ce5584553054d0cb934940586aca0186e87fa57Douglas Gregor
35799ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isInvalid(enum CXCursorKind K) {
358077128ddd3077fc045751a55bb3226802b15d5510Steve Naroff  return K >= CXCursor_FirstInvalid && K <= CXCursor_LastInvalid;
358177128ddd3077fc045751a55bb3226802b15d5510Steve Naroff}
358277128ddd3077fc045751a55bb3226802b15d5510Steve Naroff
35839ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isDeclaration(enum CXCursorKind K) {
358489922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff  return K >= CXCursor_FirstDecl && K <= CXCursor_LastDecl;
358589922f86f4e7da383af2a62ef04ad8b93b941220Steve Naroff}
35862d4d629d8a0de5112c7ae9d05c03ddbf6dcd956aSteve Naroff
35879ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbarunsigned clang_isReference(enum CXCursorKind K) {
3588f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff  return K >= CXCursor_FirstRef && K <= CXCursor_LastRef;
3589f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff}
3590f334b4e3eda5a39f041fe13f805dbb53535daa2fSteve Naroff
359197b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isExpression(enum CXCursorKind K) {
359297b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstExpr && K <= CXCursor_LastExpr;
359397b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
359497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
359597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregorunsigned clang_isStatement(enum CXCursorKind K) {
359697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  return K >= CXCursor_FirstStmt && K <= CXCursor_LastStmt;
359797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor}
359897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
35998be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregorunsigned clang_isAttribute(enum CXCursorKind K) {
36008be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor    return K >= CXCursor_FirstAttr && K <= CXCursor_LastAttr;
36018be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor}
36028be80e1e6effd5a333bc70e7f030dc9397d0554eDouglas Gregor
36037eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregorunsigned clang_isTranslationUnit(enum CXCursorKind K) {
36047eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor  return K == CXCursor_TranslationUnit;
36057eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor}
36067eaa8ae8692c5cd3eed8cb334fe5346470522091Douglas Gregor
36079f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregorunsigned clang_isPreprocessing(enum CXCursorKind K) {
36089f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  return K >= CXCursor_FirstPreprocessing && K <= CXCursor_LastPreprocessing;
36099f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor}
36109f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor
3611ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenekunsigned clang_isUnexposed(enum CXCursorKind K) {
3612ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  switch (K) {
3613ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedDecl:
3614ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedExpr:
3615ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedStmt:
3616ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    case CXCursor_UnexposedAttr:
3617ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return true;
3618ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek    default:
3619ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek      return false;
3620ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek  }
3621ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek}
3622ad6eff611a4391f89fd6c458db16993f76e7f5d0Ted Kremenek
36239ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel DunbarCXCursorKind clang_getCursorKind(CXCursor C) {
36249efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff  return C.kind;
36259efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff}
36269efa767be8e9f2dae509d3a0be93ade01bfa1560Steve Naroff
362798258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas GregorCXSourceLocation clang_getCursorLocation(CXCursor C) {
362898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  if (clang_isReference(C.kind)) {
3629f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    switch (C.kind) {
3630f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCSuperClassRef: {
3631f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3632f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCSuperClassRef(C);
3633a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3634f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3635f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3636f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
3637f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCProtocolDecl *, SourceLocation> P
3638f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCProtocolRef(C);
3639a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3640f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
3641f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor
3642f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCClassRef: {
3643f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      std::pair<ObjCInterfaceDecl *, SourceLocation> P
3644f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor        = getCursorObjCClassRef(C);
3645a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3646f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
36477d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3648f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef: {
36497d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor      std::pair<TypeDecl *, SourceLocation> P = getCursorTypeRef(C);
3650a297de20bde673b1a2ad67fa5eafd37fea4227fcTed Kremenek      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36517d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor    }
36520b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
36530b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef: {
36540b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      std::pair<TemplateDecl *, SourceLocation> P = getCursorTemplateRef(C);
36550b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36560b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    }
36570b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
36586931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef: {
36596931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      std::pair<NamedDecl *, SourceLocation> P = getCursorNamespaceRef(C);
36606931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
36616931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    }
36626931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
3663a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef: {
3664a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      std::pair<FieldDecl *, SourceLocation> P = getCursorMemberRef(C);
3665a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return cxloc::translateSourceLocation(P.first->getASTContext(), P.second);
3666a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    }
3667a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
36683064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
36691b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      CXXBaseSpecifier *BaseSpec = getCursorCXXBaseSpecifier(C);
36701b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (!BaseSpec)
36711b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return clang_getNullLocation();
36721b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
36731b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      if (TypeSourceInfo *TSInfo = BaseSpec->getTypeSourceInfo())
36741b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor        return cxloc::translateSourceLocation(getCursorContext(C),
36751b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor                                            TSInfo->getTypeLoc().getBeginLoc());
36761b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor
36771b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
36781b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor                                        BaseSpec->getSourceRange().getBegin());
36793064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
3680f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
368136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef: {
368236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      std::pair<LabelStmt *, SourceLocation> P = getCursorLabelRef(C);
368336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C), P.second);
368436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    }
368536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
36861f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
36871f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return cxloc::translateSourceLocation(getCursorContext(C),
36881f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                                          getCursorOverloadedDeclRef(C).second);
36891f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3690f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    default:
3691f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3692f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor      llvm_unreachable("Missed a reference kind");
3693f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    }
369498258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor  }
369597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
369697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3697f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    return cxloc::translateSourceLocation(getCursorContext(C),
369897b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor                                   getLocationFromExpr(getCursorExpr(C)));
369997b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
370036897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind))
370136897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C),
370236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor                                          getCursorStmt(C)->getLocStart());
370336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
37049f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  if (C.kind == CXCursor_PreprocessingDirective) {
37059f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    SourceLocation L = cxcursor::getCursorPreprocessingDirective(C).getBegin();
37069f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
37079f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  }
37084807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
37099b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
37104ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor    SourceLocation L
37119e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth      = cxcursor::getCursorMacroExpansion(C)->getSourceRange().getBegin();
37124807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
37134807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor  }
3714572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3715572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  if (C.kind == CXCursor_MacroDefinition) {
3716572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    SourceLocation L = cxcursor::getCursorMacroDefinition(C)->getLocation();
3717572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3718572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor  }
3719ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3720ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective) {
3721ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    SourceLocation L
3722ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor      = cxcursor::getCursorInclusionDirective(C)->getSourceRange().getBegin();
3723ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return cxloc::translateSourceLocation(getCursorContext(C), L);
3724ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  }
3725ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
37269a700d277c38d9afaa7cb3fe93a714bfe9b62eecTed Kremenek  if (C.kind < CXCursor_FirstDecl || C.kind > CXCursor_LastDecl)
37275352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullLocation();
372898258afae66bab39b0c57a3efb6b20d4fbb5746cDouglas Gregor
3729f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  Decl *D = getCursorDecl(C);
3730f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  SourceLocation Loc = D->getLocation();
3731f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor  if (ObjCInterfaceDecl *Class = dyn_cast<ObjCInterfaceDecl>(D))
3732f46034af49435a4d1a0085a4738343122aeb6521Douglas Gregor    Loc = Class->getClassLoc();
3733007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // FIXME: Multiple variables declared in a single declaration
3734007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // currently lack the information needed to correctly determine their
3735007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // ranges when accounting for the type-specifier.  We use context
3736007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3737007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  // and if so, whether it is the first decl.
3738007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3739007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (!cxcursor::isFirstInDeclGroup(C))
3740007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      Loc = VD->getLocation();
3741007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
3742007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek
37432ca54feee89d7277fb967e3247a64f40ef155a82Douglas Gregor  return cxloc::translateSourceLocation(getCursorContext(C), Loc);
374488145034694ed5267fa6fa5febc54fadc02bd479Steve Naroff}
3745a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor
3746a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor} // end extern "C"
3747a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3748a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorstatic SourceRange getRawCursorExtent(CXCursor C) {
3749a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  if (clang_isReference(C.kind)) {
3750a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    switch (C.kind) {
3751a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCSuperClassRef:
3752a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return  getCursorObjCSuperClassRef(C).second;
3753f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3754a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCProtocolRef:
3755a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCProtocolRef(C).second;
3756f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3757a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_ObjCClassRef:
3758a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorObjCClassRef(C).second;
37597d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3760a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    case CXCursor_TypeRef:
3761a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      return getCursorTypeRef(C).second;
37620b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
37630b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
37640b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return getCursorTemplateRef(C).second;
37650b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
37666931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
37676931900f43cea558c6974075256c07728dbfecc6Douglas Gregor      return getCursorNamespaceRef(C).second;
3768a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
3769a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
3770a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor      return getCursorMemberRef(C).second;
3771a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
37723064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier:
37731b0f7af64113b63253ced088a2bc64eb98e6f388Douglas Gregor      return getCursorCXXBaseSpecifier(C)->getSourceRange();
3774f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
377536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
377636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return getCursorLabelRef(C).second;
377736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
37781f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
37791f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return getCursorOverloadedDeclRef(C).second;
37801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3781a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    default:
3782a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      // FIXME: Need a way to enumerate all non-reference cases.
3783a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor      llvm_unreachable("Missed a reference kind");
3784a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor    }
3785a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor  }
378697b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
378797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind))
3788a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorExpr(C)->getSourceRange();
378933e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor
379033e9abd21083a0191a7676a04b497006d2da184dDouglas Gregor  if (clang_isStatement(C.kind))
3791a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return getCursorStmt(C)->getSourceRange();
3792f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
37936639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis  if (clang_isAttribute(C.kind))
37946639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis    return getCursorAttr(C)->getRange();
37956639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
3796a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (C.kind == CXCursor_PreprocessingDirective)
3797a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return cxcursor::getCursorPreprocessingDirective(C);
37984807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor
37999b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
38009e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    return cxcursor::getCursorMacroExpansion(C)->getSourceRange();
3801572feb2a190b5e8b04fb06c4ac50ee0f61e93ff0Douglas Gregor
3802a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (C.kind == CXCursor_MacroDefinition)
3803a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor    return cxcursor::getCursorMacroDefinition(C)->getSourceRange();
3804ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3805ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (C.kind == CXCursor_InclusionDirective)
3806ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return cxcursor::getCursorInclusionDirective(C)->getSourceRange();
3807ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
3808007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
3809007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    Decl *D = cxcursor::getCursorDecl(C);
3810007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    SourceRange R = D->getSourceRange();
3811007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // FIXME: Multiple variables declared in a single declaration
3812007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // currently lack the information needed to correctly determine their
3813007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // ranges when accounting for the type-specifier.  We use context
3814007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
3815007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    // and if so, whether it is the first decl.
3816007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
3817007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek      if (!cxcursor::isFirstInDeclGroup(C))
3818007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek        R.setBegin(VD->getLocation());
3819007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    }
3820007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek    return R;
3821007a7c9d8dcdb2e9cd94b6075108bfc4c90e6ccdTed Kremenek  }
38226653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return SourceRange();
38236653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
38246653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38256653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// \brief Retrieves the "raw" cursor extent, which is then extended to include
38266653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor/// the decl-specifier-seq for declarations.
38276653798ff5ce6deb58112777e21307ccc453133dDouglas Gregorstatic SourceRange getFullCursorExtent(CXCursor C, SourceManager &SrcMgr) {
38286653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  if (C.kind >= CXCursor_FirstDecl && C.kind <= CXCursor_LastDecl) {
38296653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    Decl *D = cxcursor::getCursorDecl(C);
38306653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    SourceRange R = D->getSourceRange();
38312494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
38322494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // Adjust the start of the location for declarations preceded by
38332494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // declaration specifiers.
38342494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
38356653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    if (const DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) {
38362494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
38372494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
38382494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    } else if (TypedefDecl *Typedef = dyn_cast<TypedefDecl>(D)) {
38392494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
38402494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
38412494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    }
38426653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38432494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && R.getBegin().isValid() &&
38442494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, R.getBegin()))
38452494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      R.setBegin(StartLoc);
38462494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
38472494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // FIXME: Multiple variables declared in a single declaration
38482494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // currently lack the information needed to correctly determine their
38492494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // ranges when accounting for the type-specifier.  We use context
38502494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // stored in the CXCursor to determine if the VarDecl is in a DeclGroup,
38512494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    // and if so, whether it is the first decl.
38522494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
38532494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (!cxcursor::isFirstInDeclGroup(C))
38542494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        R.setBegin(VD->getLocation());
38556653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    }
38566653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38576653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor    return R;
38586653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  }
38596653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor
38606653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor  return getRawCursorExtent(C);
38616653798ff5ce6deb58112777e21307ccc453133dDouglas Gregor}
3862a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3863a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregorextern "C" {
3864a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor
3865a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas GregorCXSourceRange clang_getCursorExtent(CXCursor C) {
3866a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  SourceRange R = getRawCursorExtent(C);
3867a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  if (R.isInvalid())
38685352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
3869f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3870a8e5c5bdbe387b2552c1c23b828f54abcf085a40Douglas Gregor  return cxloc::translateSourceRange(getCursorContext(C), R);
3871a7bde20f8c6334ccc3a7ef4dd77243d0921a8497Douglas Gregor}
3872c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
3873c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas GregorCXCursor clang_getCursorReferenced(CXCursor C) {
3874b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
3875b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
3876f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3877a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit tu = getCursorTU(C);
38781f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (clang_isDeclaration(C.kind)) {
38791f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getCursorDecl(C);
38801f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
3881a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Using, D->getLocation(), tu);
38821f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (ObjCClassDecl *Classes = dyn_cast<ObjCClassDecl>(D))
3883a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Classes, D->getLocation(), tu);
38841f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (ObjCForwardProtocolDecl *Protocols
38851f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                                        = dyn_cast<ObjCForwardProtocolDecl>(D))
3886a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Protocols, D->getLocation(), tu);
38875f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    if (ObjCPropertyImplDecl *PropImpl =dyn_cast<ObjCPropertyImplDecl>(D))
3888e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor      if (ObjCPropertyDecl *Property = PropImpl->getPropertyDecl())
3889e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor        return MakeCXCursor(Property, tu);
3890e3c60a7ce9e0f42c7ca2344b33203266aceca1dbDouglas Gregor
3891c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return C;
38921f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
38931f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
389497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isExpression(C.kind)) {
38951f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Expr *E = getCursorExpr(C);
38961f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    Decl *D = getDeclFromExpr(E);
389797b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    if (D)
3898a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(D, tu);
38991f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
39001f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    if (OverloadExpr *Ovl = dyn_cast_or_null<OverloadExpr>(E))
3901a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCursorOverloadedDeclRef(Ovl, tu);
39021f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
390397b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor    return clang_getNullCursor();
390497b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  }
390597b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor
390636897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  if (clang_isStatement(C.kind)) {
390736897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    Stmt *S = getCursorStmt(C);
390836897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    if (GotoStmt *Goto = dyn_cast_or_null<GotoStmt>(S))
390937c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek      if (LabelDecl *label = Goto->getLabel())
391037c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        if (LabelStmt *labelS = label->getStmt())
391137c2e9664316b013b9a86f841f143f19ffbc0a02Ted Kremenek        return MakeCXCursor(labelS, getCursorDecl(C), tu);
391236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
391336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    return clang_getNullCursor();
391436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor  }
391536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
39169b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion) {
39179e5bb85ac899eeab7c21b5ff9030c3da6ff4837bChandler Carruth    if (MacroDefinition *Def = getCursorMacroExpansion(C)->getDefinition())
3918a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeMacroDefinitionCursor(Def, tu);
3919bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor  }
3920bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
3921c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  if (!clang_isReference(C.kind))
3922c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    return clang_getNullCursor();
3923f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3924c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  switch (C.kind) {
3925c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    case CXCursor_ObjCSuperClassRef:
3926a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCSuperClassRef(C).first, tu);
3927f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3928f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCProtocolRef: {
3929a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCProtocolRef(C).first, tu);
3930f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3931f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_ObjCClassRef:
3932a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorObjCClassRef(C).first, tu );
39337d0d40e58807f73e06ff5eb637a48e9f978b0e2aDouglas Gregor
3934f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek    case CXCursor_TypeRef:
3935a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTypeRef(C).first, tu );
39360b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
39370b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor    case CXCursor_TemplateRef:
3938a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorTemplateRef(C).first, tu );
39390b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor
39406931900f43cea558c6974075256c07728dbfecc6Douglas Gregor    case CXCursor_NamespaceRef:
3941a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorNamespaceRef(C).first, tu );
39426931900f43cea558c6974075256c07728dbfecc6Douglas Gregor
3943a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor    case CXCursor_MemberRef:
3944a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(getCursorMemberRef(C).first, tu );
3945a67e03fdf1ae8a1f92463a307d0b6281f1161f40Douglas Gregor
39463064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    case CXCursor_CXXBaseSpecifier: {
39473064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      CXXBaseSpecifier *B = cxcursor::getCursorCXXBaseSpecifier(C);
39483064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek      return clang_getTypeDeclaration(cxtype::MakeCXType(B->getType(),
3949a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                                         tu ));
39503064ef9e604d19a0cfd0d8e3ed3055bfd83f88fdTed Kremenek    }
3951f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
395236897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor    case CXCursor_LabelRef:
395336897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // FIXME: We end up faking the "parent" declaration here because we
395436897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      // don't want to make CXCursor larger.
395536897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor      return MakeCXCursor(getCursorLabelRef(C).first,
3956a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek               static_cast<ASTUnit*>(tu->TUData)->getASTContext()
3957a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          .getTranslationUnitDecl(),
3958a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          tu);
395936897b05ca2886e287f01802614bc10cbadcec22Douglas Gregor
39601f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    case CXCursor_OverloadedDeclRef:
39611f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor      return C;
39621f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
3963c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    default:
3964c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      // We would prefer to enumerate all non-reference cursor kinds here.
3965c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      llvm_unreachable("Unhandled reference cursor kind");
3966c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor      break;
3967c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor    }
3968c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  }
3969f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3970c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor  return clang_getNullCursor();
3971c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor}
3972c5d1e9375d71e66d22456e7cc52cc7c0a5c65c3fDouglas Gregor
3973b699866820102a69d83d6ac6941985c5ef4e8c40Douglas GregorCXCursor clang_getCursorDefinition(CXCursor C) {
3974b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor  if (clang_isInvalid(C.kind))
3975b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return clang_getNullCursor();
3976f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3977a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(C);
3978f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3979b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  bool WasReference = false;
398097b9872d5775446cb8aca1380e437649fe848d91Douglas Gregor  if (clang_isReference(C.kind) || clang_isExpression(C.kind)) {
3981b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    C = clang_getCursorReferenced(C);
3982b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    WasReference = true;
3983b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
3984b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
39859b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth  if (C.kind == CXCursor_MacroExpansion)
3986bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor    return clang_getCursorReferenced(C);
3987bf7efa2742dc94363d3561d284c9d634fc5a780fDouglas Gregor
3988b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
3989b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
3990b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
3991b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  Decl *D = getCursorDecl(C);
3992b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!D)
3993b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
3994f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
3995b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  switch (D->getKind()) {
3996b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't really separate the notions of
3997b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // declaration and definition.
3998b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Namespace:
3999b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Typedef:
4000162e1c1b487352434552147967c3dd296ebee2f7Richard Smith  case Decl::TypeAlias:
40013e4c6c4c79a03f5cb0c4671d7c282d623c6dc35eRichard Smith  case Decl::TypeAliasTemplate:
4002b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTypeParm:
4003b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::EnumConstant:
4004b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Field:
4005d98114647e16796a976b04af79975b4f0eacf22bBenjamin Kramer  case Decl::IndirectField:
4006b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCIvar:
4007b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCAtDefsField:
4008b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ImplicitParam:
4009b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ParmVar:
4010b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NonTypeTemplateParm:
4011b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TemplateTemplateParm:
4012b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategoryImpl:
4013b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCImplementation:
40146206d53f67613958ae1b023aba337ebb46f11a8bAbramo Bagnara  case Decl::AccessSpec:
4015b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::LinkageSpec:
4016b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCPropertyImpl:
4017b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FileScopeAsm:
4018b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::StaticAssert:
4019b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Block:
4020ad8dcf4a9df0e24051dc31bf9e6f3cd138a34298Chris Lattner  case Decl::Label:  // FIXME: Is this right??
4021af0f4d0b2e38c810effc8b024ad2fb6604eec5d3Francois Pichet  case Decl::ClassScopeFunctionSpecialization:
4022b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return C;
4023b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4024b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds that don't make any sense here, but are
4025b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // nonetheless harmless.
4026b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::TranslationUnit:
4027b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4028b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4029b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  // Declaration kinds for which the definition is not resolvable.
4030b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingTypename:
4031b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UnresolvedUsingValue:
4032b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    break;
4033b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4034b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingDirective:
4035b2cd48756119f4d8d2a865b4b3e0e8efd02e26a0Douglas Gregor    return MakeCXCursor(cast<UsingDirectiveDecl>(D)->getNominatedNamespace(),
4036a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                        TU);
4037b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4038b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::NamespaceAlias:
4039a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<NamespaceAliasDecl>(D)->getNamespace(), TU);
4040b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4041b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Enum:
4042b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Record:
4043b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXRecord:
4044b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplateSpecialization:
4045b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplatePartialSpecialization:
4046952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor    if (TagDecl *Def = cast<TagDecl>(D)->getDefinition())
4047a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
4048b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4049b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4050b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Function:
4051b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXMethod:
4052b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConstructor:
4053b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXDestructor:
4054b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::CXXConversion: {
4055b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4056b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionDecl>(D)->getBody(Def))
4057a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(const_cast<FunctionDecl *>(Def), TU);
4058b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4059b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4060b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4061b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Var: {
406231310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    // Ask the variable if it has a definition.
406331310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    if (VarDecl *Def = cast<VarDecl>(D)->getDefinition())
4064a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def, TU);
406531310a21fb2a9f13950f864f681c86080b05d5b2Sebastian Redl    return clang_getNullCursor();
4066b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4067f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4068b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FunctionTemplate: {
4069b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    const FunctionDecl *Def = 0;
4070b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (cast<FunctionTemplateDecl>(D)->getTemplatedDecl()->getBody(Def))
4071a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Def->getDescribedFunctionTemplate(), TU);
4072b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4073b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4074f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4075b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ClassTemplate: {
4076b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (RecordDecl *Def = cast<ClassTemplateDecl>(D)->getTemplatedDecl()
4077952b017601f9c82b51119c3a1600f1312a833db9Douglas Gregor                                                            ->getDefinition())
40780b36e614aa19df72885d5e0de996f7fbb9874ec3Douglas Gregor      return MakeCXCursor(cast<CXXRecordDecl>(Def)->getDescribedClassTemplate(),
4079a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                          TU);
4080b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4081b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4082b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
40831f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::Using:
40841f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return MakeCursorOverloadedDeclRef(cast<UsingDecl>(D),
4085a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       D->getLocation(), TU);
4086b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4087b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::UsingShadow:
4088b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getCursorDefinition(
4089f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek                       MakeCXCursor(cast<UsingShadowDecl>(D)->getTargetDecl(),
4090a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                    TU));
4091b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4092b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCMethod: {
4093b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    ObjCMethodDecl *Method = cast<ObjCMethodDecl>(D);
4094b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (Method->isThisDeclarationADefinition())
4095b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      return C;
4096b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4097b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // Dig out the method definition in the associated
4098b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // @implementation, if we have it.
4099b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: The ASTs should make finding the definition easier.
4100b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4101b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                       = dyn_cast<ObjCInterfaceDecl>(Method->getDeclContext()))
4102b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (ObjCImplementationDecl *ClassImpl = Class->getImplementation())
4103b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor        if (ObjCMethodDecl *Def = ClassImpl->getMethod(Method->getSelector(),
4104b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                                                  Method->isInstanceMethod()))
4105b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          if (Def->isThisDeclarationADefinition())
4106a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek            return MakeCXCursor(Def, TU);
4107b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4108b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4109b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4110b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4111b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCategory:
4112b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCCategoryImplDecl *Impl
4113b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                               = cast<ObjCCategoryDecl>(D)->getImplementation())
4114a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4115b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4116b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4117b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProtocol:
4118b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (!cast<ObjCProtocolDecl>(D)->isForwardDecl())
4119b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      return C;
4120b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4121b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4122b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCInterface:
4123b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // There are two notions of a "definition" for an Objective-C
4124b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // class: the interface and its implementation. When we resolved a
4125b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // reference to an Objective-C class, produce the @interface as
4126b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // the definition; when we were provided with the interface,
4127b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // produce the @implementation as the definition.
4128b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (WasReference) {
4129b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (!cast<ObjCInterfaceDecl>(D)->isForwardDecl())
4130b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor        return C;
4131b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    } else if (ObjCImplementationDecl *Impl
4132b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor                              = cast<ObjCInterfaceDecl>(D)->getImplementation())
4133a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(Impl, TU);
4134b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4135f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4136b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCProperty:
4137b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // FIXME: We don't really know where to find the
4138b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    // ObjCPropertyImplDecls that implement this property.
4139b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4140b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4141b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::ObjCCompatibleAlias:
4142b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (ObjCInterfaceDecl *Class
4143b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor          = cast<ObjCCompatibleAliasDecl>(D)->getClassInterface())
4144b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor      if (!Class->isForwardDecl())
4145a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek        return MakeCXCursor(Class, TU);
4146f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4147b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4148b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41491f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::ObjCForwardProtocol:
41501f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return MakeCursorOverloadedDeclRef(cast<ObjCForwardProtocolDecl>(D),
4151a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       D->getLocation(), TU);
4152b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41531f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  case Decl::ObjCClass:
41549e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar    return MakeCursorOverloadedDeclRef(cast<ObjCClassDecl>(D), D->getLocation(),
4155a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                       TU);
4156b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4157b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::Friend:
4158b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendDecl>(D)->getFriendDecl())
4159a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4160b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4161b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4162b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  case Decl::FriendTemplate:
4163b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    if (NamedDecl *Friend = cast<FriendTemplateDecl>(D)->getFriendDecl())
4164a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return clang_getCursorDefinition(MakeCXCursor(Friend, TU));
4165b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return clang_getNullCursor();
4166b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  }
4167b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4168b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getNullCursor();
4169b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4170b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4171b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregorunsigned clang_isCursorDefinition(CXCursor C) {
4172b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  if (!clang_isDeclaration(C.kind))
4173b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor    return 0;
4174b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
4175b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor  return clang_getCursorDefinition(C) == C;
4176b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor}
4177b699866820102a69d83d6ac6941985c5ef4e8c40Douglas Gregor
41781a9d0503b67a499797141af0fd6d315d5045f0eaDouglas GregorCXCursor clang_getCanonicalCursor(CXCursor C) {
41791a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  if (!clang_isDeclaration(C.kind))
41801a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return C;
41811a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
4182e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  if (Decl *D = getCursorDecl(C)) {
4183debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis    if (ObjCCategoryImplDecl *CatImplD = dyn_cast<ObjCCategoryImplDecl>(D))
4184debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis      if (ObjCCategoryDecl *CatD = CatImplD->getCategoryDecl())
4185debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis        return MakeCXCursor(CatD, getCursorTU(C));
4186debb00f9ce1dd0f855d2b4fff3372b2ceeb20735Argyrios Kyrtzidis
4187e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis    if (ObjCImplDecl *ImplD = dyn_cast<ObjCImplDecl>(D))
4188e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis      if (ObjCInterfaceDecl *IFD = ImplD->getClassInterface())
4189e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis        return MakeCXCursor(IFD, getCursorTU(C));
4190e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis
41911a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor    return MakeCXCursor(D->getCanonicalDecl(), getCursorTU(C));
4192e2f854ddd365e6837cef3e1a1b7621b32200fc71Argyrios Kyrtzidis  }
41931a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
41941a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor  return C;
41951a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor}
41961a9d0503b67a499797141af0fd6d315d5045f0eaDouglas Gregor
41971f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregorunsigned clang_getNumOverloadedDecls(CXCursor C) {
41987c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (C.kind != CXCursor_OverloadedDeclRef)
41991f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return 0;
42001f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42011f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(C).first;
42021f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
42031f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return E->getNumDecls();
42041f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42051f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
42061f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
42071f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return S->size();
42081f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42091f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
42101f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D))
4211826faa22bae112e01293a58534a40711043cce65Argyrios Kyrtzidis    return Using->shadow_size();
421295ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian  if (isa<ObjCClassDecl>(D))
421395ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian    return 1;
42141f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCForwardProtocolDecl *Protocols =dyn_cast<ObjCForwardProtocolDecl>(D))
42151f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return Protocols->protocol_size();
42161f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42171f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return 0;
42181f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
42191f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42201f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas GregorCXCursor clang_getOverloadedDecl(CXCursor cursor, unsigned index) {
42217c432dd959609a3689c2e4406450c092e6d76d6dDouglas Gregor  if (cursor.kind != CXCursor_OverloadedDeclRef)
42221f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42231f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42241f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (index >= clang_getNumOverloadedDecls(cursor))
42251f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    return clang_getNullCursor();
42261f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
4227a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(cursor);
42281f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  OverloadedDeclRefStorage Storage = getCursorOverloadedDeclRef(cursor).first;
42291f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadExpr *E = Storage.dyn_cast<OverloadExpr *>())
4230a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(E->decls_begin()[index], TU);
42311f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42321f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (OverloadedTemplateStorage *S
42331f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor                              = Storage.dyn_cast<OverloadedTemplateStorage*>())
4234a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(S->begin()[index], TU);
42351f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42361f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  Decl *D = Storage.get<Decl*>();
42371f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (UsingDecl *Using = dyn_cast<UsingDecl>(D)) {
42381f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    // FIXME: This is, unfortunately, linear time.
42391f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    UsingDecl::shadow_iterator Pos = Using->shadow_begin();
42401f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor    std::advance(Pos, index);
4241a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(cast<UsingShadowDecl>(*Pos)->getTargetDecl(), TU);
42421f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  }
42431f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCClassDecl *Classes = dyn_cast<ObjCClassDecl>(D))
424495ed7784a335aca53b0c6e952cf31a4cfb633360Fariborz Jahanian    return MakeCXCursor(Classes->getForwardInterfaceDecl(), TU);
42451f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  if (ObjCForwardProtocolDecl *Protocols = dyn_cast<ObjCForwardProtocolDecl>(D))
4246a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return MakeCXCursor(Protocols->protocol_begin()[index], TU);
42471f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42481f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor  return clang_getNullCursor();
42491f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor}
42501f60d9ea523fc321d811fe880ba9a1ec74fa8f9bDouglas Gregor
42510d7dd222381390731151455ff06358b93d04d6d1Daniel Dunbarvoid clang_getDefinitionSpellingAndExtent(CXCursor C,
42524ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **startBuf,
42534ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          const char **endBuf,
42544ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startLine,
42554ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *startColumn,
42564ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff                                          unsigned *endLine,
42579ebfa31222a704690e9b6b30b7a135fe44364bc2Daniel Dunbar                                          unsigned *endColumn) {
4258283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  assert(getCursorDecl(C) && "CXCursor has null decl");
4259283cae37b03047c14ef918503bc46b08405c3b69Douglas Gregor  NamedDecl *ND = static_cast<NamedDecl *>(getCursorDecl(C));
42604ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  FunctionDecl *FD = dyn_cast<FunctionDecl>(ND);
42614ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  CompoundStmt *Body = dyn_cast<CompoundStmt>(FD->getBody());
4262f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
42634ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  SourceManager &SM = FD->getASTContext().getSourceManager();
42644ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startBuf = SM.getCharacterData(Body->getLBracLoc());
42654ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endBuf = SM.getCharacterData(Body->getRBracLoc());
42664ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startLine = SM.getSpellingLineNumber(Body->getLBracLoc());
42674ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *startColumn = SM.getSpellingColumnNumber(Body->getLBracLoc());
42684ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endLine = SM.getSpellingLineNumber(Body->getRBracLoc());
42694ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff  *endColumn = SM.getSpellingColumnNumber(Body->getRBracLoc());
42704ade6d6eae934f796ca43c81a5aa185e456dde9bSteve Naroff}
4271f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4272430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4273430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas GregorCXSourceRange clang_getCursorReferenceNameRange(CXCursor C, unsigned NameFlags,
4274430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                                                unsigned PieceIndex) {
4275430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  RefNamePieces Pieces;
4276430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4277430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  switch (C.kind) {
4278430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_MemberRefExpr:
4279430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (MemberExpr *E = dyn_cast<MemberExpr>(getCursorExpr(C)))
4280430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, true, E->getMemberNameInfo(),
4281430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange());
4282430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4283430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4284430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_DeclRefExpr:
4285430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (DeclRefExpr *E = dyn_cast<DeclRefExpr>(getCursorExpr(C)))
4286430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Pieces = buildPieces(NameFlags, false, E->getNameInfo(),
4287430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getQualifierLoc().getSourceRange(),
4288430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                           E->getExplicitTemplateArgsOpt());
4289430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4290430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4291430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  case CXCursor_CallExpr:
4292430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (CXXOperatorCallExpr *OCE =
4293430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        dyn_cast<CXXOperatorCallExpr>(getCursorExpr(C))) {
4294430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      Expr *Callee = OCE->getCallee();
4295430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(Callee))
4296430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Callee = ICE->getSubExpr();
4297430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4298430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Callee))
4299430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        Pieces = buildPieces(NameFlags, false, DRE->getNameInfo(),
4300430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor                             DRE->getQualifierLoc().getSourceRange());
4301430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    }
4302430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4303430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4304430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  default:
4305430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    break;
4306430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4307430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4308430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  if (Pieces.empty()) {
4309430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor    if (PieceIndex == 0)
4310430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      return clang_getCursorExtent(C);
4311430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  } else if (PieceIndex < Pieces.size()) {
4312430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      SourceRange R = Pieces[PieceIndex];
4313430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor      if (R.isValid())
4314430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor        return cxloc::translateSourceRange(getCursorContext(C), R);
4315430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  }
4316430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
4317430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor  return clang_getNullRange();
4318430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor}
4319430d7a1a621a126a9ffe442ad8987ba02b46dae9Douglas Gregor
43200a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregorvoid clang_enableStackTraces(void) {
43210a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor  llvm::sys::PrintStackTraceOnErrorSignal();
43220a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor}
43230a812cf707da15dadd19fdeb0178b9707f4e01a6Douglas Gregor
4324995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbarvoid clang_executeOnThread(void (*fn)(void*), void *user_data,
4325995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar                           unsigned stack_size) {
4326995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar  llvm::llvm_execute_on_thread(fn, user_data, stack_size);
4327995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar}
4328995aaf9c8f0131bef0215a9a0bc794b83a49e0b7Daniel Dunbar
4329fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek} // end: extern "C"
4330fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek
4331fb48049fd5f0457d9052269f6d84768412f5f6dfTed Kremenek//===----------------------------------------------------------------------===//
4332fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor// Token-based Operations.
4333fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
4334fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4335fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor/* CXToken layout:
4336fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[0]: a CXTokenKind
4337fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[1]: starting token location
4338fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[2]: token length
4339fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   int_data[3]: reserved
4340f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek *   ptr_data: for identifiers and keywords, an IdentifierInfo*.
4341fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor *   otherwise unused.
4342fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor */
4343fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorextern "C" {
4344fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4345fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXTokenKind clang_getTokenKind(CXToken CXTok) {
4346fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return static_cast<CXTokenKind>(CXTok.int_data[0]);
4347fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4348fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4349fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXString clang_getTokenSpelling(CXTranslationUnit TU, CXToken CXTok) {
4350fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  switch (clang_getTokenKind(CXTok)) {
4351fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Identifier:
4352fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Keyword:
4353fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We know we have an IdentifierInfo*, so use that.
4354ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString(static_cast<IdentifierInfo *>(CXTok.ptr_data)
4355ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek                            ->getNameStart());
4356fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4357fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Literal: {
4358fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // We have stashed the starting pointer in the ptr_data field. Use it.
4359fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    const char *Text = static_cast<const char *>(CXTok.ptr_data);
43605f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner    return createCXString(StringRef(Text, CXTok.int_data[2]));
4361fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4362f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4363fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Punctuation:
4364fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  case CXToken_Comment:
4365fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    break;
4366fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  }
4367f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4368f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // We have to find the starting buffer pointer the hard way, by
4369fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // deconstructing the source location.
4370a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4371fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4372ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek    return createCXString("");
4373f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4374fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceLocation Loc = SourceLocation::getFromRawEncoding(CXTok.int_data[1]);
4375fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> LocInfo
4376a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    = CXXUnit->getSourceManager().getDecomposedSpellingLoc(Loc);
4377f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
43785f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4379f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = CXXUnit->getSourceManager().getBufferData(LocInfo.first, &Invalid);
4380f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  if (Invalid)
4381aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor    return createCXString("");
4382fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4383f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  return createCXString(Buffer.substr(LocInfo.second, CXTok.int_data[2]));
4384fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4385f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4386fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceLocation clang_getTokenLocation(CXTranslationUnit TU, CXToken CXTok) {
4387a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4388fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit)
4389fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return clang_getNullLocation();
4390f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4391fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  return cxloc::translateSourceLocation(CXXUnit->getASTContext(),
4392fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4393fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4394fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
4395fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas GregorCXSourceRange clang_getTokenExtent(CXTranslationUnit TU, CXToken CXTok) {
4396a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
43975352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor  if (!CXXUnit)
43985352ac06d8f6194825bb2a99ffa009b61bafb503Douglas Gregor    return clang_getNullRange();
4399f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4400f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  return cxloc::translateSourceRange(CXXUnit->getASTContext(),
4401fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                        SourceLocation::getFromRawEncoding(CXTok.int_data[1]));
4402fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
4403f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4404fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregorvoid clang_tokenize(CXTranslationUnit TU, CXSourceRange Range,
4405fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor                    CXToken **Tokens, unsigned *NumTokens) {
4406fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (Tokens)
4407fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *Tokens = 0;
4408fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (NumTokens)
4409fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    *NumTokens = 0;
4410f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4411a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
4412fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (!CXXUnit || !Tokens || !NumTokens)
4413fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4414f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4415bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
4416bdf6062bc10aa3b73b16402b440b8073310acd06Douglas Gregor
441785b988fdfa6adab6d43e16efd19ad4f3f7e2b49bDaniel Dunbar  SourceRange R = cxloc::translateCXSourceRange(Range);
4418fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (R.isInvalid())
4419fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4420f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4421fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  SourceManager &SourceMgr = CXXUnit->getSourceManager();
4422fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> BeginLocInfo
4423fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    = SourceMgr.getDecomposedLoc(R.getBegin());
4424fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  std::pair<FileID, unsigned> EndLocInfo
4425fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    = SourceMgr.getDecomposedLoc(R.getEnd());
4426f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4427fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Cannot tokenize across files.
4428fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (BeginLocInfo.first != EndLocInfo.first)
4429fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4430f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4431f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek  // Create a lexer
4432f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor  bool Invalid = false;
44335f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer
4434f715ca12bfc9fddfde75f98a197424434428b821Douglas Gregor    = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid);
443547a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor  if (Invalid)
443647a3fcd4afe122b23f9e7b6148f147bfa460cfe8Douglas Gregor    return;
4437aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor
4438fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
4439fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor            CXXUnit->getASTContext().getLangOptions(),
4440f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer            Buffer.begin(), Buffer.data() + BeginLocInfo.second, Buffer.end());
4441fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Lex.SetCommentRetentionState(true);
4442f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4443fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  // Lex tokens until we hit the end of the range.
4444f6ac97b101c8840efa92bf29166077ce4049e293Benjamin Kramer  const char *EffectiveBufferEnd = Buffer.data() + EndLocInfo.second;
44455f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<CXToken, 32> CXTokens;
4446fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  Token Tok;
4447096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall  bool previousWasAt = false;
4448fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  do {
4449fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Lex the next token
4450fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    Lex.LexFromRawLexer(Tok);
4451fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.is(tok::eof))
4452fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      break;
4453f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4454fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    // Initialize the CXToken.
4455fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXToken CXTok;
4456f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4457fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Common fields
4458fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[1] = Tok.getLocation().getRawEncoding();
4459fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[2] = Tok.getLength();
4460fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTok.int_data[3] = 0;
4461f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4462fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    //   - Kind-specific fields
4463fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    if (Tok.isLiteral()) {
4464fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Literal;
4465fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = (void *)Tok.getLiteralData();
4466c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara    } else if (Tok.is(tok::raw_identifier)) {
4467aea67dbd653a2dd6dd5cc2159279e81e855b2482Douglas Gregor      // Lookup the identifier to determine whether we have a keyword.
4468fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      IdentifierInfo *II
4469c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        = CXXUnit->getPreprocessor().LookUpIdentifierInfo(Tok);
4470aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek
4471096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall      if ((II->getObjCKeywordID() != tok::objc_not_keyword) && previousWasAt) {
4472aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek        CXTok.int_data[0] = CXToken_Keyword;
4473aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4474aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      else {
4475c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara        CXTok.int_data[0] = Tok.is(tok::identifier)
4476c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          ? CXToken_Identifier
4477c4bf2b9afb7d47445a9dc6bc848657098a4e3851Abramo Bagnara          : CXToken_Keyword;
4478aa8a66de0e7951ba5a305f3e5a39c9b14a318c42Ted Kremenek      }
4479fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = II;
4480fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else if (Tok.is(tok::comment)) {
4481fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Comment;
4482fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4483fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    } else {
4484fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.int_data[0] = CXToken_Punctuation;
4485fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor      CXTok.ptr_data = 0;
4486fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    }
4487fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    CXTokens.push_back(CXTok);
4488096428b351ebf5de9871ce11e06ba6f2d8276ab5David Chisnall    previousWasAt = Tok.is(tok::at);
4489fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  } while (Lex.getBufferLocation() <= EffectiveBufferEnd);
4490f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4491fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  if (CXTokens.empty())
4492fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor    return;
4493f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
4494fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *Tokens = (CXToken *)malloc(sizeof(CXToken) * CXTokens.size());
4495fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  memmove(*Tokens, CXTokens.data(), sizeof(CXToken) * CXTokens.size());
4496fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor  *NumTokens = CXTokens.size();
4497fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
44980045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
44996db610934bedc6896393c1e1099525b35380acd6Ted Kremenekvoid clang_disposeTokens(CXTranslationUnit TU,
45006db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                         CXToken *Tokens, unsigned NumTokens) {
45016db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  free(Tokens);
45026db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
45036db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45046db610934bedc6896393c1e1099525b35380acd6Ted Kremenek} // end: extern "C"
45056db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45066db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
45076db610934bedc6896393c1e1099525b35380acd6Ted Kremenek// Token annotation APIs.
45086db610934bedc6896393c1e1099525b35380acd6Ted Kremenek//===----------------------------------------------------------------------===//
45096db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
45100045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregortypedef llvm::DenseMap<unsigned, CXCursor> AnnotateTokensData;
4511fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
4512fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXCursor parent,
4513fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                                                     CXClientData client_data);
45146db610934bedc6896393c1e1099525b35380acd6Ted Kremeneknamespace {
45156db610934bedc6896393c1e1099525b35380acd6Ted Kremenekclass AnnotateTokensWorker {
45166db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  AnnotateTokensData &Annotated;
451711949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXToken *Tokens;
451811949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  CXCursor *Cursors;
451911949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  unsigned NumTokens;
4520fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned TokIdx;
45214419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  unsigned PreprocessingTokIdx;
4522fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CursorVisitor AnnotateVis;
4523fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceManager &SrcMgr;
4524f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool HasContextSensitiveKeywords;
4525f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4526fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  bool MoreTokens() const { return TokIdx < NumTokens; }
4527fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  unsigned NextToken() const { return TokIdx; }
4528fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void AdvanceToken() { ++TokIdx; }
4529fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation GetTokenLoc(unsigned tokI) {
4530fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4531fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
45325f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  bool isFunctionMacroToken(unsigned tokI) const {
4533a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return Tokens[tokI].int_data[3] != 0;
4534a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
45355f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  SourceLocation getFunctionMacroTokenLoc(unsigned tokI) const {
4536a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[3]);
4537a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4538a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4539a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  void annotateAndAdvanceTokens(CXCursor, RangeComparisonResult, SourceRange);
45405f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void annotateAndAdvanceFunctionMacroTokens(CXCursor, RangeComparisonResult,
45415f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                             SourceRange);
4542fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
45436db610934bedc6896393c1e1099525b35380acd6Ted Kremenekpublic:
454411949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  AnnotateTokensWorker(AnnotateTokensData &annotated,
4545fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek                       CXToken *tokens, CXCursor *cursors, unsigned numTokens,
4546a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                       CXTranslationUnit tu, SourceRange RegionOfInterest)
454711949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    : Annotated(annotated), Tokens(tokens), Cursors(cursors),
45484419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      NumTokens(numTokens), TokIdx(0), PreprocessingTokIdx(0),
4549a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      AnnotateVis(tu,
455008e0bc16b3312c27e87d33be7dcf3d4fe5bdd2e2Douglas Gregor                  AnnotateTokensVisitor, this, true, RegionOfInterest),
4551f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      SrcMgr(static_cast<ASTUnit*>(tu->TUData)->getSourceManager()),
4552f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      HasContextSensitiveKeywords(false) { }
455311949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4554fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void VisitChildren(CXCursor C) { AnnotateVis.VisitChildren(C); }
45556db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  enum CXChildVisitResult Visit(CXCursor cursor, CXCursor parent);
4556fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  void AnnotateTokens(CXCursor parent);
4557ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek  void AnnotateTokens() {
4558a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    AnnotateTokens(clang_getTranslationUnitCursor(AnnotateVis.getTU()));
4559ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek  }
4560f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4561f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// \brief Determine whether the annotator saw any cursors that have
4562f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  /// context-sensitive keywords.
4563f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  bool hasContextSensitiveKeywords() const {
4564f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    return HasContextSensitiveKeywords;
4565f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
45666db610934bedc6896393c1e1099525b35380acd6Ted Kremenek};
45676db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
45680045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
4569fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenekvoid AnnotateTokensWorker::AnnotateTokens(CXCursor parent) {
4570fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Walk the AST within the region of interest, annotating tokens
4571fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // along the way.
4572fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  VisitChildren(parent);
4573fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4574fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = 0 ; I < TokIdx ; ++I) {
457511949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
45764419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    if (Pos != Annotated.end() &&
45774419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        (clang_isInvalid(Cursors[I].kind) ||
45784419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor         Pos->second.kind != CXCursor_PreprocessingDirective))
4579fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      Cursors[I] = Pos->second;
4580fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4581fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4582fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Finish up annotating any tokens left.
4583fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (!MoreTokens())
4584fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return;
458511949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4586fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor &C = clang_getNullCursor();
4587fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = TokIdx ; I < NumTokens ; ++I) {
4588fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    AnnotateTokensData::iterator Pos = Annotated.find(Tokens[I].int_data[1]);
4589fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = (Pos == Annotated.end()) ? C : Pos->second;
459011949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek  }
459111949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek}
459211949cbae3f42c32a2933fb85b125620666d98eaTed Kremenek
4593a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief It annotates and advances tokens with a cursor until the comparison
4594a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis//// between the cursor location and the source range is the same as
4595a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \arg compResult.
4596a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis///
4597a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeBefore to annotate tokens with a cursor until a range is reached.
4598a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// Pass RangeOverlap to annotate tokens inside a range.
4599a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceTokens(CXCursor updateC,
4600a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4601a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
4602a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  while (MoreTokens()) {
4603a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    const unsigned I = NextToken();
46045f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    if (isFunctionMacroToken(I))
46055f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      return annotateAndAdvanceFunctionMacroTokens(updateC, compResult, range);
4606a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4607a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceLocation TokLoc = GetTokenLoc(I);
4608a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4609a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      Cursors[I] = updateC;
4610a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      AdvanceToken();
4611a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue;
4612a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4613a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    break;
4614a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4615a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4616a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4617a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Special annotation handling for macro argument tokens.
46185f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidisvoid AnnotateTokensWorker::annotateAndAdvanceFunctionMacroTokens(
46195f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis                                               CXCursor updateC,
4620a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               RangeComparisonResult compResult,
4621a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                               SourceRange range) {
46225f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(MoreTokens());
46235f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  assert(isFunctionMacroToken(NextToken()) &&
4624a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis         "Should be called only for macro arg tokens");
4625a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4626a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // This works differently than annotateAndAdvanceTokens; because expanded
4627a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // macro arguments can have arbitrary translation-unit source order, we do not
4628a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // advance the token index one by one until a token fails the range test.
4629a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // We only advance once past all of the macro arg tokens if all of them
4630a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // pass the range test. If one of them fails we keep the token index pointing
4631a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // at the start of the macro arg tokens so that the failing token will be
4632a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // annotated by a subsequent annotation try.
4633a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4634a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  bool atLeastOneCompFail = false;
4635a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4636a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned I = NextToken();
46375f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  for (; I < NumTokens && isFunctionMacroToken(I); ++I) {
46385f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis    SourceLocation TokLoc = getFunctionMacroTokenLoc(I);
4639a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (TokLoc.isFileID())
4640a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      continue; // not macro arg token, it's parens or comma.
4641a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (LocationCompare(SrcMgr, TokLoc, range) == compResult) {
4642a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (clang_isInvalid(clang_getCursorKind(Cursors[I])))
4643a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        Cursors[I] = updateC;
4644a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    } else
4645a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      atLeastOneCompFail = true;
4646a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4647a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4648a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (!atLeastOneCompFail)
4649a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    TokIdx = I; // All of the tokens were handled, advance beyond all of them.
4650a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4651a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
46526db610934bedc6896393c1e1099525b35380acd6Ted Kremenekenum CXChildVisitResult
46534419b675577d7c281a659fab1fec10e1bfbe04c5Douglas GregorAnnotateTokensWorker::Visit(CXCursor cursor, CXCursor parent) {
4654fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  CXSourceLocation Loc = clang_getCursorLocation(cursor);
46554419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  SourceRange cursorRange = getRawCursorExtent(cursor);
465681d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor  if (cursorRange.isInvalid())
465781d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor    return CXChildVisit_Recurse;
4658f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
4659f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (!HasContextSensitiveKeywords) {
4660f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C properties can have context-sensitive keywords.
4661f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    if (cursor.kind == CXCursor_ObjCPropertyDecl) {
4662f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCPropertyDecl *Property
4663f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(cursor)))
4664f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        HasContextSensitiveKeywords = Property->getPropertyAttributesAsWritten() != 0;
4665f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4666f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // Objective-C methods can have context-sensitive keywords.
4667f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_ObjCInstanceMethodDecl ||
4668f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ObjCClassMethodDecl) {
4669f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (ObjCMethodDecl *Method
4670f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            = dyn_cast_or_null<ObjCMethodDecl>(getCursorDecl(cursor))) {
4671f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->getObjCDeclQualifier())
4672f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4673f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        else {
4674f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          for (ObjCMethodDecl::param_iterator P = Method->param_begin(),
4675f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                                           PEnd = Method->param_end();
4676f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor               P != PEnd; ++P) {
4677f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            if ((*P)->getObjCDeclQualifier()) {
4678f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              HasContextSensitiveKeywords = true;
4679f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              break;
4680f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            }
4681f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          }
4682f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
4683f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4684f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4685f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ methods can have context-sensitive keywords.
4686f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_CXXMethod) {
4687f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (CXXMethodDecl *Method
4688f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor                  = dyn_cast_or_null<CXXMethodDecl>(getCursorDecl(cursor))) {
4689f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (Method->hasAttr<FinalAttr>() || Method->hasAttr<OverrideAttr>())
4690f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4691f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
4692f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4693f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    // C++ classes can have context-sensitive keywords.
4694f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    else if (cursor.kind == CXCursor_StructDecl ||
4695f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassDecl ||
4696f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplate ||
4697f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor             cursor.kind == CXCursor_ClassTemplatePartialSpecialization) {
4698f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Decl *D = getCursorDecl(cursor))
4699f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (D->hasAttr<FinalAttr>())
4700f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          HasContextSensitiveKeywords = true;
4701f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
4702f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
4703f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
47044419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor  if (clang_isPreprocessing(cursor.kind)) {
4705cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // For macro expansions, just note where the beginning of the macro
4706cea731a9cb7de3f473d60e5ea544e25621cebd76Chandler Carruth    // expansion occurs.
47079b2a0ac970a077bdc0bf08c6c682f80ad733c892Chandler Carruth    if (cursor.kind == CXCursor_MacroExpansion) {
47084419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      Annotated[Loc.int_data] = cursor;
47094419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      return CXChildVisit_Recurse;
47104419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47114419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47124419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Items in the preprocessing record are kept separate from items in
47134419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // declarations, so we keep a separate token index.
47144419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    unsigned SavedTokIdx = TokIdx;
47154419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = PreprocessingTokIdx;
47164419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47174419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Skip tokens up until we catch up to the beginning of the preprocessing
47184419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // entry.
47194419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
47204419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
47214419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
47224419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
47234419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
47244419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
47254419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
47264419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
47274419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
47284419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
47294419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
47304419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
47314419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47324419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47334419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Look at all of the tokens within this range.
47344419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    while (MoreTokens()) {
47354419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      const unsigned I = NextToken();
47364419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      SourceLocation TokLoc = GetTokenLoc(I);
47374419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      switch (LocationCompare(SrcMgr, TokLoc, cursorRange)) {
47384419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeBefore:
47394419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        assert(0 && "Infeasible");
47404419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeAfter:
47414419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        break;
47424419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      case RangeOverlap:
47434419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        Cursors[I] = cursor;
47444419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        AdvanceToken();
47454419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor        continue;
47464419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      }
47474419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor      break;
47484419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    }
47494419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47504419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // Save the preprocessing token index; restore the non-preprocessing
47514419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    // token index.
47524419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    PreprocessingTokIdx = TokIdx;
47534419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor    TokIdx = SavedTokIdx;
47540045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor    return CXChildVisit_Recurse;
47550045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  }
4756fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4757fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  if (cursorRange.isInvalid())
4758fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    return CXChildVisit_Continue;
4759a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek
4760fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  SourceLocation L = SourceLocation::getFromRawEncoding(Loc.int_data);
4761fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4762a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  // Adjust the annotated range based specific declarations.
4763a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  const enum CXCursorKind cursorK = clang_getCursorKind(cursor);
4764a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  if (cursorK >= CXCursor_FirstDecl && cursorK <= CXCursor_LastDecl) {
476523173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    Decl *D = cxcursor::getCursorDecl(cursor);
47662494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
47672494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    SourceLocation StartLoc;
476823173d7f029f430611caceea72ae61ba6b80af1cTed Kremenek    if (const DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) {
47692494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = DD->getTypeSourceInfo())
47702494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
47712494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    } else if (TypedefDecl *Typedef = dyn_cast<TypedefDecl>(D)) {
47722494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      if (TypeSourceInfo *TI = Typedef->getTypeSourceInfo())
47732494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        StartLoc = TI->getTypeLoc().getSourceRange().getBegin();
4774a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek    }
47752494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor
47762494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor    if (StartLoc.isValid() && L.isValid() &&
47772494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor        SrcMgr.isBeforeInTranslationUnit(StartLoc, L))
47782494dd024b392b8def58bf067cc94b51c214cf77Douglas Gregor      cursorRange.setBegin(StartLoc);
4779a333c66b94f6d6dcd5ee7e625d816d3c0300e8e6Ted Kremenek  }
478081d3c04b0934c43518355289ad104d34f6fde06fDouglas Gregor
47813f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // If the location of the cursor occurs within a macro instantiation, record
47823f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // the spelling location of the cursor in our annotation map.  We can then
47833f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // paper over the token labelings during a post-processing step to try and
47843f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // get cursor mappings for tokens that are the *arguments* of a macro
47853f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  // instantiation.
47863f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  if (L.isMacroID()) {
47873f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    unsigned rawEncoding = SrcMgr.getSpellingLoc(L).getRawEncoding();
47883f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // Only invalidate the old annotation if it isn't part of a preprocessing
47893f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // directive.  Here we assume that the default construction of CXCursor
47903f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // results in CXCursor.kind being an initialized value (i.e., 0).  If
47913f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    // this isn't the case, we can fix by doing lookup + insertion.
47924419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
47933f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    CXCursor &oldC = Annotated[rawEncoding];
47943f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek    if (!clang_isPreprocessing(oldC.kind))
47953f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek      oldC = cursor;
47963f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek  }
47973f4046004be223b03f1f895bb934e44921ccf805Ted Kremenek
4798fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const enum CXCursorKind K = clang_getCursorKind(parent);
4799fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const CXCursor updateC =
4800d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek    (clang_isInvalid(K) || K == CXCursor_TranslationUnit)
4801d8b0a84d586bc0a08695968acf2f169c9d01da69Ted Kremenek     ? clang_getNullCursor() : parent;
4802fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4803a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(updateC, RangeBefore, cursorRange);
4804fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
48055517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // Avoid having the cursor of an expression "overwrite" the annotation of the
48065517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // variable declaration that it belongs to.
48075517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // This can happen for C++ constructor expressions whose range generally
48085517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  // include the variable declaration, e.g.:
48095517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  //  MyCXXClass foo; // Make sure we don't annotate 'foo' as a CallExpr cursor.
48105517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  if (clang_isExpression(cursorK)) {
48115517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    Expr *E = getCursorExpr(cursor);
48128ccac3de1335f1cfd7cea56ba1cefcf0b724ce3fArgyrios Kyrtzidis    if (Decl *D = getCursorParentDecl(cursor)) {
48135517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      const unsigned I = NextToken();
48145517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      if (E->getLocStart().isValid() && D->getLocation().isValid() &&
48155517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == D->getLocation() &&
48165517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis          E->getLocStart() == GetTokenLoc(I)) {
48175517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        Cursors[I] = updateC;
48185517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis        AdvanceToken();
48195517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis      }
48205517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis    }
48215517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis  }
48225517b89953e3c9276f161ce29831de388bb2573dArgyrios Kyrtzidis
4823fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Visit children to get their cursor information.
4824fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned BeforeChildren = NextToken();
4825fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  VisitChildren(cursor);
4826fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  const unsigned AfterChildren = NextToken();
4827fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4828a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // Scan the tokens that are at the end of the cursor, but are not captured
4829a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  // but the child cursors.
4830a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  annotateAndAdvanceTokens(cursor, RangeOverlap, cursorRange);
48316db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
4832fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Scan the tokens that are at the beginning of the cursor, but are not
4833fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // capture by the child cursors.
4834fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  for (unsigned I = BeforeChildren; I != AfterChildren; ++I) {
4835fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    if (!clang_isInvalid(clang_getCursorKind(Cursors[I])))
4836fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek      break;
48374419b675577d7c281a659fab1fec10e1bfbe04c5Douglas Gregor
4838fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    Cursors[I] = cursor;
4839fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  }
4840fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
4841fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  return CXChildVisit_Continue;
48420045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor}
48430045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor
48446db610934bedc6896393c1e1099525b35380acd6Ted Kremenekstatic enum CXChildVisitResult AnnotateTokensVisitor(CXCursor cursor,
48456db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXCursor parent,
48466db610934bedc6896393c1e1099525b35380acd6Ted Kremenek                                                     CXClientData client_data) {
48476db610934bedc6896393c1e1099525b35380acd6Ted Kremenek  return static_cast<AnnotateTokensWorker*>(client_data)->Visit(cursor, parent);
48486db610934bedc6896393c1e1099525b35380acd6Ted Kremenek}
48496db610934bedc6896393c1e1099525b35380acd6Ted Kremenek
48506628a614c504263ae539462f049d523dd07ac1baTed Kremeneknamespace {
4851a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4852a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// \brief Uses the macro expansions in the preprocessing record to find
4853a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// and mark tokens that are macro arguments. This info is used by the
4854a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis/// AnnotateTokensWorker.
4855a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisclass MarkMacroArgTokensVisitor {
4856a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceManager &SM;
4857a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXToken *Tokens;
4858a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned NumTokens;
4859a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  unsigned CurIdx;
4860a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4861a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidispublic:
4862a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  MarkMacroArgTokensVisitor(SourceManager &SM,
4863a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                            CXToken *tokens, unsigned numTokens)
4864a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    : SM(SM), Tokens(tokens), NumTokens(numTokens), CurIdx(0) { }
4865a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4866a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  CXChildVisitResult visit(CXCursor cursor, CXCursor parent) {
4867a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (cursor.kind != CXCursor_MacroExpansion)
4868a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue;
4869a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4870a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    SourceRange macroRange = getCursorMacroExpansion(cursor)->getSourceRange();
4871a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (macroRange.getBegin() == macroRange.getEnd())
4872a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Continue; // it's not a function macro.
4873a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4874a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4875a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(getTokenLoc(CurIdx),
4876a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                        macroRange.getBegin()))
4877a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4878a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4879a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4880a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4881a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4882a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4883a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    for (; CurIdx < NumTokens; ++CurIdx) {
4884a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      SourceLocation tokLoc = getTokenLoc(CurIdx);
4885a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      if (!SM.isBeforeInTranslationUnit(tokLoc, macroRange.getEnd()))
4886a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis        break;
4887a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
48885f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis      setFunctionMacroTokenLoc(CurIdx, SM.getMacroArgExpandedLocation(tokLoc));
4889a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    }
4890a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4891a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    if (CurIdx == NumTokens)
4892a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis      return CXChildVisit_Break;
4893a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4894a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return CXChildVisit_Continue;
4895a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4896a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4897a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisprivate:
4898a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  SourceLocation getTokenLoc(unsigned tokI) {
4899a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    return SourceLocation::getFromRawEncoding(Tokens[tokI].int_data[1]);
4900a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4901a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
49025f616b726fdf5db3cbc2fd8d92c42cc624dd4550Argyrios Kyrtzidis  void setFunctionMacroTokenLoc(unsigned tokI, SourceLocation loc) {
4903a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // The third field is reserved and currently not used. Use it here
4904a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // to mark macro arg expanded tokens with their expanded locations.
4905a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    Tokens[tokI].int_data[3] = loc.getRawEncoding();
4906a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
4907a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis};
4908a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4909a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis} // end anonymous namespace
4910a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4911a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisstatic CXChildVisitResult
4912a676379b26edc959193f9f919ba9c6d296a57824Argyrios KyrtzidisMarkMacroArgTokensVisitorDelegate(CXCursor cursor, CXCursor parent,
4913a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                  CXClientData client_data) {
4914a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  return static_cast<MarkMacroArgTokensVisitor*>(client_data)->visit(cursor,
4915a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                                                     parent);
4916a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis}
4917a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
4918a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidisnamespace {
49196628a614c504263ae539462f049d523dd07ac1baTed Kremenek  struct clang_annotateTokens_Data {
49206628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXTranslationUnit TU;
49216628a614c504263ae539462f049d523dd07ac1baTed Kremenek    ASTUnit *CXXUnit;
49226628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXToken *Tokens;
49236628a614c504263ae539462f049d523dd07ac1baTed Kremenek    unsigned NumTokens;
49246628a614c504263ae539462f049d523dd07ac1baTed Kremenek    CXCursor *Cursors;
49256628a614c504263ae539462f049d523dd07ac1baTed Kremenek  };
4926ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek}
4927ab97961fb4424d0822076eb0fd4f8faee9992763Ted Kremenek
49286628a614c504263ae539462f049d523dd07ac1baTed Kremenek// This gets run a separate thread to avoid stack blowout.
49296628a614c504263ae539462f049d523dd07ac1baTed Kremenekstatic void clang_annotateTokensImpl(void *UserData) {
49306628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXTranslationUnit TU = ((clang_annotateTokens_Data*)UserData)->TU;
49316628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = ((clang_annotateTokens_Data*)UserData)->CXXUnit;
49326628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXToken *Tokens = ((clang_annotateTokens_Data*)UserData)->Tokens;
49336628a614c504263ae539462f049d523dd07ac1baTed Kremenek  const unsigned NumTokens = ((clang_annotateTokens_Data*)UserData)->NumTokens;
49346628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor *Cursors = ((clang_annotateTokens_Data*)UserData)->Cursors;
4935fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
49360396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Determine the region of interest, which contains all of the tokens.
49370045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  SourceRange RegionOfInterest;
49386628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setBegin(
49396628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU, Tokens[0])));
49406628a614c504263ae539462f049d523dd07ac1baTed Kremenek  RegionOfInterest.setEnd(
49416628a614c504263ae539462f049d523dd07ac1baTed Kremenek    cxloc::translateSourceLocation(clang_getTokenLocation(TU,
49426628a614c504263ae539462f049d523dd07ac1baTed Kremenek                                                         Tokens[NumTokens-1])));
4943fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek
49440396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // A mapping from the source locations found when re-lexing or traversing the
49450396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // region of interest to the corresponding cursors.
49460045e9fe1f7dfc37f1ea7bdb9b70bcdb6700f0c0Douglas Gregor  AnnotateTokensData Annotated;
49476628a614c504263ae539462f049d523dd07ac1baTed Kremenek
4948fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // Relex the tokens within the source range to look for preprocessing
49490396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // directives.
49509f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  SourceManager &SourceMgr = CXXUnit->getSourceManager();
49519f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  std::pair<FileID, unsigned> BeginLocInfo
49529f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    = SourceMgr.getDecomposedLoc(RegionOfInterest.getBegin());
49539f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor  std::pair<FileID, unsigned> EndLocInfo
49549f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    = SourceMgr.getDecomposedLoc(RegionOfInterest.getEnd());
49556628a614c504263ae539462f049d523dd07ac1baTed Kremenek
49565f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  StringRef Buffer;
49570396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  bool Invalid = false;
49580396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  if (BeginLocInfo.first == EndLocInfo.first &&
49590396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor      ((Buffer = SourceMgr.getBufferData(BeginLocInfo.first, &Invalid)),true) &&
49600396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor      !Invalid) {
49619f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    Lexer Lex(SourceMgr.getLocForStartOfFile(BeginLocInfo.first),
49629f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor              CXXUnit->getASTContext().getLangOptions(),
4963fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek              Buffer.begin(), Buffer.data() + BeginLocInfo.second,
49644ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor              Buffer.end());
49659f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    Lex.SetCommentRetentionState(true);
49666628a614c504263ae539462f049d523dd07ac1baTed Kremenek
4967fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek    // Lex tokens in raw mode until we hit the end of the range, to avoid
49689f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    // entering #includes or expanding macros.
49694807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor    while (true) {
49709f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor      Token Tok;
49719f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor      Lex.LexFromRawLexer(Tok);
49726628a614c504263ae539462f049d523dd07ac1baTed Kremenek
49739f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    reprocess:
49749f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor      if (Tok.is(tok::hash) && Tok.isAtStartOfLine()) {
49759f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        // We have found a preprocessing directive. Gobble it up so that we
49769e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar        // don't see it while preprocessing these tokens later, but keep track
49779e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar        // of all of the token locations inside this preprocessing directive so
49789e1ebdd97ba0c593900cb7f103324c43471e96eeDaniel Dunbar        // that we can annotate them appropriately.
49799f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        //
49809f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        // FIXME: Some simple tests here could identify macro definitions and
49819f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        // #undefs, to provide specific cursor kinds for those.
49825f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner        SmallVector<SourceLocation, 32> Locations;
49839f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        do {
49849f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor          Locations.push_back(Tok.getLocation());
4985fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek          Lex.LexFromRawLexer(Tok);
49869f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        } while (!Tok.isAtStartOfLine() && !Tok.is(tok::eof));
49876628a614c504263ae539462f049d523dd07ac1baTed Kremenek
49889f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        using namespace cxcursor;
49899f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        CXCursor Cursor
49906628a614c504263ae539462f049d523dd07ac1baTed Kremenek        = MakePreprocessingDirectiveCursor(SourceRange(Locations.front(),
49916628a614c504263ae539462f049d523dd07ac1baTed Kremenek                                                       Locations.back()),
4992a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                                           TU);
49939f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        for (unsigned I = 0, N = Locations.size(); I != N; ++I) {
49949f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor          Annotated[Locations[I].getRawEncoding()] = Cursor;
49959f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        }
49966628a614c504263ae539462f049d523dd07ac1baTed Kremenek
49979f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        if (Tok.isAtStartOfLine())
49989f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor          goto reprocess;
49996628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50009f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        continue;
50019f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor      }
50026628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50034807231938d8aff28de09f78f301f9ba5845e5e4Douglas Gregor      if (Tok.is(tok::eof))
50049f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor        break;
50059f1e3ff3b3095967e2b92b57a53524e2d6bb141cDouglas Gregor    }
50064ae8f298b1ea51b4c2234f9148e2e4349c9bdd23Douglas Gregor  }
50076628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5008a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  if (CXXUnit->getPreprocessor().getPreprocessingRecord()) {
5009a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    // Search and mark tokens that are macro argument expansions.
5010a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MarkMacroArgTokensVisitor Visitor(CXXUnit->getSourceManager(),
5011a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                      Tokens, NumTokens);
5012a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    CursorVisitor MacroArgMarker(TU,
5013a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis                                 MarkMacroArgTokensVisitorDelegate, &Visitor,
501408e0bc16b3312c27e87d33be7dcf3d4fe5bdd2e2Douglas Gregor                                 true, RegionOfInterest);
5015a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis    MacroArgMarker.visitPreprocessedEntitiesInRegion();
5016a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis  }
5017a676379b26edc959193f9f919ba9c6d296a57824Argyrios Kyrtzidis
50180396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  // Annotate all of the source locations in the region of interest that map to
5019fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  // a specific cursor.
5020fbd84caf62a21afa4db5f730e2e2603ead17bb65Ted Kremenek  AnnotateTokensWorker W(Annotated, Tokens, Cursors, NumTokens,
5021a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek                         TU, RegionOfInterest);
50226628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50236c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // FIXME: We use a ridiculous stack size here because the data-recursion
50246c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm uses a large stack frame than the non-data recursive version,
50256c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // and AnnotationTokensWorker currently transforms the data-recursion
50266c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // algorithm back into a traditional recursion by explicitly calling
50276c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  // VisitChildren().  We will need to remove this explicit recursive call.
50286628a614c504263ae539462f049d523dd07ac1baTed Kremenek  W.AnnotateTokens();
50296628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5030f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // If we ran into any entities that involve context-sensitive keywords,
5031f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  // take another pass through the tokens to mark them as such.
5032f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  if (W.hasContextSensitiveKeywords()) {
5033f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    for (unsigned I = 0; I != NumTokens; ++I) {
5034f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (clang_getTokenKind(Tokens[I]) != CXToken_Identifier)
5035f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5036f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5037f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCPropertyDecl) {
5038f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5039f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (ObjCPropertyDecl *Property
50406628a614c504263ae539462f049d523dd07ac1baTed Kremenek            = dyn_cast_or_null<ObjCPropertyDecl>(getCursorDecl(Cursors[I]))) {
5041f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          if (Property->getPropertyAttributesAsWritten() != 0 &&
5042f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor              llvm::StringSwitch<bool>(II->getName())
50436628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readonly", true)
50446628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("assign", true)
5045f85e193739c953358c865005855253af4f68a497John McCall              .Case("unsafe_unretained", true)
50466628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("readwrite", true)
50476628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("retain", true)
50486628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("copy", true)
50496628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("nonatomic", true)
50506628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("atomic", true)
50516628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("getter", true)
50526628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Case("setter", true)
5053f85e193739c953358c865005855253af4f68a497John McCall              .Case("strong", true)
5054f85e193739c953358c865005855253af4f68a497John McCall              .Case("weak", true)
50556628a614c504263ae539462f049d523dd07ac1baTed Kremenek              .Default(false))
5056f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor            Tokens[I].int_data[0] = CXToken_Keyword;
5057f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        }
5058f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5059f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5060f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor
5061f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      if (Cursors[I].kind == CXCursor_ObjCInstanceMethodDecl ||
5062f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Cursors[I].kind == CXCursor_ObjCClassMethodDecl) {
5063f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        IdentifierInfo *II = static_cast<IdentifierInfo *>(Tokens[I].ptr_data);
5064f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        if (llvm::StringSwitch<bool>(II->getName())
50656628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("in", true)
50666628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("out", true)
50676628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("inout", true)
50686628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("oneway", true)
50696628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("bycopy", true)
50706628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Case("byref", true)
50716628a614c504263ae539462f049d523dd07ac1baTed Kremenek            .Default(false))
5072f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor          Tokens[I].int_data[0] = CXToken_Keyword;
5073f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor        continue;
5074f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
50756639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis
50766639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis      if (Cursors[I].kind == CXCursor_CXXFinalAttr ||
50776639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis          Cursors[I].kind == CXCursor_CXXOverrideAttr) {
50786639e9255489ad8e10278d5658fdd4b3c0e1e4cdArgyrios Kyrtzidis        Tokens[I].int_data[0] = CXToken_Keyword;
50796628a614c504263ae539462f049d523dd07ac1baTed Kremenek        continue;
5080f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor      }
5081f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor    }
5082f52516038ab5d0b1b90a6dd32f46b7d6dabd04c8Douglas Gregor  }
5083fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor}
50846628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50856628a614c504263ae539462f049d523dd07ac1baTed Kremenekextern "C" {
50866628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50876628a614c504263ae539462f049d523dd07ac1baTed Kremenekvoid clang_annotateTokens(CXTranslationUnit TU,
50886628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXToken *Tokens, unsigned NumTokens,
50896628a614c504263ae539462f049d523dd07ac1baTed Kremenek                          CXCursor *Cursors) {
50906628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50916628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (NumTokens == 0 || !Tokens || !Cursors)
50926628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
50936628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50946628a614c504263ae539462f049d523dd07ac1baTed Kremenek  // Any token we don't specifically annotate will have a NULL cursor.
50956628a614c504263ae539462f049d523dd07ac1baTed Kremenek  CXCursor C = clang_getNullCursor();
50966628a614c504263ae539462f049d523dd07ac1baTed Kremenek  for (unsigned I = 0; I != NumTokens; ++I)
50976628a614c504263ae539462f049d523dd07ac1baTed Kremenek    Cursors[I] = C;
50986628a614c504263ae539462f049d523dd07ac1baTed Kremenek
50996628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit *CXXUnit = static_cast<ASTUnit *>(TU->TUData);
51006628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!CXXUnit)
51016628a614c504263ae539462f049d523dd07ac1baTed Kremenek    return;
51026628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51036628a614c504263ae539462f049d523dd07ac1baTed Kremenek  ASTUnit::ConcurrencyCheck Check(*CXXUnit);
51046628a614c504263ae539462f049d523dd07ac1baTed Kremenek
51056628a614c504263ae539462f049d523dd07ac1baTed Kremenek  clang_annotateTokens_Data data = { TU, CXXUnit, Tokens, NumTokens, Cursors };
51066628a614c504263ae539462f049d523dd07ac1baTed Kremenek  llvm::CrashRecoveryContext CRC;
51076628a614c504263ae539462f049d523dd07ac1baTed Kremenek  if (!RunSafely(CRC, clang_annotateTokensImpl, &data,
51086628a614c504263ae539462f049d523dd07ac1baTed Kremenek                 GetSafetyThreadStackSize() * 2)) {
51096628a614c504263ae539462f049d523dd07ac1baTed Kremenek    fprintf(stderr, "libclang: crash detected while annotating tokens\n");
51106628a614c504263ae539462f049d523dd07ac1baTed Kremenek  }
51116628a614c504263ae539462f049d523dd07ac1baTed Kremenek}
51126628a614c504263ae539462f049d523dd07ac1baTed Kremenek
5113fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor} // end: extern "C"
5114fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor
5115fc8ea23eb6cbaaa5046f2abb4c033e24c8659efdDouglas Gregor//===----------------------------------------------------------------------===//
511616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek// Operations for querying linkage of a cursor.
511716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
511816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
511916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenekextern "C" {
512016b4259aecaa22b642d35d36fd89965ed700c1e0Ted KremenekCXLinkageKind clang_getCursorLinkage(CXCursor cursor) {
51210396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor  if (!clang_isDeclaration(cursor.kind))
51220396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor    return CXLinkage_Invalid;
51230396f466978d65789eca85b2738e9df53adc78f5Douglas Gregor
512416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  Decl *D = cxcursor::getCursorDecl(cursor);
512516b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  if (NamedDecl *ND = dyn_cast_or_null<NamedDecl>(D))
512616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    switch (ND->getLinkage()) {
512716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case NoLinkage: return CXLinkage_NoLinkage;
512816b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case InternalLinkage: return CXLinkage_Internal;
512916b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case UniqueExternalLinkage: return CXLinkage_UniqueExternal;
513016b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek      case ExternalLinkage: return CXLinkage_External;
513116b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek    };
513216b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
513316b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek  return CXLinkage_Invalid;
513416b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek}
513516b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek} // end: extern "C"
513616b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek
513716b4259aecaa22b642d35d36fd89965ed700c1e0Ted Kremenek//===----------------------------------------------------------------------===//
513845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek// Operations for querying language of a cursor.
513945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
514045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
514145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekstatic CXLanguageKind getDeclLanguage(const Decl *D) {
514245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  switch (D->getKind()) {
514345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    default:
514445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      break;
514545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ImplicitParam:
514645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCAtDefsField:
514745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategory:
514845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCategoryImpl:
514945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCClass:
515045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCCompatibleAlias:
515145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCForwardProtocol:
515245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCImplementation:
515345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCInterface:
515445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCIvar:
515545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCMethod:
515645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProperty:
515745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCPropertyImpl:
515845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ObjCProtocol:
515945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_ObjC;
516045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConstructor:
516145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXConversion:
516245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXDestructor:
516345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXMethod:
516445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::CXXRecord:
516545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplate:
516645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplatePartialSpecialization:
516745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::ClassTemplateSpecialization:
516845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Friend:
516945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FriendTemplate:
517045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::FunctionTemplate:
517145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::LinkageSpec:
517245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Namespace:
517345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NamespaceAlias:
517445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::NonTypeTemplateParm:
517545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::StaticAssert:
517645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTemplateParm:
517745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::TemplateTypeParm:
517845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingTypename:
517945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UnresolvedUsingValue:
518045e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::Using:
518145e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingDirective:
518245e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    case Decl::UsingShadow:
518345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek      return CXLanguage_CPlusPlus;
518445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  }
518545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
518645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_C;
518745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
518845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
518945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenekextern "C" {
519058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
519158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregorenum CXAvailabilityKind clang_getCursorAvailability(CXCursor cursor) {
519258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  if (clang_isDeclaration(cursor.kind))
519358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    if (Decl *D = cxcursor::getCursorDecl(cursor)) {
51940a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      if (isa<FunctionDecl>(D) && cast<FunctionDecl>(D)->isDeleted())
519558ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Available;
519658ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
51970a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      switch (D->getAvailability()) {
51980a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Available:
51990a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_NotYetIntroduced:
52000a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_Available;
52010a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52020a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Deprecated:
520358ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor        return CXAvailability_Deprecated;
52040a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
52050a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      case AR_Unavailable:
52060a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor        return CXAvailability_NotAvailable;
52070a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor      }
520858ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor    }
52090a0d2b179085a52c10402feebeb6db8b4d96a140Douglas Gregor
521058ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor  return CXAvailability_Available;
521158ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor}
521258ddb60f409125eda5436c4a1f070f7fa4744295Douglas Gregor
521345e1dae500bba7a9ef5b8206263a5609c07c6f03Ted KremenekCXLanguageKind clang_getCursorLanguage(CXCursor cursor) {
521445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  if (clang_isDeclaration(cursor.kind))
521545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek    return getDeclLanguage(cxcursor::getCursorDecl(cursor));
521645e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
521745e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek  return CXLanguage_Invalid;
521845e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek}
52193910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52203910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// \brief If the given cursor is the "templated" declaration
52213910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// descibing a class or function template, return the class or
52223910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor /// function template.
52233910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregorstatic Decl *maybeGetTemplateCursor(Decl *D) {
52243910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (!D)
52253910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    return 0;
52263910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52273910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
52283910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (FunctionTemplateDecl *FunTmpl = FD->getDescribedFunctionTemplate())
52293910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return FunTmpl;
52303910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52313910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  if (CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D))
52323910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor    if (ClassTemplateDecl *ClassTmpl = RD->getDescribedClassTemplate())
52333910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return ClassTmpl;
52343910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52353910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor  return D;
52363910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor}
52373910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52382be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorSemanticParent(CXCursor cursor) {
52392be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
52402be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
52412be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getDeclContext();
52423910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
52433910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
52443910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52453910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
52463910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
52472be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
52482be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52492be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52502be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isStatement(cursor.kind) || clang_isExpression(cursor.kind)) {
52512be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor))
5252a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      return MakeCXCursor(D, getCursorTU(cursor));
52532be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52542be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52552be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
52562be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
52572be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52582be5bc9ad3981347a000742f81b91ab3080f1214Douglas GregorCXCursor clang_getCursorLexicalParent(CXCursor cursor) {
52592be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  if (clang_isDeclaration(cursor.kind)) {
52602be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    if (Decl *D = getCursorDecl(cursor)) {
52612be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor      DeclContext *DC = D->getLexicalDeclContext();
52623910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      if (!DC)
52633910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor        return clang_getNullCursor();
52643910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor
52653910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor      return MakeCXCursor(maybeGetTemplateCursor(cast<Decl>(DC)),
52663910cfd17fcd99ac80158e625fc63e4784d26435Douglas Gregor                          getCursorTU(cursor));
52672be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor    }
52682be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  }
52692be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52702be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // FIXME: Note that we can't easily compute the lexical context of a
52712be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  // statement or expression, so we return nothing.
52722be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor  return clang_getNullCursor();
52732be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor}
52742be5bc9ad3981347a000742f81b91ab3080f1214Douglas Gregor
52759f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorstatic void CollectOverriddenMethods(DeclContext *Ctx,
52769f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                     ObjCMethodDecl *Method,
52775f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner                            SmallVectorImpl<ObjCMethodDecl *> &Methods) {
52789f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!Ctx)
52799f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
52809f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
52819f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  // If we have a class or category implementation, jump straight to the
52829f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  // interface.
52839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (ObjCImplDecl *Impl = dyn_cast<ObjCImplDecl>(Ctx))
52849f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return CollectOverriddenMethods(Impl->getClassInterface(), Method, Methods);
52859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
52869f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  ObjCContainerDecl *Container = dyn_cast<ObjCContainerDecl>(Ctx);
52879f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!Container)
52889f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
52899f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
52909f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  // Check whether we have a matching method at this level.
52919f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (ObjCMethodDecl *Overridden = Container->getMethod(Method->getSelector(),
52929f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                                    Method->isInstanceMethod()))
52939f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (Method != Overridden) {
52949f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      // We found an override at this level; there is no need to look
52959f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      // into other protocols or categories.
52969f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      Methods.push_back(Overridden);
52979f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      return;
52989f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    }
52999f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53009f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (ObjCProtocolDecl *Protocol = dyn_cast<ObjCProtocolDecl>(Container)) {
53019f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    for (ObjCProtocolDecl::protocol_iterator P = Protocol->protocol_begin(),
53029f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                          PEnd = Protocol->protocol_end();
53039f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor         P != PEnd; ++P)
53049f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      CollectOverriddenMethods(*P, Method, Methods);
53059f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  }
53069f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53079f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (ObjCCategoryDecl *Category = dyn_cast<ObjCCategoryDecl>(Container)) {
53089f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    for (ObjCCategoryDecl::protocol_iterator P = Category->protocol_begin(),
53099f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                          PEnd = Category->protocol_end();
53109f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor         P != PEnd; ++P)
53119f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      CollectOverriddenMethods(*P, Method, Methods);
53129f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  }
53139f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53149f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (ObjCInterfaceDecl *Interface = dyn_cast<ObjCInterfaceDecl>(Container)) {
53159f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    for (ObjCInterfaceDecl::protocol_iterator P = Interface->protocol_begin(),
53169f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                           PEnd = Interface->protocol_end();
53179f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor         P != PEnd; ++P)
53189f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      CollectOverriddenMethods(*P, Method, Methods);
53199f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53209f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    for (ObjCCategoryDecl *Category = Interface->getCategoryList();
53219f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor         Category; Category = Category->getNextClassCategory())
53229f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      CollectOverriddenMethods(Category, Method, Methods);
53239f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53249f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    // We only look into the superclass if we haven't found anything yet.
53259f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (Methods.empty())
53269f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      if (ObjCInterfaceDecl *Super = Interface->getSuperClass())
53279f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor        return CollectOverriddenMethods(Super, Method, Methods);
53289f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  }
53299f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53309f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53319f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_getOverriddenCursors(CXCursor cursor,
53329f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                CXCursor **overridden,
53339f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor                                unsigned *num_overridden) {
53349f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (overridden)
53359f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *overridden = 0;
53369f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (num_overridden)
53379f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *num_overridden = 0;
53389f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!overridden || !num_overridden)
53399f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53409f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53419f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!clang_isDeclaration(cursor.kind))
53429f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53439f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53449f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  Decl *D = getCursorDecl(cursor);
53459f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!D)
53469f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53479f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53489f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  // Handle C++ member functions.
5349a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek  CXTranslationUnit TU = getCursorTU(cursor);
53509f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (CXXMethodDecl *CXXMethod = dyn_cast<CXXMethodDecl>(D)) {
53519f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *num_overridden = CXXMethod->size_overridden_methods();
53529f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    if (!*num_overridden)
53539f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor      return;
53549f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53559f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    *overridden = new CXCursor [*num_overridden];
53569f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    unsigned I = 0;
53579f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    for (CXXMethodDecl::method_iterator
53589f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor              M = CXXMethod->begin_overridden_methods(),
53599f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor           MEnd = CXXMethod->end_overridden_methods();
53609f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor         M != MEnd; (void)++M, ++I)
5361a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek      (*overridden)[I] = MakeCXCursor(const_cast<CXXMethodDecl*>(*M), TU);
53629f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53639f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  }
53649f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53659f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  ObjCMethodDecl *Method = dyn_cast<ObjCMethodDecl>(D);
53669f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (!Method)
53679f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53689f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53699f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  // Handle Objective-C methods.
53705f9e272e632e951b1efe824cd16acb4d96077930Chris Lattner  SmallVector<ObjCMethodDecl *, 4> Methods;
53719f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  CollectOverriddenMethods(Method->getDeclContext(), Method, Methods);
53729f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53739f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  if (Methods.empty())
53749f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor    return;
53759f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53769f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  *num_overridden = Methods.size();
53779f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  *overridden = new CXCursor [Methods.size()];
53789f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  for (unsigned I = 0, N = Methods.size(); I != N; ++I)
5379a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    (*overridden)[I] = MakeCXCursor(Methods[I], TU);
53809f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53819f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
53829f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregorvoid clang_disposeOverriddenCursors(CXCursor *overridden) {
53839f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor  delete [] overridden;
53849f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor}
53859f59234a91d057cee7c5e3cee91da8696858c692Douglas Gregor
5386ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas GregorCXFile clang_getIncludedFile(CXCursor cursor) {
5387ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  if (cursor.kind != CXCursor_InclusionDirective)
5388ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor    return 0;
5389ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
5390ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  InclusionDirective *ID = getCursorInclusionDirective(cursor);
5391ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor  return (void *)ID->getFile();
5392ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor}
5393ecdcb883cbc6bb4a2445dc6f02d58d9bdb54a0edDouglas Gregor
539445e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek} // end: extern "C"
539545e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek
53969ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
53979ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
53989ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek// C++ AST instrospection.
53999ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek//===----------------------------------------------------------------------===//
54009ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
54019ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekextern "C" {
54029ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenekunsigned clang_CXXMethod_isStatic(CXCursor C) {
54039ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek  if (!clang_isDeclaration(C.kind))
54049ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek    return 0;
540549f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor
540649f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  CXXMethodDecl *Method = 0;
540749f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
540849f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
540949f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
541049f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  else
541149f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
541249f6f5489483beaffc7ce48dfc000af4e65b9216Douglas Gregor  return (Method && Method->isStatic()) ? 1 : 0;
541340b492a43bac3ed0c465772aa6921d011cfc273fTed Kremenek}
5414b12903e1a4b8d1b611b8c7e4f910665d628e68cdTed Kremenek
5415211924b563aa31421836cee7655be729ad02733fDouglas Gregorunsigned clang_CXXMethod_isVirtual(CXCursor C) {
5416211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (!clang_isDeclaration(C.kind))
5417211924b563aa31421836cee7655be729ad02733fDouglas Gregor    return 0;
5418211924b563aa31421836cee7655be729ad02733fDouglas Gregor
5419211924b563aa31421836cee7655be729ad02733fDouglas Gregor  CXXMethodDecl *Method = 0;
5420211924b563aa31421836cee7655be729ad02733fDouglas Gregor  Decl *D = cxcursor::getCursorDecl(C);
5421211924b563aa31421836cee7655be729ad02733fDouglas Gregor  if (FunctionTemplateDecl *FunTmpl = dyn_cast_or_null<FunctionTemplateDecl>(D))
5422211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast<CXXMethodDecl>(FunTmpl->getTemplatedDecl());
5423211924b563aa31421836cee7655be729ad02733fDouglas Gregor  else
5424211924b563aa31421836cee7655be729ad02733fDouglas Gregor    Method = dyn_cast_or_null<CXXMethodDecl>(D);
5425211924b563aa31421836cee7655be729ad02733fDouglas Gregor  return (Method && Method->isVirtual()) ? 1 : 0;
5426211924b563aa31421836cee7655be729ad02733fDouglas Gregor}
54279ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek} // end: extern "C"
54289ada39a4ac82ff5f5087b0a7fa9ed0d32be55a3bTed Kremenek
542945e1dae500bba7a9ef5b8206263a5609c07c6f03Ted Kremenek//===----------------------------------------------------------------------===//
543095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek// Attribute introspection.
543195f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
543295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
543395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenekextern "C" {
543495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted KremenekCXType clang_getIBOutletCollectionType(CXCursor C) {
543595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  if (C.kind != CXCursor_IBOutletCollectionAttr)
5436a60ed47da13393796d8552b9fdca12abbb3eea42Ted Kremenek    return cxtype::MakeCXType(QualType(), cxcursor::getCursorTU(C));
543795f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
543895f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek  IBOutletCollectionAttr *A =
543995f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek    cast<IBOutletCollectionAttr>(cxcursor::getCursorAttr(C));
544095f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
544118aa2ff4641847d7f8866e8c5912d4d0ddb858ceArgyrios Kyrtzidis  return cxtype::MakeCXType(A->getInterface(), cxcursor::getCursorTU(C));
544295f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek}
544395f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek} // end: extern "C"
544495f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek
544595f33555a6d51b6537a9ed3968c3d1c2e4991b51Ted Kremenek//===----------------------------------------------------------------------===//
544659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek// Inspecting memory usage.
544759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
544859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5449f787002478f09af1741fb0f82a562002e6799c49Ted Kremenektypedef std::vector<CXTUResourceUsageEntry> MemUsageEntries;
545059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5451f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekstatic inline void createCXTUResourceUsageEntry(MemUsageEntries &entries,
5452f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek                                              enum CXTUResourceUsageKind k,
5453ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek                                              unsigned long amount) {
5454f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsageEntry entry = { k, amount };
545559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.push_back(entry);
545659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
545759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
545859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenekextern "C" {
545959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5460f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekconst char *clang_getTUResourceUsageName(CXTUResourceUsageKind kind) {
546159fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  const char *str = "";
546259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  switch (kind) {
5463f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_AST:
546459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: expressions, declarations, and types";
546559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5466f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Identifiers:
546759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: identifiers";
546859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      break;
5469f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_Selectors:
547059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek      str = "ASTContext: selectors";
5471e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5472f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    case CXTUResourceUsage_GlobalCompletionResults:
54734e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      str = "Code completion: cached global results";
5474e294ab717fc9535429ca5d8f575d41ae4441d822Ted Kremenek      break;
5475457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek    case CXTUResourceUsage_SourceManagerContentCache:
5476457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      str = "SourceManager: content cache allocator";
5477457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek      break;
5478ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    case CXTUResourceUsage_AST_SideTables:
5479ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      str = "ASTContext: side tables";
5480ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek      break;
5481f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_Malloc:
5482f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: malloc'ed memory buffers";
5483f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5484f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    case CXTUResourceUsage_SourceManager_Membuffer_MMap:
5485f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      str = "SourceManager: mmap'ed memory buffers";
5486f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek      break;
5487e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc:
5488e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: malloc'ed memory buffers";
5489e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
5490e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    case CXTUResourceUsage_ExternalASTSource_Membuffer_MMap:
5491e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      str = "ExternalASTSource: mmap'ed memory buffers";
5492e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      break;
54935e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_Preprocessor:
54945e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: malloc'ed memory";
54955e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
54965e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    case CXTUResourceUsage_PreprocessingRecord:
54975e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      str = "Preprocessor: PreprocessingRecord";
54985e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek      break;
5499ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek    case CXTUResourceUsage_SourceManager_DataStructures:
5500ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      str = "SourceManager: data structures and tables";
5501ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek      break;
5502d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek    case CXTUResourceUsage_Preprocessor_HeaderSearch:
5503d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      str = "Preprocessor: header search tables";
5504d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek      break;
550559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
550659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return str;
550759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
550859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5509f787002478f09af1741fb0f82a562002e6799c49Ted KremenekCXTUResourceUsage clang_getCXTUResourceUsage(CXTranslationUnit TU) {
551059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (!TU) {
5511f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek    CXTUResourceUsage usage = { (void*) 0, 0, 0 };
551259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    return usage;
551359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  }
551459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
551559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTUnit *astUnit = static_cast<ASTUnit*>(TU->TUData);
551659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  llvm::OwningPtr<MemUsageEntries> entries(new MemUsageEntries());
551759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  ASTContext &astContext = astUnit->getASTContext();
551859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
551959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by AST nodes and types?
5520f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST,
5521ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getASTAllocatedMemory());
552259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
552359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used by identifiers?
5524f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Identifiers,
552559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Idents.getAllocator().getTotalMemory());
552659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
552759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  // How much memory is used for selectors?
5528f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_Selectors,
552959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    (unsigned long) astContext.Selectors.getTotalMemory());
553059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5531ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  // How much memory is used by ASTContext's side tables?
5532ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek  createCXTUResourceUsageEntry(*entries, CXTUResourceUsage_AST_SideTables,
5533ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek    (unsigned long) astContext.getSideTableAllocatedMemory());
5534ba29bd25515fbd99e98ba0fedb9d93617b27609eTed Kremenek
55354e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  // How much memory is used for caching global code completion results?
55364e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  unsigned long completionBytes = 0;
55374e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  if (GlobalCodeCompletionAllocator *completionAllocator =
55384e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek      astUnit->getCachedCompletionAllocator().getPtr()) {
55395e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    completionBytes = completionAllocator->getTotalMemory();
55404e6a3f7310d3d9232877ed6f439247b1054b1e47Ted Kremenek  }
5541457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5542457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               CXTUResourceUsage_GlobalCompletionResults,
5543457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek                               completionBytes);
5544457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek
5545457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  // How much memory is being used by SourceManager's content cache?
5546457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek  createCXTUResourceUsageEntry(*entries,
5547457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          CXTUResourceUsage_SourceManagerContentCache,
5548457aaf0692dfb2d9638f383334b81027f637f20cTed Kremenek          (unsigned long) astContext.getSourceManager().getContentCacheSize());
5549f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5550f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  // How much memory is being used by the MemoryBuffer's in SourceManager?
5551f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  const SourceManager::MemoryBufferSizes &srcBufs =
5552f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek    astUnit->getSourceManager().getMemoryBufferSizes();
5553f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek
5554f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5555f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_Malloc,
5556f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.malloc_bytes);
5557ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5558f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               CXTUResourceUsage_SourceManager_Membuffer_MMap,
5559f61b831d7f6a15676b07647f507de80324cb7056Ted Kremenek                               (unsigned long) srcBufs.mmap_bytes);
5560ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5561ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               CXTUResourceUsage_SourceManager_DataStructures,
5562ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                               (unsigned long) astContext.getSourceManager()
5563ca7dc2b755eb81ac95121ce1a1f1aa44a4a0fe12Ted Kremenek                                .getDataStructureSizes());
5564e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5565e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  // How much memory is being used by the ExternalASTSource?
5566e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  if (ExternalASTSource *esrc = astContext.getExternalSource()) {
5567e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    const ExternalASTSource::MemoryBufferSizes &sizes =
5568e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      esrc->getMemoryBufferSizes();
5569e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek
5570e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5571e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_Malloc,
5572e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.malloc_bytes);
5573e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek    createCXTUResourceUsageEntry(*entries,
5574e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek      CXTUResourceUsage_ExternalASTSource_Membuffer_MMap,
5575e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek                                 (unsigned long) sizes.mmap_bytes);
5576e9b5f3d4acfc2ad6e8b65a4072464e997dea9ed3Ted Kremenek  }
55775e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
55785e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  // How much memory is being used by the Preprocessor?
55795e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  Preprocessor &pp = astUnit->getPreprocessor();
55805e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  createCXTUResourceUsageEntry(*entries,
55815e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                               CXTUResourceUsage_Preprocessor,
5582c5c5e92ec53f7e6ac7ebbbf77c6d8e4b7d88daecArgyrios Kyrtzidis                               pp.getTotalMemory());
55835e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
55845e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  if (PreprocessingRecord *pRec = pp.getPreprocessingRecord()) {
55855e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek    createCXTUResourceUsageEntry(*entries,
55865e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 CXTUResourceUsage_PreprocessingRecord,
55875e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek                                 pRec->getTotalMemory());
55885e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek  }
55895e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5590d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek  createCXTUResourceUsageEntry(*entries,
5591d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               CXTUResourceUsage_Preprocessor_HeaderSearch,
5592d1194fbbf65374bfa3578eb40a547e4f97b497d1Ted Kremenek                               pp.getHeaderSearchInfo().getTotalMemory());
55935e1db6a434d0e3fe0fbde0bca2ec44552818fb22Ted Kremenek
5594f787002478f09af1741fb0f82a562002e6799c49Ted Kremenek  CXTUResourceUsage usage = { (void*) entries.get(),
559559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            (unsigned) entries->size(),
559659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek                            entries->size() ? &(*entries)[0] : 0 };
559759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  entries.take();
559859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  return usage;
559959fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
560059fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5601f787002478f09af1741fb0f82a562002e6799c49Ted Kremenekvoid clang_disposeCXTUResourceUsage(CXTUResourceUsage usage) {
560259fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek  if (usage.data)
560359fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek    delete (MemUsageEntries*) usage.data;
560459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek}
560559fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
560659fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek} // end extern "C"
560759fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
56086df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregorvoid clang::PrintLibclangResourceUsage(CXTranslationUnit TU) {
56096df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  CXTUResourceUsage Usage = clang_getCXTUResourceUsage(TU);
56106df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  for (unsigned I = 0; I != Usage.numEntries; ++I)
56116df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor    fprintf(stderr, "  %s: %lu\n",
56126df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            clang_getTUResourceUsageName(Usage.entries[I].kind),
56136df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor            Usage.entries[I].amount);
56146df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
56156df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor  clang_disposeCXTUResourceUsage(Usage);
56166df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor}
56176df78739f3b3f672f9735445741cbcfa7c10fb1fDouglas Gregor
561859fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek//===----------------------------------------------------------------------===//
561904bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek// Misc. utility functions.
562004bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek//===----------------------------------------------------------------------===//
5621f0e23e8ad647f41d090d76cc09684ecf4bb05a5bTed Kremenek
5622abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbar/// Default to using an 8 MB stack size on "safety" threads.
5623abdce7abc8a22dd2fe79a05c0b71864039bd8296Daniel Dunbarstatic unsigned SafetyStackThreadSize = 8 << 20;
5624bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5625bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarnamespace clang {
5626bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5627bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarbool RunSafely(llvm::CrashRecoveryContext &CRC,
56286c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               void (*Fn)(void*), void *UserData,
56296c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek               unsigned Size) {
56306c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (!Size)
56316c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek    Size = GetSafetyThreadStackSize();
56326c53fdd88f0d75875365463822fa817d7b1a9573Ted Kremenek  if (Size)
5633bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar    return CRC.RunSafelyOnThread(Fn, UserData, Size);
5634bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return CRC.RunSafely(Fn, UserData);
5635bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5636bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5637bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarunsigned GetSafetyThreadStackSize() {
5638bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  return SafetyStackThreadSize;
5639bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5640bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5641bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbarvoid SetSafetyThreadStackSize(unsigned Value) {
5642bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar  SafetyStackThreadSize = Value;
5643bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5644bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
5645bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar}
5646bf44c3b099602c9c967f1b20995919fb4ef39a51Daniel Dunbar
564704bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenekextern "C" {
564804bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
5649a2a9d6e4e5b6001b86b7dfc5db1ea296ce29a3d3Ted KremenekCXString clang_getClangVersion() {
5650ee4db4fccbcab7e6bd614fe8757c5453e7ad0e86Ted Kremenek  return createCXString(getClangFullVersion());
565104bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek}
565204bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek
565304bb716aea8fd2372ac10b0c640cabc5e5caa615Ted Kremenek} // end: extern "C"
565459fc1e55da9c856d1703d3d3ac14a36320d26b30Ted Kremenek
5655