forked from OSchip/llvm-project
539 lines
20 KiB
C++
539 lines
20 KiB
C++
//===-- GlobalCompilationDatabaseTests.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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "GlobalCompilationDatabase.h"
|
|
|
|
#include "Config.h"
|
|
#include "Matchers.h"
|
|
#include "TestFS.h"
|
|
#include "support/Path.h"
|
|
#include "support/ThreadsafeFS.h"
|
|
#include "clang/Tooling/CompilationDatabase.h"
|
|
#include "llvm/ADT/Optional.h"
|
|
#include "llvm/ADT/STLExtras.h"
|
|
#include "llvm/ADT/SmallString.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/FormatVariadic.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "gmock/gmock.h"
|
|
#include "gtest/gtest.h"
|
|
#include <chrono>
|
|
#include <fstream>
|
|
#include <string>
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
namespace {
|
|
using ::testing::AllOf;
|
|
using ::testing::Contains;
|
|
using ::testing::ElementsAre;
|
|
using ::testing::EndsWith;
|
|
using ::testing::HasSubstr;
|
|
using ::testing::IsEmpty;
|
|
using ::testing::Not;
|
|
using ::testing::StartsWith;
|
|
using ::testing::UnorderedElementsAre;
|
|
|
|
TEST(GlobalCompilationDatabaseTest, FallbackCommand) {
|
|
MockFS TFS;
|
|
DirectoryBasedGlobalCompilationDatabase DB(TFS);
|
|
auto Cmd = DB.getFallbackCommand(testPath("foo/bar.cc"));
|
|
EXPECT_EQ(Cmd.Directory, testPath("foo"));
|
|
EXPECT_THAT(Cmd.CommandLine, ElementsAre("clang", testPath("foo/bar.cc")));
|
|
EXPECT_EQ(Cmd.Output, "");
|
|
|
|
// .h files have unknown language, so they are parsed liberally as obj-c++.
|
|
Cmd = DB.getFallbackCommand(testPath("foo/bar.h"));
|
|
EXPECT_THAT(Cmd.CommandLine, ElementsAre("clang", "-xobjective-c++-header",
|
|
testPath("foo/bar.h")));
|
|
Cmd = DB.getFallbackCommand(testPath("foo/bar"));
|
|
EXPECT_THAT(Cmd.CommandLine, ElementsAre("clang", "-xobjective-c++-header",
|
|
testPath("foo/bar")));
|
|
}
|
|
|
|
static tooling::CompileCommand cmd(llvm::StringRef File, llvm::StringRef Arg) {
|
|
return tooling::CompileCommand(
|
|
testRoot(), File, {"clang", std::string(Arg), std::string(File)}, "");
|
|
}
|
|
|
|
class OverlayCDBTest : public ::testing::Test {
|
|
class BaseCDB : public GlobalCompilationDatabase {
|
|
public:
|
|
llvm::Optional<tooling::CompileCommand>
|
|
getCompileCommand(llvm::StringRef File) const override {
|
|
if (File == testPath("foo.cc"))
|
|
return cmd(File, "-DA=1");
|
|
return None;
|
|
}
|
|
|
|
tooling::CompileCommand
|
|
getFallbackCommand(llvm::StringRef File) const override {
|
|
return cmd(File, "-DA=2");
|
|
}
|
|
|
|
llvm::Optional<ProjectInfo> getProjectInfo(PathRef File) const override {
|
|
return ProjectInfo{testRoot()};
|
|
}
|
|
};
|
|
|
|
protected:
|
|
OverlayCDBTest() : Base(std::make_unique<BaseCDB>()) {}
|
|
std::unique_ptr<GlobalCompilationDatabase> Base;
|
|
};
|
|
|
|
TEST_F(OverlayCDBTest, GetCompileCommand) {
|
|
OverlayCDB CDB(Base.get());
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("foo.cc"))->CommandLine,
|
|
AllOf(Contains(testPath("foo.cc")), Contains("-DA=1")));
|
|
EXPECT_EQ(CDB.getCompileCommand(testPath("missing.cc")), llvm::None);
|
|
|
|
auto Override = cmd(testPath("foo.cc"), "-DA=3");
|
|
CDB.setCompileCommand(testPath("foo.cc"), Override);
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("foo.cc"))->CommandLine,
|
|
Contains("-DA=3"));
|
|
EXPECT_EQ(CDB.getCompileCommand(testPath("missing.cc")), llvm::None);
|
|
CDB.setCompileCommand(testPath("missing.cc"), Override);
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("missing.cc"))->CommandLine,
|
|
Contains("-DA=3"));
|
|
}
|
|
|
|
TEST_F(OverlayCDBTest, GetFallbackCommand) {
|
|
OverlayCDB CDB(Base.get(), {"-DA=4"});
|
|
EXPECT_THAT(CDB.getFallbackCommand(testPath("bar.cc")).CommandLine,
|
|
ElementsAre("clang", "-DA=2", testPath("bar.cc"), "-DA=4"));
|
|
}
|
|
|
|
TEST_F(OverlayCDBTest, NoBase) {
|
|
OverlayCDB CDB(nullptr, {"-DA=6"});
|
|
EXPECT_EQ(CDB.getCompileCommand(testPath("bar.cc")), None);
|
|
auto Override = cmd(testPath("bar.cc"), "-DA=5");
|
|
CDB.setCompileCommand(testPath("bar.cc"), Override);
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("bar.cc"))->CommandLine,
|
|
Contains("-DA=5"));
|
|
|
|
EXPECT_THAT(CDB.getFallbackCommand(testPath("foo.cc")).CommandLine,
|
|
ElementsAre("clang", testPath("foo.cc"), "-DA=6"));
|
|
}
|
|
|
|
TEST_F(OverlayCDBTest, Watch) {
|
|
OverlayCDB Inner(nullptr);
|
|
OverlayCDB Outer(&Inner);
|
|
|
|
std::vector<std::vector<std::string>> Changes;
|
|
auto Sub = Outer.watch([&](const std::vector<std::string> &ChangedFiles) {
|
|
Changes.push_back(ChangedFiles);
|
|
});
|
|
|
|
Inner.setCompileCommand("A.cpp", tooling::CompileCommand());
|
|
Outer.setCompileCommand("B.cpp", tooling::CompileCommand());
|
|
Inner.setCompileCommand("A.cpp", llvm::None);
|
|
Outer.setCompileCommand("C.cpp", llvm::None);
|
|
EXPECT_THAT(Changes, ElementsAre(ElementsAre("A.cpp"), ElementsAre("B.cpp"),
|
|
ElementsAre("A.cpp"), ElementsAre("C.cpp")));
|
|
}
|
|
|
|
TEST_F(OverlayCDBTest, Adjustments) {
|
|
OverlayCDB CDB(Base.get(), {"-DFallback"},
|
|
[](const std::vector<std::string> &Cmd, llvm::StringRef File) {
|
|
auto Ret = Cmd;
|
|
Ret.push_back(
|
|
("-DAdjust_" + llvm::sys::path::filename(File)).str());
|
|
return Ret;
|
|
});
|
|
// Command from underlying gets adjusted.
|
|
auto Cmd = CDB.getCompileCommand(testPath("foo.cc")).getValue();
|
|
EXPECT_THAT(Cmd.CommandLine, ElementsAre("clang", "-DA=1", testPath("foo.cc"),
|
|
"-DAdjust_foo.cc"));
|
|
|
|
// Command from overlay gets adjusted.
|
|
tooling::CompileCommand BarCommand;
|
|
BarCommand.Filename = testPath("bar.cc");
|
|
BarCommand.CommandLine = {"clang++", "-DB=1", testPath("bar.cc")};
|
|
CDB.setCompileCommand(testPath("bar.cc"), BarCommand);
|
|
Cmd = CDB.getCompileCommand(testPath("bar.cc")).getValue();
|
|
EXPECT_THAT(
|
|
Cmd.CommandLine,
|
|
ElementsAre("clang++", "-DB=1", testPath("bar.cc"), "-DAdjust_bar.cc"));
|
|
|
|
// Fallback gets adjusted.
|
|
Cmd = CDB.getFallbackCommand("baz.cc");
|
|
EXPECT_THAT(Cmd.CommandLine, ElementsAre("clang", "-DA=2", "baz.cc",
|
|
"-DFallback", "-DAdjust_baz.cc"));
|
|
}
|
|
|
|
TEST(GlobalCompilationDatabaseTest, DiscoveryWithNestedCDBs) {
|
|
const char *const CDBOuter =
|
|
R"cdb(
|
|
[
|
|
{
|
|
"file": "a.cc",
|
|
"command": "",
|
|
"directory": "{0}",
|
|
},
|
|
{
|
|
"file": "build/gen.cc",
|
|
"command": "",
|
|
"directory": "{0}",
|
|
},
|
|
{
|
|
"file": "build/gen2.cc",
|
|
"command": "",
|
|
"directory": "{0}",
|
|
}
|
|
]
|
|
)cdb";
|
|
const char *const CDBInner =
|
|
R"cdb(
|
|
[
|
|
{
|
|
"file": "gen.cc",
|
|
"command": "",
|
|
"directory": "{0}/build",
|
|
}
|
|
]
|
|
)cdb";
|
|
MockFS FS;
|
|
FS.Files[testPath("compile_commands.json")] =
|
|
llvm::formatv(CDBOuter, llvm::sys::path::convert_to_slash(testRoot()));
|
|
FS.Files[testPath("build/compile_commands.json")] =
|
|
llvm::formatv(CDBInner, llvm::sys::path::convert_to_slash(testRoot()));
|
|
FS.Files[testPath("foo/compile_flags.txt")] = "-DFOO";
|
|
|
|
// Note that gen2.cc goes missing with our following model, not sure this
|
|
// happens in practice though.
|
|
{
|
|
SCOPED_TRACE("Default ancestor scanning");
|
|
DirectoryBasedGlobalCompilationDatabase DB(FS);
|
|
std::vector<std::string> DiscoveredFiles;
|
|
auto Sub =
|
|
DB.watch([&DiscoveredFiles](const std::vector<std::string> Changes) {
|
|
DiscoveredFiles = Changes;
|
|
});
|
|
|
|
DB.getCompileCommand(testPath("build/../a.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, UnorderedElementsAre(AllOf(
|
|
EndsWith("a.cc"), Not(HasSubstr("..")))));
|
|
DiscoveredFiles.clear();
|
|
|
|
DB.getCompileCommand(testPath("build/gen.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, UnorderedElementsAre(EndsWith("gen.cc")));
|
|
}
|
|
|
|
{
|
|
SCOPED_TRACE("With config");
|
|
DirectoryBasedGlobalCompilationDatabase::Options Opts(FS);
|
|
Opts.ContextProvider = [&](llvm::StringRef Path) {
|
|
Config Cfg;
|
|
if (Path.endswith("a.cc")) {
|
|
// a.cc uses another directory's CDB, so it won't be discovered.
|
|
Cfg.CompileFlags.CDBSearch.Policy = Config::CDBSearchSpec::FixedDir;
|
|
Cfg.CompileFlags.CDBSearch.FixedCDBPath = testPath("foo");
|
|
} else if (Path.endswith("gen.cc")) {
|
|
// gen.cc has CDB search disabled, so it won't be discovered.
|
|
Cfg.CompileFlags.CDBSearch.Policy = Config::CDBSearchSpec::NoCDBSearch;
|
|
} else if (Path.endswith("gen2.cc")) {
|
|
// gen2.cc explicitly lists this directory, so it will be discovered.
|
|
Cfg.CompileFlags.CDBSearch.Policy = Config::CDBSearchSpec::FixedDir;
|
|
Cfg.CompileFlags.CDBSearch.FixedCDBPath = testRoot();
|
|
}
|
|
return Context::current().derive(Config::Key, std::move(Cfg));
|
|
};
|
|
DirectoryBasedGlobalCompilationDatabase DB(Opts);
|
|
std::vector<std::string> DiscoveredFiles;
|
|
auto Sub =
|
|
DB.watch([&DiscoveredFiles](const std::vector<std::string> Changes) {
|
|
DiscoveredFiles = Changes;
|
|
});
|
|
|
|
// Does not use the root CDB, so no broadcast.
|
|
auto Cmd = DB.getCompileCommand(testPath("build/../a.cc"));
|
|
ASSERT_TRUE(Cmd.hasValue());
|
|
EXPECT_THAT(Cmd->CommandLine, Contains("-DFOO")) << "a.cc uses foo/ CDB";
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, IsEmpty()) << "Root CDB not discovered yet";
|
|
|
|
// No special config for b.cc, so we trigger broadcast of the root CDB.
|
|
DB.getCompileCommand(testPath("b.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, ElementsAre(testPath("build/gen2.cc")));
|
|
DiscoveredFiles.clear();
|
|
|
|
// No CDB search so no discovery/broadcast triggered for build/ CDB.
|
|
DB.getCompileCommand(testPath("build/gen.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, IsEmpty());
|
|
}
|
|
|
|
{
|
|
SCOPED_TRACE("With custom compile commands dir");
|
|
DirectoryBasedGlobalCompilationDatabase::Options Opts(FS);
|
|
Opts.CompileCommandsDir = testRoot();
|
|
DirectoryBasedGlobalCompilationDatabase DB(Opts);
|
|
std::vector<std::string> DiscoveredFiles;
|
|
auto Sub =
|
|
DB.watch([&DiscoveredFiles](const std::vector<std::string> Changes) {
|
|
DiscoveredFiles = Changes;
|
|
});
|
|
|
|
DB.getCompileCommand(testPath("a.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles,
|
|
UnorderedElementsAre(EndsWith("a.cc"), EndsWith("gen.cc"),
|
|
EndsWith("gen2.cc")));
|
|
DiscoveredFiles.clear();
|
|
|
|
DB.getCompileCommand(testPath("build/gen.cc"));
|
|
ASSERT_TRUE(DB.blockUntilIdle(timeoutSeconds(10)));
|
|
EXPECT_THAT(DiscoveredFiles, IsEmpty());
|
|
}
|
|
}
|
|
|
|
TEST(GlobalCompilationDatabaseTest, BuildDir) {
|
|
MockFS FS;
|
|
auto Command = [&](llvm::StringRef Relative) {
|
|
DirectoryBasedGlobalCompilationDatabase::Options Opts(FS);
|
|
return DirectoryBasedGlobalCompilationDatabase(Opts)
|
|
.getCompileCommand(testPath(Relative))
|
|
.getValueOr(tooling::CompileCommand())
|
|
.CommandLine;
|
|
};
|
|
EXPECT_THAT(Command("x/foo.cc"), IsEmpty());
|
|
const char *const CDB =
|
|
R"cdb(
|
|
[
|
|
{
|
|
"file": "{0}/x/foo.cc",
|
|
"command": "clang -DXYZZY {0}/x/foo.cc",
|
|
"directory": "{0}",
|
|
},
|
|
{
|
|
"file": "{0}/bar.cc",
|
|
"command": "clang -DXYZZY {0}/bar.cc",
|
|
"directory": "{0}",
|
|
}
|
|
]
|
|
)cdb";
|
|
FS.Files[testPath("x/build/compile_commands.json")] =
|
|
llvm::formatv(CDB, llvm::sys::path::convert_to_slash(testRoot()));
|
|
EXPECT_THAT(Command("x/foo.cc"), Contains("-DXYZZY"));
|
|
EXPECT_THAT(Command("bar.cc"), IsEmpty())
|
|
<< "x/build/compile_flags.json only applicable to x/";
|
|
}
|
|
|
|
TEST(GlobalCompilationDatabaseTest, CompileFlagsDirectory) {
|
|
MockFS FS;
|
|
FS.Files[testPath("x/compile_flags.txt")] = "-DFOO";
|
|
DirectoryBasedGlobalCompilationDatabase CDB(FS);
|
|
auto Commands = CDB.getCompileCommand(testPath("x/y.cpp"));
|
|
ASSERT_TRUE(Commands.hasValue());
|
|
EXPECT_THAT(Commands.getValue().CommandLine, Contains("-DFOO"));
|
|
// Make sure we pick the right working directory.
|
|
EXPECT_EQ(testPath("x"), Commands.getValue().Directory);
|
|
}
|
|
|
|
MATCHER_P(hasArg, Flag, "") {
|
|
if (!arg.hasValue()) {
|
|
*result_listener << "command is null";
|
|
return false;
|
|
}
|
|
if (!llvm::is_contained(arg->CommandLine, Flag)) {
|
|
*result_listener << "flags are " << printArgv(arg->CommandLine);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
TEST(GlobalCompilationDatabaseTest, Config) {
|
|
MockFS FS;
|
|
FS.Files[testPath("x/compile_flags.txt")] = "-DX";
|
|
FS.Files[testPath("x/y/z/compile_flags.txt")] = "-DZ";
|
|
|
|
Config::CDBSearchSpec Spec;
|
|
DirectoryBasedGlobalCompilationDatabase::Options Opts(FS);
|
|
Opts.ContextProvider = [&](llvm::StringRef Path) {
|
|
Config C;
|
|
C.CompileFlags.CDBSearch = Spec;
|
|
return Context::current().derive(Config::Key, std::move(C));
|
|
};
|
|
DirectoryBasedGlobalCompilationDatabase CDB(Opts);
|
|
|
|
// Default ancestor behavior.
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("foo.cc")));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/foo.cc")), hasArg("-DX"));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/y/foo.cc")), hasArg("-DX"));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/y/z/foo.cc")), hasArg("-DZ"));
|
|
|
|
Spec.Policy = Config::CDBSearchSpec::NoCDBSearch;
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/y/foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/y/z/foo.cc")));
|
|
|
|
Spec.Policy = Config::CDBSearchSpec::FixedDir;
|
|
Spec.FixedCDBPath = testPath("w"); // doesn't exist
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/y/foo.cc")));
|
|
EXPECT_FALSE(CDB.getCompileCommand(testPath("x/y/z/foo.cc")));
|
|
|
|
Spec.FixedCDBPath = testPath("x/y/z");
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("foo.cc")), hasArg("-DZ"));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/foo.cc")), hasArg("-DZ"));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/y/foo.cc")), hasArg("-DZ"));
|
|
EXPECT_THAT(CDB.getCompileCommand(testPath("x/y/z/foo.cc")), hasArg("-DZ"));
|
|
}
|
|
|
|
TEST(GlobalCompilationDatabaseTest, NonCanonicalFilenames) {
|
|
OverlayCDB DB(nullptr);
|
|
std::vector<std::string> DiscoveredFiles;
|
|
auto Sub =
|
|
DB.watch([&DiscoveredFiles](const std::vector<std::string> Changes) {
|
|
DiscoveredFiles = Changes;
|
|
});
|
|
|
|
llvm::SmallString<128> Root(testRoot());
|
|
llvm::sys::path::append(Root, "build", "..", "a.cc");
|
|
DB.setCompileCommand(Root.str(), tooling::CompileCommand());
|
|
EXPECT_THAT(DiscoveredFiles, UnorderedElementsAre(testPath("a.cc")));
|
|
DiscoveredFiles.clear();
|
|
|
|
llvm::SmallString<128> File(testRoot());
|
|
llvm::sys::path::append(File, "blabla", "..", "a.cc");
|
|
|
|
EXPECT_TRUE(DB.getCompileCommand(File));
|
|
EXPECT_FALSE(DB.getProjectInfo(File));
|
|
}
|
|
|
|
TEST_F(OverlayCDBTest, GetProjectInfo) {
|
|
OverlayCDB DB(Base.get());
|
|
Path File = testPath("foo.cc");
|
|
Path Header = testPath("foo.h");
|
|
|
|
EXPECT_EQ(DB.getProjectInfo(File)->SourceRoot, testRoot());
|
|
EXPECT_EQ(DB.getProjectInfo(Header)->SourceRoot, testRoot());
|
|
|
|
// Shouldn't change after an override.
|
|
DB.setCompileCommand(File, tooling::CompileCommand());
|
|
EXPECT_EQ(DB.getProjectInfo(File)->SourceRoot, testRoot());
|
|
EXPECT_EQ(DB.getProjectInfo(Header)->SourceRoot, testRoot());
|
|
}
|
|
} // namespace
|
|
|
|
// Friend test has access to internals.
|
|
class DirectoryBasedGlobalCompilationDatabaseCacheTest
|
|
: public ::testing::Test {
|
|
protected:
|
|
std::shared_ptr<const tooling::CompilationDatabase>
|
|
lookupCDB(const DirectoryBasedGlobalCompilationDatabase &GDB,
|
|
llvm::StringRef Path,
|
|
std::chrono::steady_clock::time_point FreshTime) {
|
|
DirectoryBasedGlobalCompilationDatabase::CDBLookupRequest Req;
|
|
Req.FileName = Path;
|
|
Req.FreshTime = Req.FreshTimeMissing = FreshTime;
|
|
if (auto Result = GDB.lookupCDB(Req))
|
|
return std::move(Result->CDB);
|
|
return nullptr;
|
|
}
|
|
};
|
|
|
|
// Matches non-null CDBs which include the specified flag.
|
|
MATCHER_P2(hasFlag, Flag, Path, "") {
|
|
if (arg == nullptr)
|
|
return false;
|
|
auto Cmds = arg->getCompileCommands(Path);
|
|
if (Cmds.empty()) {
|
|
*result_listener << "yields no commands";
|
|
return false;
|
|
}
|
|
if (!llvm::is_contained(Cmds.front().CommandLine, Flag)) {
|
|
*result_listener << "flags are: " << printArgv(Cmds.front().CommandLine);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
auto hasFlag(llvm::StringRef Flag) {
|
|
return hasFlag(Flag, "mock_file_name.cc");
|
|
}
|
|
|
|
TEST_F(DirectoryBasedGlobalCompilationDatabaseCacheTest, Cacheable) {
|
|
MockFS FS;
|
|
auto Stale = std::chrono::steady_clock::now() - std::chrono::minutes(1);
|
|
auto Fresh = std::chrono::steady_clock::now() + std::chrono::hours(24);
|
|
|
|
DirectoryBasedGlobalCompilationDatabase GDB(FS);
|
|
FS.Files["compile_flags.txt"] = "-DROOT";
|
|
auto Root = lookupCDB(GDB, testPath("foo/test.cc"), Stale);
|
|
EXPECT_THAT(Root, hasFlag("-DROOT"));
|
|
|
|
// Add a compilation database to a subdirectory - CDB loaded.
|
|
FS.Files["foo/compile_flags.txt"] = "-DFOO";
|
|
EXPECT_EQ(Root, lookupCDB(GDB, testPath("foo/test.cc"), Stale))
|
|
<< "cache still valid";
|
|
auto Foo = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_THAT(Foo, hasFlag("-DFOO")) << "new cdb loaded";
|
|
EXPECT_EQ(Foo, lookupCDB(GDB, testPath("foo/test.cc"), Stale))
|
|
<< "new cdb in cache";
|
|
|
|
// Mtime changed, but no content change - CDB not reloaded.
|
|
++FS.Timestamps["foo/compile_flags.txt"];
|
|
auto FooAgain = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_EQ(Foo, FooAgain) << "Same content, read but not reloaded";
|
|
// Content changed, but not size or mtime - CDB not reloaded.
|
|
FS.Files["foo/compile_flags.txt"] = "-DBAR";
|
|
auto FooAgain2 = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_EQ(Foo, FooAgain2) << "Same filesize, change not detected";
|
|
// Mtime change forces a re-read, and we notice the different content.
|
|
++FS.Timestamps["foo/compile_flags.txt"];
|
|
auto Bar = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_THAT(Bar, hasFlag("-DBAR")) << "refreshed with mtime change";
|
|
|
|
// Size and content both change - CDB reloaded.
|
|
FS.Files["foo/compile_flags.txt"] = "-DFOOBAR";
|
|
EXPECT_EQ(Bar, lookupCDB(GDB, testPath("foo/test.cc"), Stale))
|
|
<< "cache still valid";
|
|
auto FooBar = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_THAT(FooBar, hasFlag("-DFOOBAR")) << "cdb reloaded";
|
|
|
|
// compile_commands.json takes precedence over compile_flags.txt.
|
|
FS.Files["foo/compile_commands.json"] =
|
|
llvm::formatv(R"json([{
|
|
"file": "{0}/foo/mock_file.cc",
|
|
"command": "clang -DBAZ mock_file.cc",
|
|
"directory": "{0}/foo",
|
|
}])json",
|
|
llvm::sys::path::convert_to_slash(testRoot()));
|
|
EXPECT_EQ(FooBar, lookupCDB(GDB, testPath("foo/test.cc"), Stale))
|
|
<< "cache still valid";
|
|
auto Baz = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_THAT(Baz, hasFlag("-DBAZ", testPath("foo/mock_file.cc")))
|
|
<< "compile_commands overrides compile_flags";
|
|
|
|
// Removing compile_commands.json reveals compile_flags.txt again.
|
|
// However this *does* cause a CDB reload (we cache only one CDB per dir).
|
|
FS.Files.erase("foo/compile_commands.json");
|
|
auto FoobarAgain = lookupCDB(GDB, testPath("foo/test.cc"), Fresh);
|
|
EXPECT_THAT(FoobarAgain, hasFlag("-DFOOBAR")) << "reloaded compile_flags";
|
|
EXPECT_NE(FoobarAgain, FooBar) << "CDB discarded (shadowed within directory)";
|
|
|
|
// Removing the directory's CDB leaves the parent CDB active.
|
|
// The parent CDB is *not* reloaded (we cache the CDB per-directory).
|
|
FS.Files.erase("foo/compile_flags.txt");
|
|
EXPECT_EQ(Root, lookupCDB(GDB, testPath("foo/test.cc"), Fresh))
|
|
<< "CDB retained (shadowed by another directory)";
|
|
}
|
|
|
|
} // namespace clangd
|
|
} // namespace clang
|