You've already forked linux-packaging-mono
acceptance-tests
data
debian
docs
external
Newtonsoft.Json
api-doc-tools
api-snapshot
aspnetwebstack
bdwgc
binary-reference-assemblies
bockbuild
boringssl
cecil
cecil-legacy
corefx
corert
helix-binaries
ikdasm
ikvm
illinker-test-assets
linker
llvm-project
clang
INPUTS
bindings
cmake
docs
examples
include
lib
runtime
tools
unittests
AST
ASTMatchers
Analysis
Basic
CodeGen
CrossTU
CMakeLists.txt
CrossTranslationUnitTest.cpp
Driver
Format
Frontend
Lex
Rename
Rewrite
Sema
StaticAnalyzer
Tooling
libclang
CMakeLists.txt
utils
www
.arcconfig
.clang-format
.clang-tidy
.gitignore
CMakeLists.txt
CODE_OWNERS.TXT
INSTALL.txt
LICENSE.TXT
ModuleInfo.txt
NOTES.txt
README.txt
clang-tools-extra
compiler-rt
libcxx
libcxxabi
libunwind
lld
lldb
llvm
openmp
polly
nuget-buildtasks
nunit-lite
roslyn-binaries
rx
xunit-binaries
how-to-bump-roslyn-binaries.md
ikvm-native
llvm
m4
man
mcs
mk
mono
msvc
netcore
po
runtime
samples
scripts
support
tools
COPYING.LIB
LICENSE
Makefile.am
Makefile.in
NEWS
README.md
acinclude.m4
aclocal.m4
autogen.sh
code_of_conduct.md
compile
config.guess
config.h.in
config.rpath
config.sub
configure.REMOVED.git-id
configure.ac.REMOVED.git-id
depcomp
install-sh
ltmain.sh.REMOVED.git-id
missing
mkinstalldirs
mono-uninstalled.pc.in
test-driver
winconfig.h
159 lines
5.3 KiB
C++
159 lines
5.3 KiB
C++
//===- unittest/Tooling/CrossTranslationUnitTest.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/CrossTU/CrossTranslationUnit.h"
|
|
#include "clang/AST/ASTConsumer.h"
|
|
#include "clang/Frontend/FrontendAction.h"
|
|
#include "clang/Tooling/Tooling.h"
|
|
#include "llvm/Config/llvm-config.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/ToolOutputFile.h"
|
|
#include "gtest/gtest.h"
|
|
#include <cassert>
|
|
|
|
namespace clang {
|
|
namespace cross_tu {
|
|
|
|
namespace {
|
|
|
|
class CTUASTConsumer : public clang::ASTConsumer {
|
|
public:
|
|
explicit CTUASTConsumer(clang::CompilerInstance &CI, bool *Success)
|
|
: CTU(CI), Success(Success) {}
|
|
|
|
void HandleTranslationUnit(ASTContext &Ctx) {
|
|
const TranslationUnitDecl *TU = Ctx.getTranslationUnitDecl();
|
|
const FunctionDecl *FD = nullptr;
|
|
for (const Decl *D : TU->decls()) {
|
|
FD = dyn_cast<FunctionDecl>(D);
|
|
if (FD && FD->getName() == "f")
|
|
break;
|
|
}
|
|
assert(FD && FD->getName() == "f");
|
|
bool OrigFDHasBody = FD->hasBody();
|
|
|
|
// Prepare the index file and the AST file.
|
|
int ASTFD;
|
|
llvm::SmallString<256> ASTFileName;
|
|
ASSERT_FALSE(
|
|
llvm::sys::fs::createTemporaryFile("f_ast", "ast", ASTFD, ASTFileName));
|
|
llvm::ToolOutputFile ASTFile(ASTFileName, ASTFD);
|
|
|
|
int IndexFD;
|
|
llvm::SmallString<256> IndexFileName;
|
|
ASSERT_FALSE(llvm::sys::fs::createTemporaryFile("index", "txt", IndexFD,
|
|
IndexFileName));
|
|
llvm::ToolOutputFile IndexFile(IndexFileName, IndexFD);
|
|
IndexFile.os() << "c:@F@f#I# " << ASTFileName << "\n";
|
|
IndexFile.os().flush();
|
|
EXPECT_TRUE(llvm::sys::fs::exists(IndexFileName));
|
|
|
|
StringRef SourceText = "int f(int) { return 0; }\n";
|
|
// This file must exist since the saved ASTFile will reference it.
|
|
int SourceFD;
|
|
llvm::SmallString<256> SourceFileName;
|
|
ASSERT_FALSE(llvm::sys::fs::createTemporaryFile("input", "cpp", SourceFD,
|
|
SourceFileName));
|
|
llvm::ToolOutputFile SourceFile(SourceFileName, SourceFD);
|
|
SourceFile.os() << SourceText;
|
|
SourceFile.os().flush();
|
|
EXPECT_TRUE(llvm::sys::fs::exists(SourceFileName));
|
|
|
|
std::unique_ptr<ASTUnit> ASTWithDefinition =
|
|
tooling::buildASTFromCode(SourceText, SourceFileName);
|
|
ASTWithDefinition->Save(ASTFileName.str());
|
|
EXPECT_TRUE(llvm::sys::fs::exists(ASTFileName));
|
|
|
|
// Load the definition from the AST file.
|
|
llvm::Expected<const FunctionDecl *> NewFDorError =
|
|
CTU.getCrossTUDefinition(FD, "", IndexFileName);
|
|
EXPECT_TRUE((bool)NewFDorError);
|
|
const FunctionDecl *NewFD = *NewFDorError;
|
|
|
|
*Success = NewFD && NewFD->hasBody() && !OrigFDHasBody;
|
|
}
|
|
|
|
private:
|
|
CrossTranslationUnitContext CTU;
|
|
bool *Success;
|
|
};
|
|
|
|
class CTUAction : public clang::ASTFrontendAction {
|
|
public:
|
|
CTUAction(bool *Success) : Success(Success) {}
|
|
|
|
protected:
|
|
std::unique_ptr<clang::ASTConsumer>
|
|
CreateASTConsumer(clang::CompilerInstance &CI, StringRef) override {
|
|
return llvm::make_unique<CTUASTConsumer>(CI, Success);
|
|
}
|
|
|
|
private:
|
|
bool *Success;
|
|
};
|
|
|
|
} // end namespace
|
|
|
|
TEST(CrossTranslationUnit, CanLoadFunctionDefinition) {
|
|
bool Success = false;
|
|
EXPECT_TRUE(tooling::runToolOnCode(new CTUAction(&Success), "int f(int);"));
|
|
EXPECT_TRUE(Success);
|
|
}
|
|
|
|
TEST(CrossTranslationUnit, IndexFormatCanBeParsed) {
|
|
llvm::StringMap<std::string> Index;
|
|
Index["a"] = "/b/f1";
|
|
Index["c"] = "/d/f2";
|
|
Index["e"] = "/f/f3";
|
|
std::string IndexText = createCrossTUIndexString(Index);
|
|
|
|
int IndexFD;
|
|
llvm::SmallString<256> IndexFileName;
|
|
ASSERT_FALSE(llvm::sys::fs::createTemporaryFile("index", "txt", IndexFD,
|
|
IndexFileName));
|
|
llvm::ToolOutputFile IndexFile(IndexFileName, IndexFD);
|
|
IndexFile.os() << IndexText;
|
|
IndexFile.os().flush();
|
|
EXPECT_TRUE(llvm::sys::fs::exists(IndexFileName));
|
|
llvm::Expected<llvm::StringMap<std::string>> IndexOrErr =
|
|
parseCrossTUIndex(IndexFileName, "");
|
|
EXPECT_TRUE((bool)IndexOrErr);
|
|
llvm::StringMap<std::string> ParsedIndex = IndexOrErr.get();
|
|
for (const auto &E : Index) {
|
|
EXPECT_TRUE(ParsedIndex.count(E.getKey()));
|
|
EXPECT_EQ(ParsedIndex[E.getKey()], E.getValue());
|
|
}
|
|
for (const auto &E : ParsedIndex)
|
|
EXPECT_TRUE(Index.count(E.getKey()));
|
|
}
|
|
|
|
TEST(CrossTranslationUnit, CTUDirIsHandledCorrectly) {
|
|
llvm::StringMap<std::string> Index;
|
|
Index["a"] = "/b/c/d";
|
|
std::string IndexText = createCrossTUIndexString(Index);
|
|
|
|
int IndexFD;
|
|
llvm::SmallString<256> IndexFileName;
|
|
ASSERT_FALSE(llvm::sys::fs::createTemporaryFile("index", "txt", IndexFD,
|
|
IndexFileName));
|
|
llvm::ToolOutputFile IndexFile(IndexFileName, IndexFD);
|
|
IndexFile.os() << IndexText;
|
|
IndexFile.os().flush();
|
|
EXPECT_TRUE(llvm::sys::fs::exists(IndexFileName));
|
|
llvm::Expected<llvm::StringMap<std::string>> IndexOrErr =
|
|
parseCrossTUIndex(IndexFileName, "/ctudir");
|
|
EXPECT_TRUE((bool)IndexOrErr);
|
|
llvm::StringMap<std::string> ParsedIndex = IndexOrErr.get();
|
|
EXPECT_EQ(ParsedIndex["a"], "/ctudir/b/c/d");
|
|
}
|
|
|
|
} // end namespace cross_tu
|
|
} // end namespace clang
|