mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-18 14:56:44 +00:00

Previously, everytime we want to get a source file declaring a specific module, we need to scan the whole projects again and again. The performance is super bad. This patch tries to improve this by introducing a simple cache.
653 lines
19 KiB
C++
653 lines
19 KiB
C++
//===--------------- PrerequisiteModulesTests.cpp -------------------*- C++
|
|
//-*-===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// FIXME: Skip testing on windows temporarily due to the different escaping
|
|
/// code mode.
|
|
#ifndef _WIN32
|
|
|
|
#include "Annotations.h"
|
|
#include "CodeComplete.h"
|
|
#include "Compiler.h"
|
|
#include "ModulesBuilder.h"
|
|
#include "ScanningProjectModules.h"
|
|
#include "TestTU.h"
|
|
#include "support/ThreadsafeFS.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
|
|
namespace clang::clangd {
|
|
namespace {
|
|
|
|
class GlobalScanningCounterProjectModules : public ProjectModules {
|
|
public:
|
|
GlobalScanningCounterProjectModules(
|
|
std::unique_ptr<ProjectModules> Underlying, std::atomic<unsigned> &Count)
|
|
: Underlying(std::move(Underlying)), Count(Count) {}
|
|
|
|
std::vector<std::string> getRequiredModules(PathRef File) override {
|
|
return Underlying->getRequiredModules(File);
|
|
}
|
|
|
|
std::string getModuleNameForSource(PathRef File) override {
|
|
return Underlying->getModuleNameForSource(File);
|
|
}
|
|
|
|
void setCommandMangler(CommandMangler Mangler) override {
|
|
Underlying->setCommandMangler(std::move(Mangler));
|
|
}
|
|
|
|
std::string getSourceForModuleName(llvm::StringRef ModuleName,
|
|
PathRef RequiredSrcFile) override {
|
|
Count++;
|
|
return Underlying->getSourceForModuleName(ModuleName, RequiredSrcFile);
|
|
}
|
|
|
|
private:
|
|
std::unique_ptr<ProjectModules> Underlying;
|
|
std::atomic<unsigned> &Count;
|
|
};
|
|
|
|
class MockDirectoryCompilationDatabase : public MockCompilationDatabase {
|
|
public:
|
|
MockDirectoryCompilationDatabase(StringRef TestDir, const ThreadsafeFS &TFS)
|
|
: MockCompilationDatabase(TestDir),
|
|
MockedCDBPtr(std::make_shared<MockClangCompilationDatabase>(*this)),
|
|
TFS(TFS), GlobalScanningCount(0) {
|
|
this->ExtraClangFlags.push_back("-std=c++20");
|
|
this->ExtraClangFlags.push_back("-c");
|
|
}
|
|
|
|
void addFile(llvm::StringRef Path, llvm::StringRef Contents);
|
|
|
|
std::unique_ptr<ProjectModules> getProjectModules(PathRef) const override {
|
|
return std::make_unique<GlobalScanningCounterProjectModules>(
|
|
scanningProjectModules(MockedCDBPtr, TFS), GlobalScanningCount);
|
|
}
|
|
|
|
unsigned getGlobalScanningCount() const { return GlobalScanningCount; }
|
|
|
|
private:
|
|
class MockClangCompilationDatabase : public tooling::CompilationDatabase {
|
|
public:
|
|
MockClangCompilationDatabase(MockDirectoryCompilationDatabase &MCDB)
|
|
: MCDB(MCDB) {}
|
|
|
|
std::vector<tooling::CompileCommand>
|
|
getCompileCommands(StringRef FilePath) const override {
|
|
std::optional<tooling::CompileCommand> Cmd =
|
|
MCDB.getCompileCommand(FilePath);
|
|
EXPECT_TRUE(Cmd);
|
|
return {*Cmd};
|
|
}
|
|
|
|
std::vector<std::string> getAllFiles() const override { return Files; }
|
|
|
|
void AddFile(StringRef File) { Files.push_back(File.str()); }
|
|
|
|
private:
|
|
MockDirectoryCompilationDatabase &MCDB;
|
|
std::vector<std::string> Files;
|
|
};
|
|
|
|
std::shared_ptr<MockClangCompilationDatabase> MockedCDBPtr;
|
|
const ThreadsafeFS &TFS;
|
|
|
|
mutable std::atomic<unsigned> GlobalScanningCount;
|
|
};
|
|
|
|
// Add files to the working testing directory and the compilation database.
|
|
void MockDirectoryCompilationDatabase::addFile(llvm::StringRef Path,
|
|
llvm::StringRef Contents) {
|
|
ASSERT_FALSE(llvm::sys::path::is_absolute(Path));
|
|
|
|
SmallString<256> AbsPath(Directory);
|
|
llvm::sys::path::append(AbsPath, Path);
|
|
|
|
ASSERT_FALSE(
|
|
llvm::sys::fs::create_directories(llvm::sys::path::parent_path(AbsPath)));
|
|
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream OS(AbsPath, EC);
|
|
ASSERT_FALSE(EC);
|
|
OS << Contents;
|
|
|
|
MockedCDBPtr->AddFile(Path);
|
|
}
|
|
|
|
class PrerequisiteModulesTests : public ::testing::Test {
|
|
protected:
|
|
void SetUp() override {
|
|
ASSERT_FALSE(llvm::sys::fs::createUniqueDirectory("modules-test", TestDir));
|
|
}
|
|
|
|
void TearDown() override {
|
|
ASSERT_FALSE(llvm::sys::fs::remove_directories(TestDir));
|
|
}
|
|
|
|
public:
|
|
// Get the absolute path for file specified by Path under testing working
|
|
// directory.
|
|
std::string getFullPath(llvm::StringRef Path) {
|
|
SmallString<128> Result(TestDir);
|
|
llvm::sys::path::append(Result, Path);
|
|
EXPECT_TRUE(llvm::sys::fs::exists(Result.str()));
|
|
return Result.str().str();
|
|
}
|
|
|
|
ParseInputs getInputs(llvm::StringRef FileName,
|
|
const GlobalCompilationDatabase &CDB) {
|
|
std::string FullPathName = getFullPath(FileName);
|
|
|
|
ParseInputs Inputs;
|
|
std::optional<tooling::CompileCommand> Cmd =
|
|
CDB.getCompileCommand(FullPathName);
|
|
EXPECT_TRUE(Cmd);
|
|
Inputs.CompileCommand = std::move(*Cmd);
|
|
Inputs.TFS = &FS;
|
|
|
|
if (auto Contents = FS.view(TestDir)->getBufferForFile(FullPathName))
|
|
Inputs.Contents = Contents->get()->getBuffer().str();
|
|
|
|
return Inputs;
|
|
}
|
|
|
|
SmallString<256> TestDir;
|
|
// FIXME: It will be better to use the MockFS if the scanning process and
|
|
// build module process doesn't depend on reading real IO.
|
|
RealThreadsafeFS FS;
|
|
|
|
DiagnosticConsumer DiagConsumer;
|
|
};
|
|
|
|
TEST_F(PrerequisiteModulesTests, NonModularTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("foo.h", R"cpp(
|
|
inline void foo() {}
|
|
)cpp");
|
|
|
|
CDB.addFile("NonModular.cpp", R"cpp(
|
|
#include "foo.h"
|
|
void use() {
|
|
foo();
|
|
}
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
// NonModular.cpp is not related to modules. So nothing should be built.
|
|
auto NonModularInfo =
|
|
Builder.buildPrerequisiteModulesFor(getFullPath("NonModular.cpp"), FS);
|
|
EXPECT_TRUE(NonModularInfo);
|
|
|
|
HeaderSearchOptions HSOpts;
|
|
NonModularInfo->adjustHeaderSearchOptions(HSOpts);
|
|
EXPECT_TRUE(HSOpts.PrebuiltModuleFiles.empty());
|
|
|
|
auto Invocation =
|
|
buildCompilerInvocation(getInputs("NonModular.cpp", CDB), DiagConsumer);
|
|
EXPECT_TRUE(NonModularInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ModuleWithoutDepTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("foo.h", R"cpp(
|
|
inline void foo() {}
|
|
)cpp");
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
module;
|
|
#include "foo.h"
|
|
export module M;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
auto MInfo = Builder.buildPrerequisiteModulesFor(getFullPath("M.cppm"), FS);
|
|
EXPECT_TRUE(MInfo);
|
|
|
|
// Nothing should be built since M doesn't dependent on anything.
|
|
HeaderSearchOptions HSOpts;
|
|
MInfo->adjustHeaderSearchOptions(HSOpts);
|
|
EXPECT_TRUE(HSOpts.PrebuiltModuleFiles.empty());
|
|
|
|
auto Invocation =
|
|
buildCompilerInvocation(getInputs("M.cppm", CDB), DiagConsumer);
|
|
EXPECT_TRUE(MInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ModuleWithArgumentPatch) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.ExtraClangFlags.push_back("-invalid-unknown-flag");
|
|
|
|
CDB.addFile("Dep.cppm", R"cpp(
|
|
export module Dep;
|
|
)cpp");
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
export module M;
|
|
import Dep;
|
|
)cpp");
|
|
|
|
// An invalid flag will break the module compilation and the
|
|
// getRequiredModules would return an empty array
|
|
auto ProjectModules = CDB.getProjectModules(getFullPath("M.cppm"));
|
|
EXPECT_TRUE(
|
|
ProjectModules->getRequiredModules(getFullPath("M.cppm")).empty());
|
|
|
|
// Set the mangler to filter out the invalid flag
|
|
ProjectModules->setCommandMangler(
|
|
[](tooling::CompileCommand &Command, PathRef) {
|
|
auto const It =
|
|
std::find(Command.CommandLine.begin(), Command.CommandLine.end(),
|
|
"-invalid-unknown-flag");
|
|
Command.CommandLine.erase(It);
|
|
});
|
|
|
|
// And now it returns a non-empty list of required modules since the
|
|
// compilation succeeded
|
|
EXPECT_FALSE(
|
|
ProjectModules->getRequiredModules(getFullPath("M.cppm")).empty());
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ModuleWithDepTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("foo.h", R"cpp(
|
|
inline void foo() {}
|
|
)cpp");
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
module;
|
|
#include "foo.h"
|
|
export module M;
|
|
)cpp");
|
|
|
|
CDB.addFile("N.cppm", R"cpp(
|
|
export module N;
|
|
import :Part;
|
|
import M;
|
|
)cpp");
|
|
|
|
CDB.addFile("N-part.cppm", R"cpp(
|
|
// Different module name with filename intentionally.
|
|
export module N:Part;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
auto NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo);
|
|
|
|
ParseInputs NInput = getInputs("N.cppm", CDB);
|
|
std::unique_ptr<CompilerInvocation> Invocation =
|
|
buildCompilerInvocation(NInput, DiagConsumer);
|
|
// Test that `PrerequisiteModules::canReuse` works basically.
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
{
|
|
// Check that
|
|
// `PrerequisiteModules::adjustHeaderSearchOptions(HeaderSearchOptions&)`
|
|
// can appending HeaderSearchOptions correctly.
|
|
HeaderSearchOptions HSOpts;
|
|
NInfo->adjustHeaderSearchOptions(HSOpts);
|
|
|
|
EXPECT_TRUE(HSOpts.PrebuiltModuleFiles.count("M"));
|
|
EXPECT_TRUE(HSOpts.PrebuiltModuleFiles.count("N:Part"));
|
|
}
|
|
|
|
{
|
|
// Check that
|
|
// `PrerequisiteModules::adjustHeaderSearchOptions(HeaderSearchOptions&)`
|
|
// can replace HeaderSearchOptions correctly.
|
|
HeaderSearchOptions HSOpts;
|
|
HSOpts.PrebuiltModuleFiles["M"] = "incorrect_path";
|
|
HSOpts.PrebuiltModuleFiles["N:Part"] = "incorrect_path";
|
|
NInfo->adjustHeaderSearchOptions(HSOpts);
|
|
|
|
EXPECT_TRUE(StringRef(HSOpts.PrebuiltModuleFiles["M"]).ends_with(".pcm"));
|
|
EXPECT_TRUE(
|
|
StringRef(HSOpts.PrebuiltModuleFiles["N:Part"]).ends_with(".pcm"));
|
|
}
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ReusabilityTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("foo.h", R"cpp(
|
|
inline void foo() {}
|
|
)cpp");
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
module;
|
|
#include "foo.h"
|
|
export module M;
|
|
)cpp");
|
|
|
|
CDB.addFile("N.cppm", R"cpp(
|
|
export module N;
|
|
import :Part;
|
|
import M;
|
|
)cpp");
|
|
|
|
CDB.addFile("N-part.cppm", R"cpp(
|
|
// Different module name with filename intentionally.
|
|
export module N:Part;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
auto NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo);
|
|
EXPECT_TRUE(NInfo);
|
|
|
|
ParseInputs NInput = getInputs("N.cppm", CDB);
|
|
std::unique_ptr<CompilerInvocation> Invocation =
|
|
buildCompilerInvocation(NInput, DiagConsumer);
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
// Test that we can still reuse the NInfo after we touch a unrelated file.
|
|
{
|
|
CDB.addFile("L.cppm", R"cpp(
|
|
module;
|
|
#include "foo.h"
|
|
export module L;
|
|
export int ll = 43;
|
|
)cpp");
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
CDB.addFile("bar.h", R"cpp(
|
|
inline void bar() {}
|
|
inline void bar(int) {}
|
|
)cpp");
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
}
|
|
|
|
// Test that we can't reuse the NInfo after we touch a related file.
|
|
{
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
module;
|
|
#include "foo.h"
|
|
export module M;
|
|
export int mm = 44;
|
|
)cpp");
|
|
EXPECT_FALSE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
CDB.addFile("foo.h", R"cpp(
|
|
inline void foo() {}
|
|
inline void foo(int) {}
|
|
)cpp");
|
|
EXPECT_FALSE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
}
|
|
|
|
CDB.addFile("N-part.cppm", R"cpp(
|
|
export module N:Part;
|
|
// Intentioned to make it uncompilable.
|
|
export int NPart = 4LIdjwldijaw
|
|
)cpp");
|
|
EXPECT_FALSE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo);
|
|
EXPECT_FALSE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
CDB.addFile("N-part.cppm", R"cpp(
|
|
export module N:Part;
|
|
export int NPart = 43;
|
|
)cpp");
|
|
EXPECT_TRUE(NInfo);
|
|
EXPECT_FALSE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
NInfo = Builder.buildPrerequisiteModulesFor(getFullPath("N.cppm"), FS);
|
|
EXPECT_TRUE(NInfo);
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
// Test that if we changed the modification time of the file, the module files
|
|
// info is still reusable if its content doesn't change.
|
|
CDB.addFile("N-part.cppm", R"cpp(
|
|
export module N:Part;
|
|
export int NPart = 43;
|
|
)cpp");
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
|
|
CDB.addFile("N.cppm", R"cpp(
|
|
export module N;
|
|
import :Part;
|
|
import M;
|
|
|
|
export int nn = 43;
|
|
)cpp");
|
|
// NInfo should be reusable after we change its content.
|
|
EXPECT_TRUE(NInfo->canReuse(*Invocation, FS.view(TestDir)));
|
|
}
|
|
|
|
// An End-to-End test for modules.
|
|
TEST_F(PrerequisiteModulesTests, ParsedASTTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("A.cppm", R"cpp(
|
|
export module A;
|
|
export void printA();
|
|
)cpp");
|
|
|
|
CDB.addFile("Use.cpp", R"cpp(
|
|
import A;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
ParseInputs Use = getInputs("Use.cpp", CDB);
|
|
Use.ModulesManager = &Builder;
|
|
|
|
std::unique_ptr<CompilerInvocation> CI =
|
|
buildCompilerInvocation(Use, DiagConsumer);
|
|
EXPECT_TRUE(CI);
|
|
|
|
auto Preamble =
|
|
buildPreamble(getFullPath("Use.cpp"), *CI, Use, /*InMemory=*/true,
|
|
/*Callback=*/nullptr);
|
|
EXPECT_TRUE(Preamble);
|
|
EXPECT_TRUE(Preamble->RequiredModules);
|
|
|
|
auto AST = ParsedAST::build(getFullPath("Use.cpp"), Use, std::move(CI), {},
|
|
Preamble);
|
|
EXPECT_TRUE(AST);
|
|
|
|
const NamedDecl &D = findDecl(*AST, "printA");
|
|
EXPECT_TRUE(D.isFromASTFile());
|
|
}
|
|
|
|
// An end to end test for code complete in modules
|
|
TEST_F(PrerequisiteModulesTests, CodeCompleteTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("A.cppm", R"cpp(
|
|
export module A;
|
|
export void printA();
|
|
)cpp");
|
|
|
|
llvm::StringLiteral UserContents = R"cpp(
|
|
import A;
|
|
void func() {
|
|
print^
|
|
}
|
|
)cpp";
|
|
|
|
CDB.addFile("Use.cpp", UserContents);
|
|
Annotations Test(UserContents);
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
ParseInputs Use = getInputs("Use.cpp", CDB);
|
|
Use.ModulesManager = &Builder;
|
|
|
|
std::unique_ptr<CompilerInvocation> CI =
|
|
buildCompilerInvocation(Use, DiagConsumer);
|
|
EXPECT_TRUE(CI);
|
|
|
|
auto Preamble =
|
|
buildPreamble(getFullPath("Use.cpp"), *CI, Use, /*InMemory=*/true,
|
|
/*Callback=*/nullptr);
|
|
EXPECT_TRUE(Preamble);
|
|
EXPECT_TRUE(Preamble->RequiredModules);
|
|
|
|
auto Result = codeComplete(getFullPath("Use.cpp"), Test.point(),
|
|
Preamble.get(), Use, {});
|
|
EXPECT_FALSE(Result.Completions.empty());
|
|
EXPECT_EQ(Result.Completions[0].Name, "printA");
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, SignatureHelpTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("A.cppm", R"cpp(
|
|
export module A;
|
|
export void printA(int a);
|
|
)cpp");
|
|
|
|
llvm::StringLiteral UserContents = R"cpp(
|
|
import A;
|
|
void func() {
|
|
printA(^);
|
|
}
|
|
)cpp";
|
|
|
|
CDB.addFile("Use.cpp", UserContents);
|
|
Annotations Test(UserContents);
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
ParseInputs Use = getInputs("Use.cpp", CDB);
|
|
Use.ModulesManager = &Builder;
|
|
|
|
std::unique_ptr<CompilerInvocation> CI =
|
|
buildCompilerInvocation(Use, DiagConsumer);
|
|
EXPECT_TRUE(CI);
|
|
|
|
auto Preamble =
|
|
buildPreamble(getFullPath("Use.cpp"), *CI, Use, /*InMemory=*/true,
|
|
/*Callback=*/nullptr);
|
|
EXPECT_TRUE(Preamble);
|
|
EXPECT_TRUE(Preamble->RequiredModules);
|
|
|
|
auto Result = signatureHelp(getFullPath("Use.cpp"), Test.point(),
|
|
*Preamble.get(), Use, MarkupKind::PlainText);
|
|
EXPECT_FALSE(Result.signatures.empty());
|
|
EXPECT_EQ(Result.signatures[0].label, "printA(int a) -> void");
|
|
EXPECT_EQ(Result.signatures[0].parameters[0].labelString, "int a");
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ReusablePrerequisiteModulesTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
export module M;
|
|
export int M = 43;
|
|
)cpp");
|
|
CDB.addFile("A.cppm", R"cpp(
|
|
export module A;
|
|
import M;
|
|
export int A = 43 + M;
|
|
)cpp");
|
|
CDB.addFile("B.cppm", R"cpp(
|
|
export module B;
|
|
import M;
|
|
export int B = 44 + M;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
auto AInfo = Builder.buildPrerequisiteModulesFor(getFullPath("A.cppm"), FS);
|
|
EXPECT_TRUE(AInfo);
|
|
auto BInfo = Builder.buildPrerequisiteModulesFor(getFullPath("B.cppm"), FS);
|
|
EXPECT_TRUE(BInfo);
|
|
HeaderSearchOptions HSOptsA(TestDir);
|
|
HeaderSearchOptions HSOptsB(TestDir);
|
|
AInfo->adjustHeaderSearchOptions(HSOptsA);
|
|
BInfo->adjustHeaderSearchOptions(HSOptsB);
|
|
|
|
EXPECT_FALSE(HSOptsA.PrebuiltModuleFiles.empty());
|
|
EXPECT_FALSE(HSOptsB.PrebuiltModuleFiles.empty());
|
|
|
|
// Check that we're reusing the module files.
|
|
EXPECT_EQ(HSOptsA.PrebuiltModuleFiles, HSOptsB.PrebuiltModuleFiles);
|
|
|
|
// Update M.cppm to check if the modules builder can update correctly.
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
export module M;
|
|
export constexpr int M = 43;
|
|
)cpp");
|
|
|
|
ParseInputs AUse = getInputs("A.cppm", CDB);
|
|
AUse.ModulesManager = &Builder;
|
|
std::unique_ptr<CompilerInvocation> AInvocation =
|
|
buildCompilerInvocation(AUse, DiagConsumer);
|
|
EXPECT_FALSE(AInfo->canReuse(*AInvocation, FS.view(TestDir)));
|
|
|
|
ParseInputs BUse = getInputs("B.cppm", CDB);
|
|
AUse.ModulesManager = &Builder;
|
|
std::unique_ptr<CompilerInvocation> BInvocation =
|
|
buildCompilerInvocation(BUse, DiagConsumer);
|
|
EXPECT_FALSE(BInfo->canReuse(*BInvocation, FS.view(TestDir)));
|
|
|
|
auto NewAInfo =
|
|
Builder.buildPrerequisiteModulesFor(getFullPath("A.cppm"), FS);
|
|
auto NewBInfo =
|
|
Builder.buildPrerequisiteModulesFor(getFullPath("B.cppm"), FS);
|
|
EXPECT_TRUE(NewAInfo);
|
|
EXPECT_TRUE(NewBInfo);
|
|
HeaderSearchOptions NewHSOptsA(TestDir);
|
|
HeaderSearchOptions NewHSOptsB(TestDir);
|
|
NewAInfo->adjustHeaderSearchOptions(NewHSOptsA);
|
|
NewBInfo->adjustHeaderSearchOptions(NewHSOptsB);
|
|
|
|
EXPECT_FALSE(NewHSOptsA.PrebuiltModuleFiles.empty());
|
|
EXPECT_FALSE(NewHSOptsB.PrebuiltModuleFiles.empty());
|
|
|
|
EXPECT_EQ(NewHSOptsA.PrebuiltModuleFiles, NewHSOptsB.PrebuiltModuleFiles);
|
|
// Check that we didn't reuse the old and stale module files.
|
|
EXPECT_NE(NewHSOptsA.PrebuiltModuleFiles, HSOptsA.PrebuiltModuleFiles);
|
|
}
|
|
|
|
TEST_F(PrerequisiteModulesTests, ScanningCacheTest) {
|
|
MockDirectoryCompilationDatabase CDB(TestDir, FS);
|
|
|
|
CDB.addFile("M.cppm", R"cpp(
|
|
export module M;
|
|
)cpp");
|
|
CDB.addFile("A.cppm", R"cpp(
|
|
export module A;
|
|
import M;
|
|
)cpp");
|
|
CDB.addFile("B.cppm", R"cpp(
|
|
export module B;
|
|
import M;
|
|
)cpp");
|
|
|
|
ModulesBuilder Builder(CDB);
|
|
|
|
Builder.buildPrerequisiteModulesFor(getFullPath("A.cppm"), FS);
|
|
Builder.buildPrerequisiteModulesFor(getFullPath("B.cppm"), FS);
|
|
EXPECT_EQ(CDB.getGlobalScanningCount(), 1u);
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace clang::clangd
|
|
|
|
#endif
|