178 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
//===- unittests/Frontend/ASTUnitTest.cpp - ASTUnit tests -----------------===//
 | 
						|
//
 | 
						|
// 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
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include <fstream>
 | 
						|
 | 
						|
#include "clang/Basic/FileManager.h"
 | 
						|
#include "clang/Frontend/ASTUnit.h"
 | 
						|
#include "clang/Frontend/CompilerInstance.h"
 | 
						|
#include "clang/Frontend/CompilerInvocation.h"
 | 
						|
#include "clang/Frontend/PCHContainerOperations.h"
 | 
						|
#include "clang/Lex/HeaderSearch.h"
 | 
						|
#include "llvm/Support/FileSystem.h"
 | 
						|
#include "llvm/Support/Path.h"
 | 
						|
#include "llvm/Support/ToolOutputFile.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
using namespace clang;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
class ASTUnitTest : public ::testing::Test {
 | 
						|
protected:
 | 
						|
  int FD;
 | 
						|
  llvm::SmallString<256> InputFileName;
 | 
						|
  std::unique_ptr<ToolOutputFile> input_file;
 | 
						|
  IntrusiveRefCntPtr<DiagnosticsEngine> Diags;
 | 
						|
  std::shared_ptr<CompilerInvocation> CInvok;
 | 
						|
  std::shared_ptr<PCHContainerOperations> PCHContainerOps;
 | 
						|
 | 
						|
  std::unique_ptr<ASTUnit> createASTUnit(bool isVolatile) {
 | 
						|
    EXPECT_FALSE(llvm::sys::fs::createTemporaryFile("ast-unit", "cpp", FD,
 | 
						|
                                                    InputFileName));
 | 
						|
    input_file = std::make_unique<ToolOutputFile>(InputFileName, FD);
 | 
						|
    input_file->os() << "";
 | 
						|
 | 
						|
    const char *Args[] = {"clang", "-xc++", InputFileName.c_str()};
 | 
						|
 | 
						|
    Diags = CompilerInstance::createDiagnostics(new DiagnosticOptions());
 | 
						|
 | 
						|
    CInvok = createInvocationFromCommandLine(Args, Diags);
 | 
						|
 | 
						|
    if (!CInvok)
 | 
						|
      return nullptr;
 | 
						|
 | 
						|
    FileManager *FileMgr =
 | 
						|
        new FileManager(FileSystemOptions(), vfs::getRealFileSystem());
 | 
						|
    PCHContainerOps = std::make_shared<PCHContainerOperations>();
 | 
						|
 | 
						|
    return ASTUnit::LoadFromCompilerInvocation(
 | 
						|
        CInvok, PCHContainerOps, Diags, FileMgr, false, CaptureDiagsKind::None,
 | 
						|
        0, TU_Complete, false, false, isVolatile);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
TEST_F(ASTUnitTest, SaveLoadPreservesLangOptionsInPrintingPolicy) {
 | 
						|
  // Check that the printing policy is restored with the correct language
 | 
						|
  // options when loading an ASTUnit from a file.  To this end, an ASTUnit
 | 
						|
  // for a C++ translation unit is set up and written to a temporary file.
 | 
						|
 | 
						|
  // By default `UseVoidForZeroParams` is true for non-C++ language options,
 | 
						|
  // thus we can check this field after loading the ASTUnit to deduce whether
 | 
						|
  // the correct (C++) language options were used when setting up the printing
 | 
						|
  // policy.
 | 
						|
 | 
						|
  {
 | 
						|
    PrintingPolicy PolicyWithDefaultLangOpt(LangOptions{});
 | 
						|
    EXPECT_TRUE(PolicyWithDefaultLangOpt.UseVoidForZeroParams);
 | 
						|
  }
 | 
						|
 | 
						|
  std::unique_ptr<ASTUnit> AST = createASTUnit(false);
 | 
						|
 | 
						|
  if (!AST)
 | 
						|
    FAIL() << "failed to create ASTUnit";
 | 
						|
 | 
						|
  EXPECT_FALSE(AST->getASTContext().getPrintingPolicy().UseVoidForZeroParams);
 | 
						|
 | 
						|
  llvm::SmallString<256> ASTFileName;
 | 
						|
  ASSERT_FALSE(
 | 
						|
      llvm::sys::fs::createTemporaryFile("ast-unit", "ast", FD, ASTFileName));
 | 
						|
  ToolOutputFile ast_file(ASTFileName, FD);
 | 
						|
  AST->Save(ASTFileName.str());
 | 
						|
 | 
						|
  EXPECT_TRUE(llvm::sys::fs::exists(ASTFileName));
 | 
						|
 | 
						|
  std::unique_ptr<ASTUnit> AU = ASTUnit::LoadFromASTFile(
 | 
						|
      std::string(ASTFileName.str()), PCHContainerOps->getRawReader(),
 | 
						|
      ASTUnit::LoadEverything, Diags, FileSystemOptions(),
 | 
						|
      /*UseDebugInfo=*/false);
 | 
						|
 | 
						|
  if (!AU)
 | 
						|
    FAIL() << "failed to load ASTUnit";
 | 
						|
 | 
						|
  EXPECT_FALSE(AU->getASTContext().getPrintingPolicy().UseVoidForZeroParams);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(ASTUnitTest, GetBufferForFileMemoryMapping) {
 | 
						|
  std::unique_ptr<ASTUnit> AST = createASTUnit(true);
 | 
						|
 | 
						|
  if (!AST)
 | 
						|
    FAIL() << "failed to create ASTUnit";
 | 
						|
 | 
						|
  std::unique_ptr<llvm::MemoryBuffer> memoryBuffer =
 | 
						|
      AST->getBufferForFile(InputFileName);
 | 
						|
 | 
						|
  EXPECT_NE(memoryBuffer->getBufferKind(),
 | 
						|
            llvm::MemoryBuffer::MemoryBuffer_MMap);
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(ASTUnitTest, ModuleTextualHeader) {
 | 
						|
  llvm::IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFs =
 | 
						|
      new llvm::vfs::InMemoryFileSystem();
 | 
						|
  InMemoryFs->addFile("test.cpp", 0, llvm::MemoryBuffer::getMemBuffer(R"cpp(
 | 
						|
      #include "Textual.h"
 | 
						|
      void foo() {}
 | 
						|
    )cpp"));
 | 
						|
  InMemoryFs->addFile("m.modulemap", 0, llvm::MemoryBuffer::getMemBuffer(R"cpp(
 | 
						|
      module M {
 | 
						|
        module Textual {
 | 
						|
          textual header "Textual.h"
 | 
						|
        }
 | 
						|
      }
 | 
						|
    )cpp"));
 | 
						|
  InMemoryFs->addFile("Textual.h", 0, llvm::MemoryBuffer::getMemBuffer(R"cpp(
 | 
						|
      void foo();
 | 
						|
    )cpp"));
 | 
						|
 | 
						|
  const char *Args[] = {"clang", "test.cpp", "-fmodule-map-file=m.modulemap",
 | 
						|
                        "-fmodule-name=M"};
 | 
						|
  Diags = CompilerInstance::createDiagnostics(new DiagnosticOptions());
 | 
						|
  CInvok = createInvocationFromCommandLine(Args, Diags);
 | 
						|
  ASSERT_TRUE(CInvok);
 | 
						|
 | 
						|
  FileManager *FileMgr = new FileManager(FileSystemOptions(), InMemoryFs);
 | 
						|
  PCHContainerOps = std::make_shared<PCHContainerOperations>();
 | 
						|
 | 
						|
  auto AU = ASTUnit::LoadFromCompilerInvocation(
 | 
						|
      CInvok, PCHContainerOps, Diags, FileMgr, false, CaptureDiagsKind::None, 1,
 | 
						|
      TU_Complete, false, false, false);
 | 
						|
  ASSERT_TRUE(AU);
 | 
						|
  auto File = AU->getFileManager().getFileRef("Textual.h", false, false);
 | 
						|
  ASSERT_TRUE(bool(File));
 | 
						|
  // Verify that we do not crash here.
 | 
						|
  EXPECT_TRUE(AU->getPreprocessor().getHeaderSearchInfo().getExistingFileInfo(
 | 
						|
      &File->getFileEntry()));
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(ASTUnitTest, LoadFromCommandLineEarlyError) {
 | 
						|
  EXPECT_FALSE(
 | 
						|
      llvm::sys::fs::createTemporaryFile("ast-unit", "c", FD, InputFileName));
 | 
						|
  input_file = std::make_unique<ToolOutputFile>(InputFileName, FD);
 | 
						|
  input_file->os() << "";
 | 
						|
 | 
						|
  const char *Args[] = {"clang", "-target", "foobar", InputFileName.c_str()};
 | 
						|
 | 
						|
  auto Diags = CompilerInstance::createDiagnostics(new DiagnosticOptions());
 | 
						|
  auto PCHContainerOps = std::make_shared<PCHContainerOperations>();
 | 
						|
  std::unique_ptr<clang::ASTUnit> ErrUnit;
 | 
						|
 | 
						|
  ASTUnit *AST = ASTUnit::LoadFromCommandLine(
 | 
						|
      &Args[0], &Args[4], PCHContainerOps, Diags, "", false,
 | 
						|
      CaptureDiagsKind::All, None, true, 0, TU_Complete, false, false, false,
 | 
						|
      SkipFunctionBodiesScope::None, false, true, false, false, None, &ErrUnit,
 | 
						|
      nullptr);
 | 
						|
 | 
						|
  ASSERT_EQ(AST, nullptr);
 | 
						|
  ASSERT_NE(ErrUnit, nullptr);
 | 
						|
  ASSERT_TRUE(Diags->hasErrorOccurred());
 | 
						|
  ASSERT_NE(ErrUnit->stored_diag_size(), 0U);
 | 
						|
}
 | 
						|
 | 
						|
} // anonymous namespace
 |