blob: e435ea92e235c4759e4262895a4b38f3205e4d92 [file] [log] [blame]
Manuel Klimekcb971c62012-04-04 12:07:46 +00001//===- unittest/Tooling/ToolingTest.cpp - Tooling unit tests --------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "clang/AST/ASTConsumer.h"
11#include "clang/AST/DeclCXX.h"
12#include "clang/AST/DeclGroup.h"
Richard Smithd1bac8d2012-11-27 21:31:01 +000013#include "clang/Frontend/CompilerInstance.h"
Manuel Klimekcb971c62012-04-04 12:07:46 +000014#include "clang/Frontend/FrontendAction.h"
15#include "clang/Frontend/FrontendActions.h"
16#include "clang/Tooling/CompilationDatabase.h"
17#include "clang/Tooling/Tooling.h"
18#include "gtest/gtest.h"
Alexander Kornienko5f60f602012-06-01 14:50:43 +000019#include <string>
Manuel Klimekcb971c62012-04-04 12:07:46 +000020
21namespace clang {
22namespace tooling {
23
24namespace {
25/// Takes an ast consumer and returns it from CreateASTConsumer. This only
26/// works with single translation unit compilations.
27class TestAction : public clang::ASTFrontendAction {
28 public:
29 /// Takes ownership of TestConsumer.
30 explicit TestAction(clang::ASTConsumer *TestConsumer)
31 : TestConsumer(TestConsumer) {}
32
33 protected:
34 virtual clang::ASTConsumer* CreateASTConsumer(
35 clang::CompilerInstance& compiler, StringRef dummy) {
36 /// TestConsumer will be deleted by the framework calling us.
37 return TestConsumer;
38 }
39
40 private:
41 clang::ASTConsumer * const TestConsumer;
42};
43
44class FindTopLevelDeclConsumer : public clang::ASTConsumer {
45 public:
46 explicit FindTopLevelDeclConsumer(bool *FoundTopLevelDecl)
47 : FoundTopLevelDecl(FoundTopLevelDecl) {}
48 virtual bool HandleTopLevelDecl(clang::DeclGroupRef DeclGroup) {
49 *FoundTopLevelDecl = true;
50 return true;
51 }
52 private:
53 bool * const FoundTopLevelDecl;
54};
55} // end namespace
56
Meador Ingec5613b22012-06-16 03:34:49 +000057TEST(runToolOnCode, FindsNoTopLevelDeclOnEmptyCode) {
Manuel Klimekcb971c62012-04-04 12:07:46 +000058 bool FoundTopLevelDecl = false;
59 EXPECT_TRUE(runToolOnCode(
60 new TestAction(new FindTopLevelDeclConsumer(&FoundTopLevelDecl)), ""));
NAKAMURA Takumi8f190372012-06-16 06:04:05 +000061#if !defined(_MSC_VER)
Meador Ingec5613b22012-06-16 03:34:49 +000062 EXPECT_FALSE(FoundTopLevelDecl);
NAKAMURA Takumi8f190372012-06-16 06:04:05 +000063#else
64 // FIXME: LangOpts.MicrosoftExt appends "class type_info;"
65 EXPECT_TRUE(FoundTopLevelDecl);
66#endif
Manuel Klimekcb971c62012-04-04 12:07:46 +000067}
68
69namespace {
70class FindClassDeclXConsumer : public clang::ASTConsumer {
71 public:
72 FindClassDeclXConsumer(bool *FoundClassDeclX)
73 : FoundClassDeclX(FoundClassDeclX) {}
74 virtual bool HandleTopLevelDecl(clang::DeclGroupRef GroupRef) {
75 if (CXXRecordDecl* Record = dyn_cast<clang::CXXRecordDecl>(
76 *GroupRef.begin())) {
77 if (Record->getName() == "X") {
78 *FoundClassDeclX = true;
79 }
80 }
81 return true;
82 }
83 private:
84 bool *FoundClassDeclX;
85};
86} // end namespace
87
88TEST(runToolOnCode, FindsClassDecl) {
89 bool FoundClassDeclX = false;
90 EXPECT_TRUE(runToolOnCode(new TestAction(
91 new FindClassDeclXConsumer(&FoundClassDeclX)), "class X;"));
92 EXPECT_TRUE(FoundClassDeclX);
93
94 FoundClassDeclX = false;
95 EXPECT_TRUE(runToolOnCode(new TestAction(
96 new FindClassDeclXConsumer(&FoundClassDeclX)), "class Y;"));
97 EXPECT_FALSE(FoundClassDeclX);
98}
99
100TEST(newFrontendActionFactory, CreatesFrontendActionFactoryFromType) {
Dmitri Gribenkocfa88f82013-01-12 19:30:44 +0000101 OwningPtr<FrontendActionFactory> Factory(
102 newFrontendActionFactory<SyntaxOnlyAction>());
103 OwningPtr<FrontendAction> Action(Factory->create());
Manuel Klimekcb971c62012-04-04 12:07:46 +0000104 EXPECT_TRUE(Action.get() != NULL);
105}
106
107struct IndependentFrontendActionCreator {
Manuel Klimeke6df0ce2012-07-05 18:13:01 +0000108 ASTConsumer *newASTConsumer() {
109 return new FindTopLevelDeclConsumer(NULL);
110 }
Manuel Klimekcb971c62012-04-04 12:07:46 +0000111};
112
113TEST(newFrontendActionFactory, CreatesFrontendActionFactoryFromFactoryType) {
114 IndependentFrontendActionCreator Creator;
Dmitri Gribenkocfa88f82013-01-12 19:30:44 +0000115 OwningPtr<FrontendActionFactory> Factory(
116 newFrontendActionFactory(&Creator));
117 OwningPtr<FrontendAction> Action(Factory->create());
Manuel Klimekcb971c62012-04-04 12:07:46 +0000118 EXPECT_TRUE(Action.get() != NULL);
119}
120
Alexander Kornienko5f60f602012-06-01 14:50:43 +0000121TEST(ToolInvocation, TestMapVirtualFile) {
122 clang::FileManager Files((clang::FileSystemOptions()));
123 std::vector<std::string> Args;
124 Args.push_back("tool-executable");
125 Args.push_back("-Idef");
126 Args.push_back("-fsyntax-only");
127 Args.push_back("test.cpp");
128 clang::tooling::ToolInvocation Invocation(Args, new SyntaxOnlyAction, &Files);
129 Invocation.mapVirtualFile("test.cpp", "#include <abc>\n");
130 Invocation.mapVirtualFile("def/abc", "\n");
131 EXPECT_TRUE(Invocation.run());
132}
133
Edwin Vane3c16e692013-05-29 16:01:10 +0000134struct VerifyEndCallback : public SourceFileCallbacks {
135 VerifyEndCallback() : BeginCalled(0), EndCalled(0), Matched(false) {}
Edwin Vane5ec95802013-05-30 13:59:44 +0000136 virtual bool handleBeginSource(CompilerInstance &CI,
137 StringRef Filename) LLVM_OVERRIDE {
Edwin Vane3c16e692013-05-29 16:01:10 +0000138 ++BeginCalled;
139 return true;
140 }
Edwin Vane5ec95802013-05-30 13:59:44 +0000141 virtual void handleEndSource() {
Edwin Vane3c16e692013-05-29 16:01:10 +0000142 ++EndCalled;
Manuel Klimek9fb6b272012-10-25 08:49:11 +0000143 }
144 ASTConsumer *newASTConsumer() {
145 return new FindTopLevelDeclConsumer(&Matched);
146 }
Edwin Vane3c16e692013-05-29 16:01:10 +0000147 unsigned BeginCalled;
148 unsigned EndCalled;
Manuel Klimek9fb6b272012-10-25 08:49:11 +0000149 bool Matched;
150};
151
NAKAMURA Takumid9a2d5b2012-10-25 09:38:41 +0000152#if !defined(_WIN32)
Edwin Vane3c16e692013-05-29 16:01:10 +0000153TEST(newFrontendActionFactory, InjectsSourceFileCallbacks) {
Manuel Klimek9fb6b272012-10-25 08:49:11 +0000154 VerifyEndCallback EndCallback;
155
156 FixedCompilationDatabase Compilations("/", std::vector<std::string>());
157 std::vector<std::string> Sources;
158 Sources.push_back("/a.cc");
159 Sources.push_back("/b.cc");
160 ClangTool Tool(Compilations, Sources);
161
162 Tool.mapVirtualFile("/a.cc", "void a() {}");
163 Tool.mapVirtualFile("/b.cc", "void b() {}");
164
165 Tool.run(newFrontendActionFactory(&EndCallback, &EndCallback));
166
167 EXPECT_TRUE(EndCallback.Matched);
Edwin Vane3c16e692013-05-29 16:01:10 +0000168 EXPECT_EQ(2u, EndCallback.BeginCalled);
169 EXPECT_EQ(2u, EndCallback.EndCalled);
Manuel Klimek9fb6b272012-10-25 08:49:11 +0000170}
NAKAMURA Takumid9a2d5b2012-10-25 09:38:41 +0000171#endif
Manuel Klimek9fb6b272012-10-25 08:49:11 +0000172
Richard Smithd1bac8d2012-11-27 21:31:01 +0000173struct SkipBodyConsumer : public clang::ASTConsumer {
174 /// Skip the 'skipMe' function.
175 virtual bool shouldSkipFunctionBody(Decl *D) {
176 FunctionDecl *F = dyn_cast<FunctionDecl>(D);
177 return F && F->getNameAsString() == "skipMe";
178 }
179};
180
181struct SkipBodyAction : public clang::ASTFrontendAction {
182 virtual ASTConsumer *CreateASTConsumer(CompilerInstance &Compiler,
183 StringRef) {
184 Compiler.getFrontendOpts().SkipFunctionBodies = true;
185 return new SkipBodyConsumer;
186 }
187};
188
Hal Finkelb8d24412013-01-28 04:37:38 +0000189TEST(runToolOnCode, TestSkipFunctionBody) {
Richard Smithd1bac8d2012-11-27 21:31:01 +0000190 EXPECT_TRUE(runToolOnCode(new SkipBodyAction,
191 "int skipMe() { an_error_here }"));
192 EXPECT_FALSE(runToolOnCode(new SkipBodyAction,
193 "int skipMeNot() { an_error_here }"));
194}
195
Manuel Klimek48b3f0f2013-06-04 14:44:44 +0000196struct CheckSyntaxOnlyAdjuster: public ArgumentsAdjuster {
197 bool &Found;
198 bool &Ran;
199
200 CheckSyntaxOnlyAdjuster(bool &Found, bool &Ran) : Found(Found), Ran(Ran) { }
201
202 virtual CommandLineArguments
203 Adjust(const CommandLineArguments &Args) LLVM_OVERRIDE {
204 Ran = true;
205 for (unsigned I = 0, E = Args.size(); I != E; ++I) {
206 if (Args[I] == "-fsyntax-only") {
207 Found = true;
208 break;
209 }
210 }
211 return Args;
212 }
213};
214
215TEST(ClangToolTest, ArgumentAdjusters) {
216 FixedCompilationDatabase Compilations("/", std::vector<std::string>());
217
218 ClangTool Tool(Compilations, std::vector<std::string>(1, "/a.cc"));
219 Tool.mapVirtualFile("/a.cc", "void a() {}");
220
221 bool Found = false;
222 bool Ran = false;
223 Tool.appendArgumentsAdjuster(new CheckSyntaxOnlyAdjuster(Found, Ran));
224 Tool.run(newFrontendActionFactory<SyntaxOnlyAction>());
225 EXPECT_TRUE(Ran);
226 EXPECT_TRUE(Found);
227
228 Ran = Found = false;
229 Tool.clearArgumentsAdjusters();
230 Tool.appendArgumentsAdjuster(new CheckSyntaxOnlyAdjuster(Found, Ran));
231 Tool.appendArgumentsAdjuster(new ClangSyntaxOnlyAdjuster());
232 Tool.run(newFrontendActionFactory<SyntaxOnlyAction>());
233 EXPECT_TRUE(Ran);
234 EXPECT_FALSE(Found);
235}
236
Manuel Klimekcb971c62012-04-04 12:07:46 +0000237} // end namespace tooling
238} // end namespace clang