2011-02-03 12:51:52 +08:00
|
|
|
//===- unittests/Frontend/FrontendActionTest.cpp - FrontendAction tests ---===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-12-04 17:45:34 +08:00
|
|
|
#include "clang/Frontend/FrontendAction.h"
|
2011-02-03 12:51:52 +08:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2012-12-04 17:45:34 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2012-12-04 17:53:37 +08:00
|
|
|
#include "clang/AST/RecursiveASTVisitor.h"
|
2011-02-03 12:51:52 +08:00
|
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
|
|
#include "clang/Frontend/CompilerInvocation.h"
|
2013-11-24 10:12:18 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2014-10-23 01:50:19 +08:00
|
|
|
#include "clang/Sema/Sema.h"
|
2011-02-03 12:51:52 +08:00
|
|
|
#include "llvm/ADT/Triple.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class TestASTFrontendAction : public ASTFrontendAction {
|
|
|
|
public:
|
2014-10-23 01:50:19 +08:00
|
|
|
TestASTFrontendAction(bool enableIncrementalProcessing = false,
|
|
|
|
bool actOnEndOfTranslationUnit = false)
|
|
|
|
: EnableIncrementalProcessing(enableIncrementalProcessing),
|
|
|
|
ActOnEndOfTranslationUnit(actOnEndOfTranslationUnit) { }
|
2013-11-24 10:12:18 +08:00
|
|
|
|
|
|
|
bool EnableIncrementalProcessing;
|
2014-10-23 01:50:19 +08:00
|
|
|
bool ActOnEndOfTranslationUnit;
|
2011-02-03 12:51:52 +08:00
|
|
|
std::vector<std::string> decl_names;
|
|
|
|
|
2015-04-11 10:00:23 +08:00
|
|
|
bool BeginSourceFileAction(CompilerInstance &ci,
|
|
|
|
StringRef filename) override {
|
2013-11-24 10:12:18 +08:00
|
|
|
if (EnableIncrementalProcessing)
|
|
|
|
ci.getPreprocessor().enableIncrementalProcessing();
|
|
|
|
|
|
|
|
return ASTFrontendAction::BeginSourceFileAction(ci, filename);
|
|
|
|
}
|
|
|
|
|
2015-04-11 10:00:23 +08:00
|
|
|
std::unique_ptr<ASTConsumer> CreateASTConsumer(CompilerInstance &CI,
|
|
|
|
StringRef InFile) override {
|
2014-10-23 01:50:19 +08:00
|
|
|
return llvm::make_unique<Visitor>(CI, ActOnEndOfTranslationUnit,
|
|
|
|
decl_names);
|
2011-02-03 12:51:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
class Visitor : public ASTConsumer, public RecursiveASTVisitor<Visitor> {
|
|
|
|
public:
|
2014-10-23 01:50:19 +08:00
|
|
|
Visitor(CompilerInstance &CI, bool ActOnEndOfTranslationUnit,
|
|
|
|
std::vector<std::string> &decl_names) :
|
|
|
|
CI(CI), ActOnEndOfTranslationUnit(ActOnEndOfTranslationUnit),
|
|
|
|
decl_names_(decl_names) {}
|
2011-02-03 12:51:52 +08:00
|
|
|
|
2015-04-11 10:00:23 +08:00
|
|
|
void HandleTranslationUnit(ASTContext &context) override {
|
2014-10-23 01:50:19 +08:00
|
|
|
if (ActOnEndOfTranslationUnit) {
|
|
|
|
CI.getSema().ActOnEndOfTranslationUnit();
|
|
|
|
}
|
2011-02-03 12:51:52 +08:00
|
|
|
TraverseDecl(context.getTranslationUnitDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool VisitNamedDecl(NamedDecl *Decl) {
|
|
|
|
decl_names_.push_back(Decl->getQualifiedNameAsString());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2014-10-23 01:50:19 +08:00
|
|
|
CompilerInstance &CI;
|
|
|
|
bool ActOnEndOfTranslationUnit;
|
2011-02-03 12:51:52 +08:00
|
|
|
std::vector<std::string> &decl_names_;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(ASTFrontendAction, Sanity) {
|
|
|
|
CompilerInvocation *invocation = new CompilerInvocation;
|
|
|
|
invocation->getPreprocessorOpts().addRemappedFile(
|
2014-08-28 04:03:29 +08:00
|
|
|
"test.cc",
|
|
|
|
MemoryBuffer::getMemBuffer("int main() { float x; }").release());
|
2012-01-21 00:33:00 +08:00
|
|
|
invocation->getFrontendOpts().Inputs.push_back(FrontendInputFile("test.cc",
|
|
|
|
IK_CXX));
|
2011-02-03 12:51:52 +08:00
|
|
|
invocation->getFrontendOpts().ProgramAction = frontend::ParseSyntaxOnly;
|
|
|
|
invocation->getTargetOpts().Triple = "i386-unknown-linux-gnu";
|
|
|
|
CompilerInstance compiler;
|
|
|
|
compiler.setInvocation(invocation);
|
2013-01-20 09:58:28 +08:00
|
|
|
compiler.createDiagnostics();
|
2011-02-03 12:51:52 +08:00
|
|
|
|
|
|
|
TestASTFrontendAction test_action;
|
|
|
|
ASSERT_TRUE(compiler.ExecuteAction(test_action));
|
2013-12-15 18:36:26 +08:00
|
|
|
ASSERT_EQ(2U, test_action.decl_names.size());
|
|
|
|
EXPECT_EQ("main", test_action.decl_names[0]);
|
|
|
|
EXPECT_EQ("x", test_action.decl_names[1]);
|
2011-02-03 12:51:52 +08:00
|
|
|
}
|
|
|
|
|
2013-11-24 10:12:18 +08:00
|
|
|
TEST(ASTFrontendAction, IncrementalParsing) {
|
|
|
|
CompilerInvocation *invocation = new CompilerInvocation;
|
|
|
|
invocation->getPreprocessorOpts().addRemappedFile(
|
2014-08-28 04:03:29 +08:00
|
|
|
"test.cc",
|
|
|
|
MemoryBuffer::getMemBuffer("int main() { float x; }").release());
|
2013-11-24 10:12:18 +08:00
|
|
|
invocation->getFrontendOpts().Inputs.push_back(FrontendInputFile("test.cc",
|
|
|
|
IK_CXX));
|
|
|
|
invocation->getFrontendOpts().ProgramAction = frontend::ParseSyntaxOnly;
|
|
|
|
invocation->getTargetOpts().Triple = "i386-unknown-linux-gnu";
|
|
|
|
CompilerInstance compiler;
|
|
|
|
compiler.setInvocation(invocation);
|
|
|
|
compiler.createDiagnostics();
|
|
|
|
|
|
|
|
TestASTFrontendAction test_action(/*enableIncrementalProcessing=*/true);
|
|
|
|
ASSERT_TRUE(compiler.ExecuteAction(test_action));
|
2013-12-15 18:36:26 +08:00
|
|
|
ASSERT_EQ(2U, test_action.decl_names.size());
|
|
|
|
EXPECT_EQ("main", test_action.decl_names[0]);
|
|
|
|
EXPECT_EQ("x", test_action.decl_names[1]);
|
2013-11-24 10:12:18 +08:00
|
|
|
}
|
|
|
|
|
2014-10-23 01:50:19 +08:00
|
|
|
TEST(ASTFrontendAction, LateTemplateIncrementalParsing) {
|
|
|
|
CompilerInvocation *invocation = new CompilerInvocation;
|
|
|
|
invocation->getLangOpts()->CPlusPlus = true;
|
|
|
|
invocation->getLangOpts()->DelayedTemplateParsing = true;
|
|
|
|
invocation->getPreprocessorOpts().addRemappedFile(
|
|
|
|
"test.cc", MemoryBuffer::getMemBuffer(
|
|
|
|
"template<typename T> struct A { A(T); T data; };\n"
|
|
|
|
"template<typename T> struct B: public A<T> {\n"
|
|
|
|
" B();\n"
|
|
|
|
" B(B const& b): A<T>(b.data) {}\n"
|
|
|
|
"};\n"
|
|
|
|
"B<char> c() { return B<char>(); }\n").release());
|
|
|
|
invocation->getFrontendOpts().Inputs.push_back(FrontendInputFile("test.cc",
|
|
|
|
IK_CXX));
|
|
|
|
invocation->getFrontendOpts().ProgramAction = frontend::ParseSyntaxOnly;
|
|
|
|
invocation->getTargetOpts().Triple = "i386-unknown-linux-gnu";
|
|
|
|
CompilerInstance compiler;
|
|
|
|
compiler.setInvocation(invocation);
|
|
|
|
compiler.createDiagnostics();
|
|
|
|
|
|
|
|
TestASTFrontendAction test_action(/*enableIncrementalProcessing=*/true,
|
|
|
|
/*actOnEndOfTranslationUnit=*/true);
|
|
|
|
ASSERT_TRUE(compiler.ExecuteAction(test_action));
|
|
|
|
ASSERT_EQ(13U, test_action.decl_names.size());
|
|
|
|
EXPECT_EQ("A", test_action.decl_names[0]);
|
|
|
|
EXPECT_EQ("c", test_action.decl_names[12]);
|
|
|
|
}
|
|
|
|
|
2014-08-08 04:51:16 +08:00
|
|
|
struct TestPPCallbacks : public PPCallbacks {
|
|
|
|
TestPPCallbacks() : SeenEnd(false) {}
|
|
|
|
|
|
|
|
void EndOfMainFile() override { SeenEnd = true; }
|
|
|
|
|
|
|
|
bool SeenEnd;
|
|
|
|
};
|
|
|
|
|
|
|
|
class TestPPCallbacksFrontendAction : public PreprocessorFrontendAction {
|
2014-09-10 17:35:49 +08:00
|
|
|
TestPPCallbacks *Callbacks;
|
2014-08-08 04:51:16 +08:00
|
|
|
|
|
|
|
public:
|
2014-09-10 17:35:49 +08:00
|
|
|
TestPPCallbacksFrontendAction(TestPPCallbacks *C)
|
|
|
|
: Callbacks(C), SeenEnd(false) {}
|
2014-08-08 04:51:16 +08:00
|
|
|
|
|
|
|
void ExecuteAction() override {
|
|
|
|
Preprocessor &PP = getCompilerInstance().getPreprocessor();
|
2014-09-10 17:35:49 +08:00
|
|
|
PP.addPPCallbacks(std::unique_ptr<TestPPCallbacks>(Callbacks));
|
2014-08-08 04:51:16 +08:00
|
|
|
PP.EnterMainSourceFile();
|
|
|
|
}
|
|
|
|
void EndSourceFileAction() override { SeenEnd = Callbacks->SeenEnd; }
|
|
|
|
|
|
|
|
bool SeenEnd;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(PreprocessorFrontendAction, EndSourceFile) {
|
|
|
|
CompilerInvocation *Invocation = new CompilerInvocation;
|
|
|
|
Invocation->getPreprocessorOpts().addRemappedFile(
|
2014-08-28 04:03:29 +08:00
|
|
|
"test.cc",
|
|
|
|
MemoryBuffer::getMemBuffer("int main() { float x; }").release());
|
2014-08-08 04:51:16 +08:00
|
|
|
Invocation->getFrontendOpts().Inputs.push_back(
|
|
|
|
FrontendInputFile("test.cc", IK_CXX));
|
|
|
|
Invocation->getFrontendOpts().ProgramAction = frontend::ParseSyntaxOnly;
|
|
|
|
Invocation->getTargetOpts().Triple = "i386-unknown-linux-gnu";
|
|
|
|
CompilerInstance Compiler;
|
|
|
|
Compiler.setInvocation(Invocation);
|
|
|
|
Compiler.createDiagnostics();
|
|
|
|
|
2014-09-10 17:35:49 +08:00
|
|
|
TestPPCallbacks *Callbacks = new TestPPCallbacks;
|
|
|
|
TestPPCallbacksFrontendAction TestAction(Callbacks);
|
2014-08-08 04:51:16 +08:00
|
|
|
ASSERT_FALSE(Callbacks->SeenEnd);
|
|
|
|
ASSERT_FALSE(TestAction.SeenEnd);
|
|
|
|
ASSERT_TRUE(Compiler.ExecuteAction(TestAction));
|
|
|
|
// Check that EndOfMainFile was called before EndSourceFileAction.
|
|
|
|
ASSERT_TRUE(TestAction.SeenEnd);
|
|
|
|
}
|
|
|
|
|
2011-02-03 12:51:52 +08:00
|
|
|
} // anonymous namespace
|