You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			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
 |