2017-08-27 19:31:01 +08:00
|
|
|
//=== unittests/CodeGen/IncrementalProcessingTest.cpp - IncrementalCodeGen ===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-08-27 19:31:01 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/AST/ASTConsumer.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/AST/RecursiveASTVisitor.h"
|
|
|
|
#include "clang/Basic/TargetInfo.h"
|
|
|
|
#include "clang/CodeGen/ModuleBuilder.h"
|
|
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
|
|
#include "clang/Lex/Preprocessor.h"
|
|
|
|
#include "clang/Parse/Parser.h"
|
|
|
|
#include "clang/Sema/Sema.h"
|
|
|
|
#include "llvm/ADT/Triple.h"
|
|
|
|
#include "llvm/IR/LLVMContext.h"
|
|
|
|
#include "llvm/IR/Module.h"
|
|
|
|
#include "llvm/Support/Host.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Incremental processing produces several modules, all using the same "main
|
|
|
|
// file". Make sure CodeGen can cope with that, e.g. for static initializers.
|
|
|
|
const char TestProgram1[] =
|
|
|
|
"extern \"C\" int funcForProg1() { return 17; }\n"
|
|
|
|
"struct EmitCXXGlobalInitFunc1 {\n"
|
|
|
|
" EmitCXXGlobalInitFunc1() {}\n"
|
|
|
|
"} test1;";
|
|
|
|
|
|
|
|
const char TestProgram2[] =
|
|
|
|
"extern \"C\" int funcForProg2() { return 42; }\n"
|
|
|
|
"struct EmitCXXGlobalInitFunc2 {\n"
|
|
|
|
" EmitCXXGlobalInitFunc2() {}\n"
|
|
|
|
"} test2;";
|
|
|
|
|
|
|
|
|
|
|
|
/// An incremental version of ParseAST().
|
|
|
|
static std::unique_ptr<llvm::Module>
|
|
|
|
IncrementalParseAST(CompilerInstance& CI, Parser& P,
|
|
|
|
CodeGenerator& CG, const char* code) {
|
|
|
|
static int counter = 0;
|
|
|
|
struct IncreaseCounterOnRet {
|
|
|
|
~IncreaseCounterOnRet() {
|
|
|
|
++counter;
|
|
|
|
}
|
|
|
|
} ICOR;
|
|
|
|
|
|
|
|
Sema& S = CI.getSema();
|
|
|
|
clang::SourceManager &SM = S.getSourceManager();
|
|
|
|
if (!code) {
|
|
|
|
// Main file
|
|
|
|
SM.setMainFileID(SM.createFileID(
|
|
|
|
llvm::MemoryBuffer::getMemBuffer(" "), clang::SrcMgr::C_User));
|
|
|
|
|
|
|
|
S.getPreprocessor().EnterMainSourceFile();
|
|
|
|
P.Initialize();
|
|
|
|
} else {
|
|
|
|
FileID FID = SM.createFileID(
|
|
|
|
llvm::MemoryBuffer::getMemBuffer(code), clang::SrcMgr::C_User);
|
|
|
|
SourceLocation MainStartLoc = SM.getLocForStartOfFile(SM.getMainFileID());
|
|
|
|
SourceLocation InclLoc = MainStartLoc.getLocWithOffset(counter);
|
|
|
|
S.getPreprocessor().EnterSourceFile(FID, 0, InclLoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
ExternalASTSource *External = S.getASTContext().getExternalSource();
|
|
|
|
if (External)
|
|
|
|
External->StartTranslationUnit(&CG);
|
|
|
|
|
|
|
|
Parser::DeclGroupPtrTy ADecl;
|
|
|
|
for (bool AtEOF = P.ParseFirstTopLevelDecl(ADecl); !AtEOF;
|
|
|
|
AtEOF = P.ParseTopLevelDecl(ADecl)) {
|
|
|
|
// If we got a null return and something *was* parsed, ignore it. This
|
|
|
|
// is due to a top-level semicolon, an action override, or a parse error
|
|
|
|
// skipping something.
|
|
|
|
if (ADecl && !CG.HandleTopLevelDecl(ADecl.get()))
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process any TopLevelDecls generated by #pragma weak.
|
|
|
|
for (Decl *D : S.WeakTopLevelDecls())
|
|
|
|
CG.HandleTopLevelDecl(DeclGroupRef(D));
|
|
|
|
|
|
|
|
CG.HandleTranslationUnit(S.getASTContext());
|
|
|
|
|
|
|
|
std::unique_ptr<llvm::Module> M(CG.ReleaseModule());
|
|
|
|
// Switch to next module.
|
|
|
|
CG.StartModule("incremental-module-" + std::to_string(counter),
|
|
|
|
M->getContext());
|
|
|
|
return M;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Function* getGlobalInit(llvm::Module& M) {
|
|
|
|
for (const auto& Func: M)
|
|
|
|
if (Func.hasName() && Func.getName().startswith("_GLOBAL__sub_I_"))
|
|
|
|
return &Func;
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(IncrementalProcessing, EmitCXXGlobalInitFunc) {
|
|
|
|
LLVMContext Context;
|
|
|
|
CompilerInstance compiler;
|
|
|
|
|
|
|
|
compiler.createDiagnostics();
|
|
|
|
compiler.getLangOpts().CPlusPlus = 1;
|
|
|
|
compiler.getLangOpts().CPlusPlus11 = 1;
|
|
|
|
|
|
|
|
compiler.getTargetOpts().Triple = llvm::Triple::normalize(
|
|
|
|
llvm::sys::getProcessTriple());
|
|
|
|
compiler.setTarget(clang::TargetInfo::CreateTargetInfo(
|
|
|
|
compiler.getDiagnostics(),
|
|
|
|
std::make_shared<clang::TargetOptions>(
|
|
|
|
compiler.getTargetOpts())));
|
|
|
|
|
|
|
|
compiler.createFileManager();
|
|
|
|
compiler.createSourceManager(compiler.getFileManager());
|
|
|
|
compiler.createPreprocessor(clang::TU_Prefix);
|
|
|
|
compiler.getPreprocessor().enableIncrementalProcessing();
|
|
|
|
|
|
|
|
compiler.createASTContext();
|
|
|
|
|
|
|
|
CodeGenerator* CG =
|
|
|
|
CreateLLVMCodeGen(
|
|
|
|
compiler.getDiagnostics(),
|
|
|
|
"main-module",
|
|
|
|
compiler.getHeaderSearchOpts(),
|
|
|
|
compiler.getPreprocessorOpts(),
|
|
|
|
compiler.getCodeGenOpts(),
|
|
|
|
Context);
|
|
|
|
compiler.setASTConsumer(std::unique_ptr<ASTConsumer>(CG));
|
|
|
|
compiler.createSema(clang::TU_Prefix, nullptr);
|
|
|
|
Sema& S = compiler.getSema();
|
|
|
|
|
|
|
|
std::unique_ptr<Parser> ParseOP(new Parser(S.getPreprocessor(), S,
|
|
|
|
/*SkipFunctionBodies*/ false));
|
|
|
|
Parser &P = *ParseOP.get();
|
|
|
|
|
|
|
|
std::array<std::unique_ptr<llvm::Module>, 3> M;
|
|
|
|
M[0] = IncrementalParseAST(compiler, P, *CG, nullptr);
|
|
|
|
ASSERT_TRUE(M[0]);
|
|
|
|
|
|
|
|
M[1] = IncrementalParseAST(compiler, P, *CG, TestProgram1);
|
|
|
|
ASSERT_TRUE(M[1]);
|
|
|
|
ASSERT_TRUE(M[1]->getFunction("funcForProg1"));
|
|
|
|
|
|
|
|
M[2] = IncrementalParseAST(compiler, P, *CG, TestProgram2);
|
|
|
|
ASSERT_TRUE(M[2]);
|
|
|
|
ASSERT_TRUE(M[2]->getFunction("funcForProg2"));
|
|
|
|
// First code should not end up in second module:
|
|
|
|
ASSERT_FALSE(M[2]->getFunction("funcForProg1"));
|
|
|
|
|
|
|
|
// Make sure global inits exist and are unique:
|
|
|
|
const Function* GlobalInit1 = getGlobalInit(*M[1]);
|
|
|
|
ASSERT_TRUE(GlobalInit1);
|
|
|
|
|
|
|
|
const Function* GlobalInit2 = getGlobalInit(*M[2]);
|
|
|
|
ASSERT_TRUE(GlobalInit2);
|
|
|
|
|
|
|
|
ASSERT_FALSE(GlobalInit1->getName() == GlobalInit2->getName());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end anonymous namespace
|