xref: /llvm-project/clang/unittests/Index/IndexTests.cpp (revision 51094545e28d222dc11eea513904506624d715c6)
14d22172bSEric Liu //===--- IndexTests.cpp - Test indexing actions -----------------*- C++ -*-===//
24d22172bSEric Liu //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
64d22172bSEric Liu //
74d22172bSEric Liu //===----------------------------------------------------------------------===//
84d22172bSEric Liu 
94d22172bSEric Liu #include "clang/AST/ASTConsumer.h"
10f53621bbSKadir Cetinkaya #include "clang/AST/ASTContext.h"
114d22172bSEric Liu #include "clang/AST/Decl.h"
12f53621bbSKadir Cetinkaya #include "clang/Basic/SourceLocation.h"
13f53621bbSKadir Cetinkaya #include "clang/Basic/SourceManager.h"
144d22172bSEric Liu #include "clang/Frontend/CompilerInstance.h"
154d22172bSEric Liu #include "clang/Frontend/FrontendAction.h"
164d22172bSEric Liu #include "clang/Index/IndexDataConsumer.h"
174d22172bSEric Liu #include "clang/Index/IndexSymbol.h"
184d22172bSEric Liu #include "clang/Index/IndexingAction.h"
194d22172bSEric Liu #include "clang/Lex/Preprocessor.h"
204d22172bSEric Liu #include "clang/Tooling/Tooling.h"
214d22172bSEric Liu #include "llvm/ADT/StringRef.h"
22fc51490bSJonas Devlieghere #include "llvm/Support/VirtualFileSystem.h"
234d22172bSEric Liu #include "gmock/gmock.h"
244d22172bSEric Liu #include "gtest/gtest.h"
254d22172bSEric Liu #include <memory>
264d22172bSEric Liu 
274d22172bSEric Liu namespace clang {
284d22172bSEric Liu namespace index {
29f53621bbSKadir Cetinkaya namespace {
30f53621bbSKadir Cetinkaya struct Position {
31f53621bbSKadir Cetinkaya   size_t Line = 0;
32f53621bbSKadir Cetinkaya   size_t Column = 0;
33f53621bbSKadir Cetinkaya 
Positionclang::index::__anon37c2cdbb0111::Position34f53621bbSKadir Cetinkaya   Position(size_t Line = 0, size_t Column = 0) : Line(Line), Column(Column) {}
35f53621bbSKadir Cetinkaya 
fromSourceLocationclang::index::__anon37c2cdbb0111::Position36f53621bbSKadir Cetinkaya   static Position fromSourceLocation(SourceLocation Loc,
37f53621bbSKadir Cetinkaya                                      const SourceManager &SM) {
38f53621bbSKadir Cetinkaya     FileID FID;
39f53621bbSKadir Cetinkaya     unsigned Offset;
40f53621bbSKadir Cetinkaya     std::tie(FID, Offset) = SM.getDecomposedSpellingLoc(Loc);
41f53621bbSKadir Cetinkaya     Position P;
42f53621bbSKadir Cetinkaya     P.Line = SM.getLineNumber(FID, Offset);
43f53621bbSKadir Cetinkaya     P.Column = SM.getColumnNumber(FID, Offset);
44f53621bbSKadir Cetinkaya     return P;
45f53621bbSKadir Cetinkaya   }
46f53621bbSKadir Cetinkaya };
47f53621bbSKadir Cetinkaya 
operator ==(const Position & LHS,const Position & RHS)48f53621bbSKadir Cetinkaya bool operator==(const Position &LHS, const Position &RHS) {
49f53621bbSKadir Cetinkaya   return std::tie(LHS.Line, LHS.Column) == std::tie(RHS.Line, RHS.Column);
50f53621bbSKadir Cetinkaya }
51f53621bbSKadir Cetinkaya 
operator <<(llvm::raw_ostream & OS,const Position & Pos)52f53621bbSKadir Cetinkaya llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Position &Pos) {
53f53621bbSKadir Cetinkaya   return OS << Pos.Line << ':' << Pos.Column;
54f53621bbSKadir Cetinkaya }
554d22172bSEric Liu 
564d22172bSEric Liu struct TestSymbol {
574d22172bSEric Liu   std::string QName;
58f53621bbSKadir Cetinkaya   Position WrittenPos;
59f53621bbSKadir Cetinkaya   Position DeclPos;
60a87ada0dSKadir Cetinkaya   SymbolInfo SymInfo;
61e7eb27a9SKadir Cetinkaya   SymbolRoleSet Roles;
624d22172bSEric Liu   // FIXME: add more information.
634d22172bSEric Liu };
644d22172bSEric Liu 
operator <<(llvm::raw_ostream & OS,const TestSymbol & S)654d22172bSEric Liu llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const TestSymbol &S) {
66e7eb27a9SKadir Cetinkaya   return OS << S.QName << '[' << S.WrittenPos << ']' << '@' << S.DeclPos << '('
67e7eb27a9SKadir Cetinkaya             << static_cast<unsigned>(S.SymInfo.Kind) << ')';
684d22172bSEric Liu }
694d22172bSEric Liu 
704d22172bSEric Liu class Indexer : public IndexDataConsumer {
714d22172bSEric Liu public:
initialize(ASTContext & Ctx)72f53621bbSKadir Cetinkaya   void initialize(ASTContext &Ctx) override {
73f53621bbSKadir Cetinkaya     AST = &Ctx;
74f53621bbSKadir Cetinkaya     IndexDataConsumer::initialize(Ctx);
75f53621bbSKadir Cetinkaya   }
76f53621bbSKadir Cetinkaya 
handleDeclOccurrence(const Decl * D,SymbolRoleSet Roles,ArrayRef<SymbolRelation>,SourceLocation Loc,ASTNodeInfo)773b9715cbSKirill Bobyrev   bool handleDeclOccurrence(const Decl *D, SymbolRoleSet Roles,
78f53621bbSKadir Cetinkaya                             ArrayRef<SymbolRelation>, SourceLocation Loc,
794d22172bSEric Liu                             ASTNodeInfo) override {
804d22172bSEric Liu     const auto *ND = llvm::dyn_cast<NamedDecl>(D);
814d22172bSEric Liu     if (!ND)
824d22172bSEric Liu       return true;
834d22172bSEric Liu     TestSymbol S;
84a87ada0dSKadir Cetinkaya     S.SymInfo = getSymbolInfo(D);
854d22172bSEric Liu     S.QName = ND->getQualifiedNameAsString();
86f53621bbSKadir Cetinkaya     S.WrittenPos = Position::fromSourceLocation(Loc, AST->getSourceManager());
87f53621bbSKadir Cetinkaya     S.DeclPos =
88f53621bbSKadir Cetinkaya         Position::fromSourceLocation(D->getLocation(), AST->getSourceManager());
89e7eb27a9SKadir Cetinkaya     S.Roles = Roles;
904d22172bSEric Liu     Symbols.push_back(std::move(S));
914d22172bSEric Liu     return true;
924d22172bSEric Liu   }
934d22172bSEric Liu 
handleMacroOccurrence(const IdentifierInfo * Name,const MacroInfo * MI,SymbolRoleSet Roles,SourceLocation Loc)943b9715cbSKirill Bobyrev   bool handleMacroOccurrence(const IdentifierInfo *Name, const MacroInfo *MI,
95e4407876SKadir Cetinkaya                              SymbolRoleSet Roles, SourceLocation Loc) override {
964d22172bSEric Liu     TestSymbol S;
97e4407876SKadir Cetinkaya     S.SymInfo = getSymbolInfoForMacro(*MI);
98adcd0268SBenjamin Kramer     S.QName = std::string(Name->getName());
99e4407876SKadir Cetinkaya     S.WrittenPos = Position::fromSourceLocation(Loc, AST->getSourceManager());
100e4407876SKadir Cetinkaya     S.DeclPos = Position::fromSourceLocation(MI->getDefinitionLoc(),
101e4407876SKadir Cetinkaya                                              AST->getSourceManager());
102e4407876SKadir Cetinkaya     S.Roles = Roles;
1034d22172bSEric Liu     Symbols.push_back(std::move(S));
1044d22172bSEric Liu     return true;
1054d22172bSEric Liu   }
1064d22172bSEric Liu 
1074d22172bSEric Liu   std::vector<TestSymbol> Symbols;
108f53621bbSKadir Cetinkaya   const ASTContext *AST = nullptr;
1094d22172bSEric Liu };
1104d22172bSEric Liu 
1114d22172bSEric Liu class IndexAction : public ASTFrontendAction {
1124d22172bSEric Liu public:
IndexAction(std::shared_ptr<Indexer> Index,IndexingOptions Opts=IndexingOptions ())1134d22172bSEric Liu   IndexAction(std::shared_ptr<Indexer> Index,
1144d22172bSEric Liu               IndexingOptions Opts = IndexingOptions())
1154d22172bSEric Liu       : Index(std::move(Index)), Opts(Opts) {}
1164d22172bSEric Liu 
1174d22172bSEric Liu protected:
CreateASTConsumer(CompilerInstance & CI,StringRef InFile)1184d22172bSEric Liu   std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI,
1194d22172bSEric Liu                                                  StringRef InFile) override {
1204d22172bSEric Liu     class Consumer : public ASTConsumer {
1214d22172bSEric Liu       std::shared_ptr<Indexer> Index;
1224d22172bSEric Liu       std::shared_ptr<Preprocessor> PP;
1234d22172bSEric Liu       IndexingOptions Opts;
1244d22172bSEric Liu 
1254d22172bSEric Liu     public:
1264d22172bSEric Liu       Consumer(std::shared_ptr<Indexer> Index, std::shared_ptr<Preprocessor> PP,
1274d22172bSEric Liu                IndexingOptions Opts)
1284d22172bSEric Liu           : Index(std::move(Index)), PP(std::move(PP)), Opts(Opts) {}
1294d22172bSEric Liu 
1304d22172bSEric Liu       void HandleTranslationUnit(ASTContext &Ctx) override {
1314d22172bSEric Liu         std::vector<Decl *> DeclsToIndex(
1324d22172bSEric Liu             Ctx.getTranslationUnitDecl()->decls().begin(),
1334d22172bSEric Liu             Ctx.getTranslationUnitDecl()->decls().end());
1344d22172bSEric Liu         indexTopLevelDecls(Ctx, *PP, DeclsToIndex, *Index, Opts);
1354d22172bSEric Liu       }
1364d22172bSEric Liu     };
1372b3d49b6SJonas Devlieghere     return std::make_unique<Consumer>(Index, CI.getPreprocessorPtr(), Opts);
1384d22172bSEric Liu   }
1394d22172bSEric Liu 
1404d22172bSEric Liu private:
1414d22172bSEric Liu   std::shared_ptr<Indexer> Index;
1424d22172bSEric Liu   IndexingOptions Opts;
1434d22172bSEric Liu };
1444d22172bSEric Liu 
145f53621bbSKadir Cetinkaya using testing::AllOf;
1464d22172bSEric Liu using testing::Contains;
1474d22172bSEric Liu using testing::Not;
1484d22172bSEric Liu using testing::UnorderedElementsAre;
1494d22172bSEric Liu 
1504d22172bSEric Liu MATCHER_P(QName, Name, "") { return arg.QName == Name; }
151f53621bbSKadir Cetinkaya MATCHER_P(WrittenAt, Pos, "") { return arg.WrittenPos == Pos; }
152f53621bbSKadir Cetinkaya MATCHER_P(DeclAt, Pos, "") { return arg.DeclPos == Pos; }
153a87ada0dSKadir Cetinkaya MATCHER_P(Kind, SymKind, "") { return arg.SymInfo.Kind == SymKind; }
154e7eb27a9SKadir Cetinkaya MATCHER_P(HasRole, Role, "") { return arg.Roles & static_cast<unsigned>(Role); }
1554d22172bSEric Liu 
TEST(IndexTest,Simple)1564d22172bSEric Liu TEST(IndexTest, Simple) {
1574d22172bSEric Liu   auto Index = std::make_shared<Indexer>();
158b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index),
159b22804b3SDmitri Gribenko                          "class X {}; void f() {}");
1604d22172bSEric Liu   EXPECT_THAT(Index->Symbols, UnorderedElementsAre(QName("X"), QName("f")));
1614d22172bSEric Liu }
1624d22172bSEric Liu 
TEST(IndexTest,IndexPreprocessorMacros)1634d22172bSEric Liu TEST(IndexTest, IndexPreprocessorMacros) {
16493c87fc0SBen Langmuir   std::string Code = R"cpp(
16593c87fc0SBen Langmuir     #define INDEX_MAC 1
16693c87fc0SBen Langmuir     #define INDEX_MAC_UNDEF 1
16793c87fc0SBen Langmuir     #undef INDEX_MAC_UNDEF
16893c87fc0SBen Langmuir     #define INDEX_MAC_REDEF 1
16993c87fc0SBen Langmuir     #undef INDEX_MAC_REDEF
17093c87fc0SBen Langmuir     #define INDEX_MAC_REDEF 2
17193c87fc0SBen Langmuir   )cpp";
1724d22172bSEric Liu   auto Index = std::make_shared<Indexer>();
1734d22172bSEric Liu   IndexingOptions Opts;
1744d22172bSEric Liu   Opts.IndexMacrosInPreprocessor = true;
175b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
17693c87fc0SBen Langmuir   EXPECT_THAT(Index->Symbols,
17793c87fc0SBen Langmuir               Contains(AllOf(QName("INDEX_MAC"), WrittenAt(Position(2, 13)),
17893c87fc0SBen Langmuir                              DeclAt(Position(2, 13)),
17993c87fc0SBen Langmuir                              HasRole(SymbolRole::Definition))));
18093c87fc0SBen Langmuir   EXPECT_THAT(
18193c87fc0SBen Langmuir       Index->Symbols,
18293c87fc0SBen Langmuir       AllOf(Contains(AllOf(QName("INDEX_MAC_UNDEF"), WrittenAt(Position(3, 13)),
18393c87fc0SBen Langmuir                            DeclAt(Position(3, 13)),
18493c87fc0SBen Langmuir                            HasRole(SymbolRole::Definition))),
18593c87fc0SBen Langmuir             Contains(AllOf(QName("INDEX_MAC_UNDEF"), WrittenAt(Position(4, 12)),
18693c87fc0SBen Langmuir                            DeclAt(Position(3, 13)),
18793c87fc0SBen Langmuir                            HasRole(SymbolRole::Undefinition)))));
18893c87fc0SBen Langmuir   EXPECT_THAT(
18993c87fc0SBen Langmuir       Index->Symbols,
19093c87fc0SBen Langmuir       AllOf(Contains(AllOf(QName("INDEX_MAC_REDEF"), WrittenAt(Position(5, 13)),
19193c87fc0SBen Langmuir                            DeclAt(Position(5, 13)),
19293c87fc0SBen Langmuir                            HasRole(SymbolRole::Definition))),
19393c87fc0SBen Langmuir             Contains(AllOf(QName("INDEX_MAC_REDEF"), WrittenAt(Position(6, 12)),
19493c87fc0SBen Langmuir                            DeclAt(Position(5, 13)),
19593c87fc0SBen Langmuir                            HasRole(SymbolRole::Undefinition))),
19693c87fc0SBen Langmuir             Contains(AllOf(QName("INDEX_MAC_REDEF"), WrittenAt(Position(7, 13)),
19793c87fc0SBen Langmuir                            DeclAt(Position(7, 13)),
19893c87fc0SBen Langmuir                            HasRole(SymbolRole::Definition)))));
1994d22172bSEric Liu 
2004d22172bSEric Liu   Opts.IndexMacrosInPreprocessor = false;
2014d22172bSEric Liu   Index->Symbols.clear();
202b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
2034d22172bSEric Liu   EXPECT_THAT(Index->Symbols, UnorderedElementsAre());
2044d22172bSEric Liu }
2054d22172bSEric Liu 
TEST(IndexTest,IndexParametersInDecls)2060468fc0bSKadir Cetinkaya TEST(IndexTest, IndexParametersInDecls) {
2070468fc0bSKadir Cetinkaya   std::string Code = "void foo(int bar);";
2080468fc0bSKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
2090468fc0bSKadir Cetinkaya   IndexingOptions Opts;
2100468fc0bSKadir Cetinkaya   Opts.IndexFunctionLocals = true;
2110468fc0bSKadir Cetinkaya   Opts.IndexParametersInDeclarations = true;
212b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
2130468fc0bSKadir Cetinkaya   EXPECT_THAT(Index->Symbols, Contains(QName("bar")));
2140468fc0bSKadir Cetinkaya 
2150468fc0bSKadir Cetinkaya   Opts.IndexParametersInDeclarations = false;
2160468fc0bSKadir Cetinkaya   Index->Symbols.clear();
217b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
2180468fc0bSKadir Cetinkaya   EXPECT_THAT(Index->Symbols, Not(Contains(QName("bar"))));
2190468fc0bSKadir Cetinkaya }
2200468fc0bSKadir Cetinkaya 
TEST(IndexTest,IndexLabels)221535f34ddSChristian Kandeler TEST(IndexTest, IndexLabels) {
222535f34ddSChristian Kandeler   std::string Code = R"cpp(
223535f34ddSChristian Kandeler         int main() {
224535f34ddSChristian Kandeler           goto theLabel;
225535f34ddSChristian Kandeler           theLabel:
226535f34ddSChristian Kandeler             return 1;
227535f34ddSChristian Kandeler         }
228535f34ddSChristian Kandeler       )cpp";
229535f34ddSChristian Kandeler   auto Index = std::make_shared<Indexer>();
230535f34ddSChristian Kandeler   IndexingOptions Opts;
231535f34ddSChristian Kandeler   Opts.IndexFunctionLocals = true;
232535f34ddSChristian Kandeler   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
233535f34ddSChristian Kandeler   EXPECT_THAT(Index->Symbols,
234535f34ddSChristian Kandeler               Contains(AllOf(QName("theLabel"), WrittenAt(Position(3, 16)),
235535f34ddSChristian Kandeler                              DeclAt(Position(4, 11)))));
236535f34ddSChristian Kandeler 
237535f34ddSChristian Kandeler   Opts.IndexFunctionLocals = false;
238535f34ddSChristian Kandeler   Index->Symbols.clear();
239535f34ddSChristian Kandeler   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
240535f34ddSChristian Kandeler   EXPECT_THAT(Index->Symbols, Not(Contains(QName("theLabel"))));
241535f34ddSChristian Kandeler }
242535f34ddSChristian Kandeler 
TEST(IndexTest,IndexExplicitTemplateInstantiation)243f53621bbSKadir Cetinkaya TEST(IndexTest, IndexExplicitTemplateInstantiation) {
244f53621bbSKadir Cetinkaya   std::string Code = R"cpp(
245f53621bbSKadir Cetinkaya     template <typename T>
246f53621bbSKadir Cetinkaya     struct Foo { void bar() {} };
247f53621bbSKadir Cetinkaya     template <>
248f53621bbSKadir Cetinkaya     struct Foo<int> { void bar() {} };
249f53621bbSKadir Cetinkaya     void foo() {
250f53621bbSKadir Cetinkaya       Foo<char> abc;
251f53621bbSKadir Cetinkaya       Foo<int> b;
252f53621bbSKadir Cetinkaya     }
253f53621bbSKadir Cetinkaya   )cpp";
254f53621bbSKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
255f53621bbSKadir Cetinkaya   IndexingOptions Opts;
256b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
257f53621bbSKadir Cetinkaya   EXPECT_THAT(Index->Symbols,
258f53621bbSKadir Cetinkaya               AllOf(Contains(AllOf(QName("Foo"), WrittenAt(Position(8, 7)),
259f53621bbSKadir Cetinkaya                                    DeclAt(Position(5, 12)))),
260f53621bbSKadir Cetinkaya                     Contains(AllOf(QName("Foo"), WrittenAt(Position(7, 7)),
261f53621bbSKadir Cetinkaya                                    DeclAt(Position(3, 12))))));
262f53621bbSKadir Cetinkaya }
263f53621bbSKadir Cetinkaya 
TEST(IndexTest,IndexTemplateInstantiationPartial)264f53621bbSKadir Cetinkaya TEST(IndexTest, IndexTemplateInstantiationPartial) {
265f53621bbSKadir Cetinkaya   std::string Code = R"cpp(
266f53621bbSKadir Cetinkaya     template <typename T1, typename T2>
267f53621bbSKadir Cetinkaya     struct Foo { void bar() {} };
268f53621bbSKadir Cetinkaya     template <typename T>
269f53621bbSKadir Cetinkaya     struct Foo<T, int> { void bar() {} };
270f53621bbSKadir Cetinkaya     void foo() {
271f53621bbSKadir Cetinkaya       Foo<char, char> abc;
272f53621bbSKadir Cetinkaya       Foo<int, int> b;
273f53621bbSKadir Cetinkaya     }
274f53621bbSKadir Cetinkaya   )cpp";
275f53621bbSKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
276f53621bbSKadir Cetinkaya   IndexingOptions Opts;
277b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
278f53621bbSKadir Cetinkaya   EXPECT_THAT(Index->Symbols,
279f53621bbSKadir Cetinkaya               Contains(AllOf(QName("Foo"), WrittenAt(Position(8, 7)),
280f53621bbSKadir Cetinkaya                              DeclAt(Position(5, 12)))));
281f53621bbSKadir Cetinkaya }
282f53621bbSKadir Cetinkaya 
TEST(IndexTest,IndexTypeParmDecls)283b780517cSKadir Cetinkaya TEST(IndexTest, IndexTypeParmDecls) {
284b780517cSKadir Cetinkaya   std::string Code = R"cpp(
285b780517cSKadir Cetinkaya     template <typename T, int I, template<typename> class C, typename NoRef>
286b780517cSKadir Cetinkaya     struct Foo {
287b780517cSKadir Cetinkaya       T t = I;
288b780517cSKadir Cetinkaya       C<int> x;
289b780517cSKadir Cetinkaya     };
290b780517cSKadir Cetinkaya   )cpp";
291b780517cSKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
292b780517cSKadir Cetinkaya   IndexingOptions Opts;
293b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
294b780517cSKadir Cetinkaya   EXPECT_THAT(Index->Symbols, AllOf(Not(Contains(QName("Foo::T"))),
295b780517cSKadir Cetinkaya                                     Not(Contains(QName("Foo::I"))),
296b780517cSKadir Cetinkaya                                     Not(Contains(QName("Foo::C"))),
297b780517cSKadir Cetinkaya                                     Not(Contains(QName("Foo::NoRef")))));
298b780517cSKadir Cetinkaya 
299b780517cSKadir Cetinkaya   Opts.IndexTemplateParameters = true;
300b780517cSKadir Cetinkaya   Index->Symbols.clear();
301b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
302b780517cSKadir Cetinkaya   EXPECT_THAT(Index->Symbols,
30384240e0dSKadir Cetinkaya               AllOf(Contains(AllOf(QName("Foo::T"),
30484240e0dSKadir Cetinkaya                                    Kind(SymbolKind::TemplateTypeParm))),
30584240e0dSKadir Cetinkaya                     Contains(AllOf(QName("Foo::I"),
30684240e0dSKadir Cetinkaya                                    Kind(SymbolKind::NonTypeTemplateParm))),
30784240e0dSKadir Cetinkaya                     Contains(AllOf(QName("Foo::C"),
30884240e0dSKadir Cetinkaya                                    Kind(SymbolKind::TemplateTemplateParm))),
30984240e0dSKadir Cetinkaya                     Contains(QName("Foo::NoRef"))));
310b780517cSKadir Cetinkaya }
311b780517cSKadir Cetinkaya 
TEST(IndexTest,UsingDecls)312a87ada0dSKadir Cetinkaya TEST(IndexTest, UsingDecls) {
313a87ada0dSKadir Cetinkaya   std::string Code = R"cpp(
314a87ada0dSKadir Cetinkaya     void foo(int bar);
315a87ada0dSKadir Cetinkaya     namespace std {
316a87ada0dSKadir Cetinkaya       using ::foo;
317a87ada0dSKadir Cetinkaya     }
318a87ada0dSKadir Cetinkaya   )cpp";
319a87ada0dSKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
320a87ada0dSKadir Cetinkaya   IndexingOptions Opts;
321b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
322a87ada0dSKadir Cetinkaya   EXPECT_THAT(Index->Symbols,
323a87ada0dSKadir Cetinkaya               Contains(AllOf(QName("std::foo"), Kind(SymbolKind::Using))));
324a87ada0dSKadir Cetinkaya }
325a87ada0dSKadir Cetinkaya 
TEST(IndexTest,Constructors)326e7eb27a9SKadir Cetinkaya TEST(IndexTest, Constructors) {
327e7eb27a9SKadir Cetinkaya   std::string Code = R"cpp(
328e7eb27a9SKadir Cetinkaya     struct Foo {
329e7eb27a9SKadir Cetinkaya       Foo(int);
330e7eb27a9SKadir Cetinkaya       ~Foo();
331e7eb27a9SKadir Cetinkaya     };
332e7eb27a9SKadir Cetinkaya   )cpp";
333e7eb27a9SKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
334e7eb27a9SKadir Cetinkaya   IndexingOptions Opts;
335b22804b3SDmitri Gribenko   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
336e7eb27a9SKadir Cetinkaya   EXPECT_THAT(
337e7eb27a9SKadir Cetinkaya       Index->Symbols,
338e7eb27a9SKadir Cetinkaya       UnorderedElementsAre(
339e7eb27a9SKadir Cetinkaya           AllOf(QName("Foo"), Kind(SymbolKind::Struct),
340e7eb27a9SKadir Cetinkaya                 WrittenAt(Position(2, 12))),
341e7eb27a9SKadir Cetinkaya           AllOf(QName("Foo::Foo"), Kind(SymbolKind::Constructor),
342e7eb27a9SKadir Cetinkaya                 WrittenAt(Position(3, 7))),
343e7eb27a9SKadir Cetinkaya           AllOf(QName("Foo"), Kind(SymbolKind::Struct),
344e7eb27a9SKadir Cetinkaya                 HasRole(SymbolRole::NameReference), WrittenAt(Position(3, 7))),
345e7eb27a9SKadir Cetinkaya           AllOf(QName("Foo::~Foo"), Kind(SymbolKind::Destructor),
346e7eb27a9SKadir Cetinkaya                 WrittenAt(Position(4, 7))),
347e7eb27a9SKadir Cetinkaya           AllOf(QName("Foo"), Kind(SymbolKind::Struct),
348e7eb27a9SKadir Cetinkaya                 HasRole(SymbolRole::NameReference),
349e7eb27a9SKadir Cetinkaya                 WrittenAt(Position(4, 8)))));
350e7eb27a9SKadir Cetinkaya }
351e7eb27a9SKadir Cetinkaya 
TEST(IndexTest,InjecatedNameClass)352a133cbaaSHaojian Wu TEST(IndexTest, InjecatedNameClass) {
353a133cbaaSHaojian Wu   std::string Code = R"cpp(
354a133cbaaSHaojian Wu     template <typename T>
355a133cbaaSHaojian Wu     class Foo {
356a133cbaaSHaojian Wu       void f(Foo x);
357a133cbaaSHaojian Wu     };
358a133cbaaSHaojian Wu   )cpp";
359a133cbaaSHaojian Wu   auto Index = std::make_shared<Indexer>();
360a133cbaaSHaojian Wu   IndexingOptions Opts;
361a133cbaaSHaojian Wu   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
362a133cbaaSHaojian Wu   EXPECT_THAT(Index->Symbols,
363a133cbaaSHaojian Wu               UnorderedElementsAre(AllOf(QName("Foo"), Kind(SymbolKind::Class),
364a133cbaaSHaojian Wu                                          WrittenAt(Position(3, 11))),
365a133cbaaSHaojian Wu                                    AllOf(QName("Foo::f"),
366a133cbaaSHaojian Wu                                          Kind(SymbolKind::InstanceMethod),
367a133cbaaSHaojian Wu                                          WrittenAt(Position(4, 12))),
368a133cbaaSHaojian Wu                                    AllOf(QName("Foo"), Kind(SymbolKind::Class),
369a133cbaaSHaojian Wu                                          HasRole(SymbolRole::Reference),
370a133cbaaSHaojian Wu                                          WrittenAt(Position(4, 14)))));
371a133cbaaSHaojian Wu }
372a133cbaaSHaojian Wu 
TEST(IndexTest,VisitDefaultArgs)373af8b0cd5SHaojian Wu TEST(IndexTest, VisitDefaultArgs) {
374af8b0cd5SHaojian Wu   std::string Code = R"cpp(
375af8b0cd5SHaojian Wu     int var = 0;
376af8b0cd5SHaojian Wu     void f(int s = var) {}
377af8b0cd5SHaojian Wu   )cpp";
378af8b0cd5SHaojian Wu   auto Index = std::make_shared<Indexer>();
379af8b0cd5SHaojian Wu   IndexingOptions Opts;
380af8b0cd5SHaojian Wu   Opts.IndexFunctionLocals = true;
381af8b0cd5SHaojian Wu   Opts.IndexParametersInDeclarations = true;
382af8b0cd5SHaojian Wu   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
383af8b0cd5SHaojian Wu   EXPECT_THAT(Index->Symbols,
384af8b0cd5SHaojian Wu               Contains(AllOf(QName("var"), HasRole(SymbolRole::Reference),
385af8b0cd5SHaojian Wu                              WrittenAt(Position(3, 20)))));
386af8b0cd5SHaojian Wu }
387af8b0cd5SHaojian Wu 
TEST(IndexTest,RelationBaseOf)3887cd6b0c3SNathan Ridge TEST(IndexTest, RelationBaseOf) {
3897cd6b0c3SNathan Ridge   std::string Code = R"cpp(
3907cd6b0c3SNathan Ridge     class A {};
3917cd6b0c3SNathan Ridge     template <typename> class B {};
3927cd6b0c3SNathan Ridge     class C : B<A> {};
3937cd6b0c3SNathan Ridge   )cpp";
3947cd6b0c3SNathan Ridge   auto Index = std::make_shared<Indexer>();
3957cd6b0c3SNathan Ridge   tooling::runToolOnCode(std::make_unique<IndexAction>(Index), Code);
3967cd6b0c3SNathan Ridge   // A should not be the base of anything.
3977cd6b0c3SNathan Ridge   EXPECT_THAT(Index->Symbols,
3987cd6b0c3SNathan Ridge               Contains(AllOf(QName("A"), HasRole(SymbolRole::Reference),
3997cd6b0c3SNathan Ridge                              Not(HasRole(SymbolRole::RelationBaseOf)))));
4007cd6b0c3SNathan Ridge }
4017cd6b0c3SNathan Ridge 
TEST(IndexTest,EnumBase)4029ab9caf2SKadir Cetinkaya TEST(IndexTest, EnumBase) {
4039ab9caf2SKadir Cetinkaya   std::string Code = R"cpp(
4049ab9caf2SKadir Cetinkaya     typedef int MyTypedef;
4059ab9caf2SKadir Cetinkaya     enum Foo : MyTypedef;
4069ab9caf2SKadir Cetinkaya     enum Foo : MyTypedef {};
4079ab9caf2SKadir Cetinkaya   )cpp";
4089ab9caf2SKadir Cetinkaya   auto Index = std::make_shared<Indexer>();
4099ab9caf2SKadir Cetinkaya   tooling::runToolOnCode(std::make_unique<IndexAction>(Index), Code);
4109ab9caf2SKadir Cetinkaya   EXPECT_THAT(
4119ab9caf2SKadir Cetinkaya       Index->Symbols,
4129ab9caf2SKadir Cetinkaya       AllOf(Contains(AllOf(QName("MyTypedef"), HasRole(SymbolRole::Reference),
4139ab9caf2SKadir Cetinkaya                            WrittenAt(Position(3, 16)))),
4149ab9caf2SKadir Cetinkaya             Contains(AllOf(QName("MyTypedef"), HasRole(SymbolRole::Reference),
4159ab9caf2SKadir Cetinkaya                            WrittenAt(Position(4, 16))))));
4169ab9caf2SKadir Cetinkaya }
417c3075023SNathan Ridge 
TEST(IndexTest,NonTypeTemplateParameter)418c3075023SNathan Ridge TEST(IndexTest, NonTypeTemplateParameter) {
419c3075023SNathan Ridge   std::string Code = R"cpp(
420c3075023SNathan Ridge     enum class Foobar { foo };
421c3075023SNathan Ridge     template <Foobar f>
422c3075023SNathan Ridge     constexpr void func() {}
423c3075023SNathan Ridge   )cpp";
424c3075023SNathan Ridge   auto Index = std::make_shared<Indexer>();
425c3075023SNathan Ridge   tooling::runToolOnCode(std::make_unique<IndexAction>(Index), Code);
426c3075023SNathan Ridge   EXPECT_THAT(Index->Symbols,
427c3075023SNathan Ridge               Contains(AllOf(QName("Foobar"), HasRole(SymbolRole::Reference),
428c3075023SNathan Ridge                              WrittenAt(Position(3, 15)))));
429c3075023SNathan Ridge }
430c3075023SNathan Ridge 
TEST(IndexTest,ReadWriteRoles)431*51094545SAleksandr Platonov TEST(IndexTest, ReadWriteRoles) {
432*51094545SAleksandr Platonov   std::string Code = R"cpp(
433*51094545SAleksandr Platonov     int main() {
434*51094545SAleksandr Platonov       int foo = 0;
435*51094545SAleksandr Platonov       foo = 2;
436*51094545SAleksandr Platonov       foo += 1;
437*51094545SAleksandr Platonov       int bar = foo;
438*51094545SAleksandr Platonov   }
439*51094545SAleksandr Platonov   )cpp";
440*51094545SAleksandr Platonov   auto Index = std::make_shared<Indexer>();
441*51094545SAleksandr Platonov   IndexingOptions Opts;
442*51094545SAleksandr Platonov   Opts.IndexFunctionLocals = true;
443*51094545SAleksandr Platonov   tooling::runToolOnCode(std::make_unique<IndexAction>(Index, Opts), Code);
444*51094545SAleksandr Platonov   EXPECT_THAT(
445*51094545SAleksandr Platonov       Index->Symbols,
446*51094545SAleksandr Platonov       AllOf(Contains(AllOf(QName("foo"), HasRole(SymbolRole::Write),
447*51094545SAleksandr Platonov                            WrittenAt(Position(4, 7)))),
448*51094545SAleksandr Platonov             Contains(AllOf(QName("foo"),
449*51094545SAleksandr Platonov                            HasRole(static_cast<unsigned>(SymbolRole::Read) |
450*51094545SAleksandr Platonov                                    static_cast<unsigned>(SymbolRole::Write)),
451*51094545SAleksandr Platonov                            WrittenAt(Position(5, 7)))),
452*51094545SAleksandr Platonov             Contains(AllOf(QName("foo"), HasRole(SymbolRole::Read),
453*51094545SAleksandr Platonov                            WrittenAt(Position(6, 17))))));
454*51094545SAleksandr Platonov }
455*51094545SAleksandr Platonov 
4564d22172bSEric Liu } // namespace
4574d22172bSEric Liu } // namespace index
4584d22172bSEric Liu } // namespace clang
459