| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346 |
- //===- unittest/Tooling/ToolingTest.cpp - Tooling unit tests --------------===//
- //
- // The LLVM Compiler Infrastructure
- //
- // This file is distributed under the University of Illinois Open Source
- // License. See LICENSE.TXT for details.
- //
- //===----------------------------------------------------------------------===//
- #include "clang/AST/ASTConsumer.h"
- #include "clang/AST/DeclCXX.h"
- #include "clang/AST/DeclGroup.h"
- #include "clang/Frontend/ASTUnit.h"
- #include "clang/Frontend/CompilerInstance.h"
- #include "clang/Frontend/FrontendAction.h"
- #include "clang/Frontend/FrontendActions.h"
- #include "clang/Tooling/CompilationDatabase.h"
- #include "clang/Tooling/Tooling.h"
- #include "llvm/ADT/STLExtras.h"
- #include "llvm/Config/llvm-config.h"
- #include "gtest/gtest.h"
- #include <algorithm>
- #include <string>
- namespace clang {
- namespace tooling {
- namespace {
- /// Takes an ast consumer and returns it from CreateASTConsumer. This only
- /// works with single translation unit compilations.
- class TestAction : public clang::ASTFrontendAction {
- public:
- /// Takes ownership of TestConsumer.
- explicit TestAction(std::unique_ptr<clang::ASTConsumer> TestConsumer)
- : TestConsumer(std::move(TestConsumer)) {}
- protected:
- std::unique_ptr<clang::ASTConsumer>
- CreateASTConsumer(clang::CompilerInstance &compiler,
- StringRef dummy) override {
- /// TestConsumer will be deleted by the framework calling us.
- return std::move(TestConsumer);
- }
- private:
- std::unique_ptr<clang::ASTConsumer> TestConsumer;
- };
- class FindTopLevelDeclConsumer : public clang::ASTConsumer {
- public:
- explicit FindTopLevelDeclConsumer(bool *FoundTopLevelDecl)
- : FoundTopLevelDecl(FoundTopLevelDecl) {}
- bool HandleTopLevelDecl(clang::DeclGroupRef DeclGroup) override {
- *FoundTopLevelDecl = true;
- return true;
- }
- private:
- bool * const FoundTopLevelDecl;
- };
- } // end namespace
- TEST(runToolOnCode, FindsNoTopLevelDeclOnEmptyCode) {
- bool FoundTopLevelDecl = false;
- EXPECT_TRUE(
- runToolOnCode(new TestAction(llvm::make_unique<FindTopLevelDeclConsumer>(
- &FoundTopLevelDecl)),
- ""));
- EXPECT_FALSE(FoundTopLevelDecl);
- }
- namespace {
- class FindClassDeclXConsumer : public clang::ASTConsumer {
- public:
- FindClassDeclXConsumer(bool *FoundClassDeclX)
- : FoundClassDeclX(FoundClassDeclX) {}
- bool HandleTopLevelDecl(clang::DeclGroupRef GroupRef) override {
- if (CXXRecordDecl* Record = dyn_cast<clang::CXXRecordDecl>(
- *GroupRef.begin())) {
- if (Record->getName() == "X") {
- *FoundClassDeclX = true;
- }
- }
- return true;
- }
- private:
- bool *FoundClassDeclX;
- };
- bool FindClassDeclX(ASTUnit *AST) {
- for (std::vector<Decl *>::iterator i = AST->top_level_begin(),
- e = AST->top_level_end();
- i != e; ++i) {
- if (CXXRecordDecl* Record = dyn_cast<clang::CXXRecordDecl>(*i)) {
- if (Record->getName() == "X") {
- return true;
- }
- }
- }
- return false;
- }
- } // end namespace
- TEST(runToolOnCode, FindsClassDecl) {
- bool FoundClassDeclX = false;
- EXPECT_TRUE(
- runToolOnCode(new TestAction(llvm::make_unique<FindClassDeclXConsumer>(
- &FoundClassDeclX)),
- "class X;"));
- EXPECT_TRUE(FoundClassDeclX);
- FoundClassDeclX = false;
- EXPECT_TRUE(
- runToolOnCode(new TestAction(llvm::make_unique<FindClassDeclXConsumer>(
- &FoundClassDeclX)),
- "class Y;"));
- EXPECT_FALSE(FoundClassDeclX);
- }
- TEST(buildASTFromCode, FindsClassDecl) {
- std::unique_ptr<ASTUnit> AST = buildASTFromCode("class X;");
- ASSERT_TRUE(AST.get());
- EXPECT_TRUE(FindClassDeclX(AST.get()));
- AST = buildASTFromCode("class Y;");
- ASSERT_TRUE(AST.get());
- EXPECT_FALSE(FindClassDeclX(AST.get()));
- }
- TEST(newFrontendActionFactory, CreatesFrontendActionFactoryFromType) {
- std::unique_ptr<FrontendActionFactory> Factory(
- newFrontendActionFactory<SyntaxOnlyAction>());
- std::unique_ptr<FrontendAction> Action(Factory->create());
- EXPECT_TRUE(Action.get() != nullptr);
- }
- struct IndependentFrontendActionCreator {
- std::unique_ptr<ASTConsumer> newASTConsumer() {
- return llvm::make_unique<FindTopLevelDeclConsumer>(nullptr);
- }
- };
- TEST(newFrontendActionFactory, CreatesFrontendActionFactoryFromFactoryType) {
- IndependentFrontendActionCreator Creator;
- std::unique_ptr<FrontendActionFactory> Factory(
- newFrontendActionFactory(&Creator));
- std::unique_ptr<FrontendAction> Action(Factory->create());
- EXPECT_TRUE(Action.get() != nullptr);
- }
- TEST(ToolInvocation, TestMapVirtualFile) {
- IntrusiveRefCntPtr<clang::FileManager> Files(
- new clang::FileManager(clang::FileSystemOptions()));
- std::vector<std::string> Args;
- Args.push_back("tool-executable");
- Args.push_back("-Idef");
- Args.push_back("-fsyntax-only");
- Args.push_back("test.cpp");
- clang::tooling::ToolInvocation Invocation(Args, new SyntaxOnlyAction,
- Files.get());
- Invocation.mapVirtualFile("test.cpp", "#include <abc>\n");
- Invocation.mapVirtualFile("def/abc", "\n");
- EXPECT_TRUE(Invocation.run());
- }
- TEST(ToolInvocation, TestVirtualModulesCompilation) {
- // FIXME: Currently, this only tests that we don't exit with an error if a
- // mapped module.map is found on the include path. In the future, expand this
- // test to run a full modules enabled compilation, so we make sure we can
- // rerun modules compilations with a virtual file system.
- IntrusiveRefCntPtr<clang::FileManager> Files(
- new clang::FileManager(clang::FileSystemOptions()));
- std::vector<std::string> Args;
- Args.push_back("tool-executable");
- Args.push_back("-Idef");
- Args.push_back("-fsyntax-only");
- Args.push_back("test.cpp");
- clang::tooling::ToolInvocation Invocation(Args, new SyntaxOnlyAction,
- Files.get());
- Invocation.mapVirtualFile("test.cpp", "#include <abc>\n");
- Invocation.mapVirtualFile("def/abc", "\n");
- // Add a module.map file in the include directory of our header, so we trigger
- // the module.map header search logic.
- Invocation.mapVirtualFile("def/module.map", "\n");
- EXPECT_TRUE(Invocation.run());
- }
- struct VerifyEndCallback : public SourceFileCallbacks {
- VerifyEndCallback() : BeginCalled(0), EndCalled(0), Matched(false) {}
- bool handleBeginSource(CompilerInstance &CI, StringRef Filename) override {
- ++BeginCalled;
- return true;
- }
- void handleEndSource() override { ++EndCalled; }
- std::unique_ptr<ASTConsumer> newASTConsumer() {
- return llvm::make_unique<FindTopLevelDeclConsumer>(&Matched);
- }
- unsigned BeginCalled;
- unsigned EndCalled;
- bool Matched;
- };
- #if !defined(LLVM_ON_WIN32)
- TEST(newFrontendActionFactory, InjectsSourceFileCallbacks) {
- VerifyEndCallback EndCallback;
- FixedCompilationDatabase Compilations("/", std::vector<std::string>());
- std::vector<std::string> Sources;
- Sources.push_back("/a.cc");
- Sources.push_back("/b.cc");
- ClangTool Tool(Compilations, Sources);
- Tool.mapVirtualFile("/a.cc", "void a() {}");
- Tool.mapVirtualFile("/b.cc", "void b() {}");
- std::unique_ptr<FrontendActionFactory> Action(
- newFrontendActionFactory(&EndCallback, &EndCallback));
- Tool.run(Action.get());
- EXPECT_TRUE(EndCallback.Matched);
- EXPECT_EQ(2u, EndCallback.BeginCalled);
- EXPECT_EQ(2u, EndCallback.EndCalled);
- }
- #endif
- struct SkipBodyConsumer : public clang::ASTConsumer {
- /// Skip the 'skipMe' function.
- bool shouldSkipFunctionBody(Decl *D) override {
- FunctionDecl *F = dyn_cast<FunctionDecl>(D);
- return F && F->getNameAsString() == "skipMe";
- }
- };
- struct SkipBodyAction : public clang::ASTFrontendAction {
- std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &Compiler,
- StringRef) override {
- Compiler.getFrontendOpts().SkipFunctionBodies = true;
- return llvm::make_unique<SkipBodyConsumer>();
- }
- };
- TEST(runToolOnCode, TestSkipFunctionBody) {
- EXPECT_TRUE(runToolOnCode(new SkipBodyAction,
- "int skipMe() { an_error_here }"));
- EXPECT_FALSE(runToolOnCode(new SkipBodyAction,
- "int skipMeNot() { an_error_here }"));
- }
- TEST(runToolOnCodeWithArgs, TestNoDepFile) {
- llvm::SmallString<32> DepFilePath;
- ASSERT_FALSE(
- llvm::sys::fs::createTemporaryFile("depfile", "d", DepFilePath));
- std::vector<std::string> Args;
- Args.push_back("-MMD");
- Args.push_back("-MT");
- Args.push_back(DepFilePath.str());
- Args.push_back("-MF");
- Args.push_back(DepFilePath.str());
- EXPECT_TRUE(runToolOnCodeWithArgs(new SkipBodyAction, "", Args));
- EXPECT_FALSE(llvm::sys::fs::exists(DepFilePath.str()));
- EXPECT_FALSE(llvm::sys::fs::remove(DepFilePath.str()));
- }
- TEST(ClangToolTest, ArgumentAdjusters) {
- FixedCompilationDatabase Compilations("/", std::vector<std::string>());
- ClangTool Tool(Compilations, std::vector<std::string>(1, "/a.cc"));
- Tool.mapVirtualFile("/a.cc", "void a() {}");
- std::unique_ptr<FrontendActionFactory> Action(
- newFrontendActionFactory<SyntaxOnlyAction>());
- bool Found = false;
- bool Ran = false;
- ArgumentsAdjuster CheckSyntaxOnlyAdjuster =
- [&Found, &Ran](const CommandLineArguments &Args) {
- Ran = true;
- if (std::find(Args.begin(), Args.end(), "-fsyntax-only") != Args.end())
- Found = true;
- return Args;
- };
- Tool.appendArgumentsAdjuster(CheckSyntaxOnlyAdjuster);
- Tool.run(Action.get());
- EXPECT_TRUE(Ran);
- EXPECT_TRUE(Found);
- Ran = Found = false;
- Tool.clearArgumentsAdjusters();
- Tool.appendArgumentsAdjuster(CheckSyntaxOnlyAdjuster);
- Tool.appendArgumentsAdjuster(getClangSyntaxOnlyAdjuster());
- Tool.run(Action.get());
- EXPECT_TRUE(Ran);
- EXPECT_FALSE(Found);
- }
- #ifndef LLVM_ON_WIN32
- TEST(ClangToolTest, BuildASTs) {
- FixedCompilationDatabase Compilations("/", std::vector<std::string>());
- std::vector<std::string> Sources;
- Sources.push_back("/a.cc");
- Sources.push_back("/b.cc");
- ClangTool Tool(Compilations, Sources);
- Tool.mapVirtualFile("/a.cc", "void a() {}");
- Tool.mapVirtualFile("/b.cc", "void b() {}");
- std::vector<std::unique_ptr<ASTUnit>> ASTs;
- EXPECT_EQ(0, Tool.buildASTs(ASTs));
- EXPECT_EQ(2u, ASTs.size());
- }
- struct TestDiagnosticConsumer : public DiagnosticConsumer {
- TestDiagnosticConsumer() : NumDiagnosticsSeen(0) {}
- void HandleDiagnostic(DiagnosticsEngine::Level DiagLevel,
- const Diagnostic &Info) override {
- ++NumDiagnosticsSeen;
- }
- unsigned NumDiagnosticsSeen;
- };
- TEST(ClangToolTest, InjectDiagnosticConsumer) {
- FixedCompilationDatabase Compilations("/", std::vector<std::string>());
- ClangTool Tool(Compilations, std::vector<std::string>(1, "/a.cc"));
- Tool.mapVirtualFile("/a.cc", "int x = undeclared;");
- TestDiagnosticConsumer Consumer;
- Tool.setDiagnosticConsumer(&Consumer);
- std::unique_ptr<FrontendActionFactory> Action(
- newFrontendActionFactory<SyntaxOnlyAction>());
- Tool.run(Action.get());
- EXPECT_EQ(1u, Consumer.NumDiagnosticsSeen);
- }
- TEST(ClangToolTest, InjectDiagnosticConsumerInBuildASTs) {
- FixedCompilationDatabase Compilations("/", std::vector<std::string>());
- ClangTool Tool(Compilations, std::vector<std::string>(1, "/a.cc"));
- Tool.mapVirtualFile("/a.cc", "int x = undeclared;");
- TestDiagnosticConsumer Consumer;
- Tool.setDiagnosticConsumer(&Consumer);
- std::vector<std::unique_ptr<ASTUnit>> ASTs;
- Tool.buildASTs(ASTs);
- EXPECT_EQ(1u, ASTs.size());
- EXPECT_EQ(1u, Consumer.NumDiagnosticsSeen);
- }
- #endif
- } // end namespace tooling
- } // end namespace clang
|