2011-02-16 05:30:27 +08:00
|
|
|
//===- unittests/Basic/FileMangerTest.cpp ------------ FileManger tests ---===//
|
2011-02-12 02:44:49 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2011-02-12 02:44:49 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-12-04 17:45:34 +08:00
|
|
|
#include "clang/Basic/FileManager.h"
|
2011-02-12 02:44:49 +08:00
|
|
|
#include "clang/Basic/FileSystemOptions.h"
|
|
|
|
#include "clang/Basic/FileSystemStatCache.h"
|
2015-03-02 05:36:40 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2017-03-28 17:18:05 +08:00
|
|
|
#include "llvm/Support/Path.h"
|
2018-10-10 21:27:25 +08:00
|
|
|
#include "llvm/Support/VirtualFileSystem.h"
|
2015-01-14 19:23:58 +08:00
|
|
|
#include "gtest/gtest.h"
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Used to create a fake file system for running the tests with such
|
|
|
|
// that the tests are not affected by the structure/contents of the
|
|
|
|
// file system on the machine running the tests.
|
|
|
|
class FakeStatCache : public FileSystemStatCache {
|
|
|
|
private:
|
|
|
|
// Maps a file/directory path to its desired stat result. Anything
|
|
|
|
// not in this map is considered to not exist in the file system.
|
2019-03-05 10:27:12 +08:00
|
|
|
llvm::StringMap<llvm::vfs::Status, llvm::BumpPtrAllocator> StatCalls;
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2020-10-15 22:36:00 +08:00
|
|
|
void InjectFileOrDirectory(const char *Path, ino_t INode, bool IsFile,
|
|
|
|
const char *StatPath) {
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifndef _WIN32
|
2017-03-28 17:18:05 +08:00
|
|
|
SmallString<128> NormalizedPath(Path);
|
|
|
|
llvm::sys::path::native(NormalizedPath);
|
|
|
|
Path = NormalizedPath.c_str();
|
2020-10-15 22:36:00 +08:00
|
|
|
|
|
|
|
SmallString<128> NormalizedStatPath;
|
|
|
|
if (StatPath) {
|
|
|
|
NormalizedStatPath = StatPath;
|
|
|
|
llvm::sys::path::native(NormalizedStatPath);
|
|
|
|
StatPath = NormalizedStatPath.c_str();
|
|
|
|
}
|
2017-03-28 17:18:05 +08:00
|
|
|
#endif
|
|
|
|
|
2020-10-15 22:36:00 +08:00
|
|
|
if (!StatPath)
|
|
|
|
StatPath = Path;
|
|
|
|
|
2019-03-05 10:27:12 +08:00
|
|
|
auto fileType = IsFile ?
|
|
|
|
llvm::sys::fs::file_type::regular_file :
|
|
|
|
llvm::sys::fs::file_type::directory_file;
|
2020-10-15 22:36:00 +08:00
|
|
|
llvm::vfs::Status Status(StatPath, llvm::sys::fs::UniqueID(1, INode),
|
2019-03-05 10:27:12 +08:00
|
|
|
/*MTime*/{}, /*User*/0, /*Group*/0,
|
|
|
|
/*Size*/0, fileType,
|
|
|
|
llvm::sys::fs::perms::all_all);
|
|
|
|
StatCalls[Path] = Status;
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Inject a file with the given inode value to the fake file system.
|
2020-10-15 22:36:00 +08:00
|
|
|
void InjectFile(const char *Path, ino_t INode,
|
|
|
|
const char *StatPath = nullptr) {
|
|
|
|
InjectFileOrDirectory(Path, INode, /*IsFile=*/true, StatPath);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Inject a directory with the given inode value to the fake file system.
|
|
|
|
void InjectDirectory(const char *Path, ino_t INode) {
|
2020-10-15 22:36:00 +08:00
|
|
|
InjectFileOrDirectory(Path, INode, /*IsFile=*/false, nullptr);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Implement FileSystemStatCache::getStat().
|
2019-04-17 02:00:43 +08:00
|
|
|
std::error_code getStat(StringRef Path, llvm::vfs::Status &Status,
|
|
|
|
bool isFile,
|
|
|
|
std::unique_ptr<llvm::vfs::File> *F,
|
|
|
|
llvm::vfs::FileSystem &FS) override {
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifndef _WIN32
|
2017-03-28 17:18:05 +08:00
|
|
|
SmallString<128> NormalizedPath(Path);
|
|
|
|
llvm::sys::path::native(NormalizedPath);
|
|
|
|
Path = NormalizedPath.c_str();
|
|
|
|
#endif
|
|
|
|
|
2011-02-12 02:44:49 +08:00
|
|
|
if (StatCalls.count(Path) != 0) {
|
2019-03-05 10:27:12 +08:00
|
|
|
Status = StatCalls[Path];
|
2019-04-17 02:00:43 +08:00
|
|
|
return std::error_code();
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
2019-04-17 02:00:43 +08:00
|
|
|
return std::make_error_code(std::errc::no_such_file_or_directory);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// The test fixture.
|
|
|
|
class FileManagerTest : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
FileManagerTest() : manager(options) {
|
|
|
|
}
|
|
|
|
|
|
|
|
FileSystemOptions options;
|
|
|
|
FileManager manager;
|
|
|
|
};
|
|
|
|
|
|
|
|
// When a virtual file is added, its getDir() field is set correctly
|
|
|
|
// (not NULL, correct name).
|
|
|
|
TEST_F(FileManagerTest, getVirtualFileSetsTheDirFieldCorrectly) {
|
|
|
|
const FileEntry *file = manager.getVirtualFile("foo.cpp", 42, 0);
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(file != nullptr);
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
const DirectoryEntry *dir = file->getDir();
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(dir != nullptr);
|
2016-10-11 15:31:29 +08:00
|
|
|
EXPECT_EQ(".", dir->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
file = manager.getVirtualFile("x/y/z.cpp", 42, 0);
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(file != nullptr);
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
dir = file->getDir();
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(dir != nullptr);
|
2016-10-11 15:31:29 +08:00
|
|
|
EXPECT_EQ("x/y", dir->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Before any virtual file is added, no virtual directory exists.
|
|
|
|
TEST_F(FileManagerTest, NoVirtualDirectoryExistsBeforeAVirtualFileIsAdded) {
|
|
|
|
// An empty FakeStatCache causes all stat calls made by the
|
|
|
|
// FileManager to report "file/directory doesn't exist". This
|
|
|
|
// avoids the possibility of the result of this test being affected
|
|
|
|
// by what's in the real file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
manager.setStatCache(std::make_unique<FakeStatCache>());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
ASSERT_FALSE(manager.getDirectory("virtual/dir/foo"));
|
|
|
|
ASSERT_FALSE(manager.getDirectory("virtual/dir"));
|
|
|
|
ASSERT_FALSE(manager.getDirectory("virtual"));
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// When a virtual file is added, all of its ancestors should be created.
|
|
|
|
TEST_F(FileManagerTest, getVirtualFileCreatesDirectoryEntriesForAncestors) {
|
|
|
|
// Fake an empty real file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
manager.setStatCache(std::make_unique<FakeStatCache>());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
manager.getVirtualFile("virtual/dir/bar.h", 100, 0);
|
2019-08-02 05:31:49 +08:00
|
|
|
ASSERT_FALSE(manager.getDirectory("virtual/dir/foo"));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto dir = manager.getDirectory("virtual/dir");
|
|
|
|
ASSERT_TRUE(dir);
|
|
|
|
EXPECT_EQ("virtual/dir", (*dir)->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
dir = manager.getDirectory("virtual");
|
2019-08-02 05:31:49 +08:00
|
|
|
ASSERT_TRUE(dir);
|
|
|
|
EXPECT_EQ("virtual", (*dir)->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// getFile() returns non-NULL if a real file exists at the given path.
|
|
|
|
TEST_F(FileManagerTest, getFileReturnsValidFileEntryForExistingRealFile) {
|
|
|
|
// Inject fake files into the file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2011-02-12 02:44:49 +08:00
|
|
|
statCache->InjectDirectory("/tmp", 42);
|
|
|
|
statCache->InjectFile("/tmp/test", 43);
|
2013-07-29 23:47:24 +08:00
|
|
|
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifdef _WIN32
|
2013-07-29 23:47:24 +08:00
|
|
|
const char *DirName = "C:.";
|
|
|
|
const char *FileName = "C:test";
|
|
|
|
statCache->InjectDirectory(DirName, 44);
|
|
|
|
statCache->InjectFile(FileName, 45);
|
|
|
|
#endif
|
|
|
|
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto file = manager.getFile("/tmp/test");
|
|
|
|
ASSERT_TRUE(file);
|
|
|
|
ASSERT_TRUE((*file)->isValid());
|
|
|
|
EXPECT_EQ("/tmp/test", (*file)->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
const DirectoryEntry *dir = (*file)->getDir();
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(dir != nullptr);
|
2016-10-11 15:31:29 +08:00
|
|
|
EXPECT_EQ("/tmp", dir->getName());
|
2013-07-29 23:47:24 +08:00
|
|
|
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifdef _WIN32
|
2013-07-29 23:47:24 +08:00
|
|
|
file = manager.getFile(FileName);
|
2019-08-02 05:31:49 +08:00
|
|
|
ASSERT_TRUE(file);
|
2013-07-29 23:47:24 +08:00
|
|
|
|
2019-08-02 05:58:56 +08:00
|
|
|
dir = (*file)->getDir();
|
2013-07-29 23:47:24 +08:00
|
|
|
ASSERT_TRUE(dir != NULL);
|
2016-10-11 15:31:29 +08:00
|
|
|
EXPECT_EQ(DirName, dir->getName());
|
2013-07-29 23:47:24 +08:00
|
|
|
#endif
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// getFile() returns non-NULL if a virtual file exists at the given path.
|
|
|
|
TEST_F(FileManagerTest, getFileReturnsValidFileEntryForExistingVirtualFile) {
|
|
|
|
// Fake an empty real file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
manager.setStatCache(std::make_unique<FakeStatCache>());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
manager.getVirtualFile("virtual/dir/bar.h", 100, 0);
|
2019-08-02 05:31:49 +08:00
|
|
|
auto file = manager.getFile("virtual/dir/bar.h");
|
|
|
|
ASSERT_TRUE(file);
|
|
|
|
ASSERT_TRUE((*file)->isValid());
|
|
|
|
EXPECT_EQ("virtual/dir/bar.h", (*file)->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
const DirectoryEntry *dir = (*file)->getDir();
|
2014-06-08 16:38:12 +08:00
|
|
|
ASSERT_TRUE(dir != nullptr);
|
2016-10-11 15:31:29 +08:00
|
|
|
EXPECT_EQ("virtual/dir", dir->getName());
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// getFile() returns different FileEntries for different paths when
|
|
|
|
// there's no aliasing.
|
|
|
|
TEST_F(FileManagerTest, getFileReturnsDifferentFileEntriesForDifferentFiles) {
|
|
|
|
// Inject two fake files into the file system. Different inodes
|
|
|
|
// mean the files are not symlinked together.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2011-02-12 02:44:49 +08:00
|
|
|
statCache->InjectDirectory(".", 41);
|
|
|
|
statCache->InjectFile("foo.cpp", 42);
|
|
|
|
statCache->InjectFile("bar.cpp", 43);
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto fileFoo = manager.getFile("foo.cpp");
|
|
|
|
auto fileBar = manager.getFile("bar.cpp");
|
|
|
|
ASSERT_TRUE(fileFoo);
|
|
|
|
ASSERT_TRUE((*fileFoo)->isValid());
|
|
|
|
ASSERT_TRUE(fileBar);
|
|
|
|
ASSERT_TRUE((*fileBar)->isValid());
|
|
|
|
EXPECT_NE(*fileFoo, *fileBar);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
2019-08-02 05:31:56 +08:00
|
|
|
// getFile() returns an error if neither a real file nor a virtual file
|
2011-02-12 02:44:49 +08:00
|
|
|
// exists at the given path.
|
2019-08-02 05:31:56 +08:00
|
|
|
TEST_F(FileManagerTest, getFileReturnsErrorForNonexistentFile) {
|
2011-02-12 02:44:49 +08:00
|
|
|
// Inject a fake foo.cpp into the file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2011-02-12 02:44:49 +08:00
|
|
|
statCache->InjectDirectory(".", 41);
|
|
|
|
statCache->InjectFile("foo.cpp", 42);
|
2019-08-02 05:50:16 +08:00
|
|
|
statCache->InjectDirectory("MyDirectory", 49);
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
// Create a virtual bar.cpp file.
|
|
|
|
manager.getVirtualFile("bar.cpp", 200, 0);
|
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto file = manager.getFile("xyz.txt");
|
|
|
|
ASSERT_FALSE(file);
|
2019-08-02 05:31:56 +08:00
|
|
|
ASSERT_EQ(file.getError(), std::errc::no_such_file_or_directory);
|
|
|
|
|
|
|
|
auto readingDirAsFile = manager.getFile("MyDirectory");
|
|
|
|
ASSERT_FALSE(readingDirAsFile);
|
|
|
|
ASSERT_EQ(readingDirAsFile.getError(), std::errc::is_a_directory);
|
|
|
|
|
|
|
|
auto readingFileAsDir = manager.getDirectory("foo.cpp");
|
|
|
|
ASSERT_FALSE(readingFileAsDir);
|
|
|
|
ASSERT_EQ(readingFileAsDir.getError(), std::errc::not_a_directory);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// The following tests apply to Unix-like system only.
|
|
|
|
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifndef _WIN32
|
2011-02-12 02:44:49 +08:00
|
|
|
|
|
|
|
// getFile() returns the same FileEntry for real files that are aliases.
|
|
|
|
TEST_F(FileManagerTest, getFileReturnsSameFileEntryForAliasedRealFiles) {
|
|
|
|
// Inject two real files with the same inode.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2011-02-12 02:44:49 +08:00
|
|
|
statCache->InjectDirectory("abc", 41);
|
|
|
|
statCache->InjectFile("abc/foo.cpp", 42);
|
|
|
|
statCache->InjectFile("abc/bar.cpp", 42);
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto f1 = manager.getFile("abc/foo.cpp");
|
|
|
|
auto f2 = manager.getFile("abc/bar.cpp");
|
|
|
|
|
|
|
|
EXPECT_EQ(f1 ? *f1 : nullptr,
|
|
|
|
f2 ? *f2 : nullptr);
|
2020-10-15 22:36:00 +08:00
|
|
|
|
|
|
|
// Check that getFileRef also does the right thing.
|
|
|
|
auto r1 = manager.getFileRef("abc/foo.cpp");
|
|
|
|
auto r2 = manager.getFileRef("abc/bar.cpp");
|
|
|
|
ASSERT_FALSE(!r1);
|
|
|
|
ASSERT_FALSE(!r2);
|
|
|
|
|
|
|
|
EXPECT_EQ("abc/foo.cpp", r1->getName());
|
|
|
|
EXPECT_EQ("abc/bar.cpp", r2->getName());
|
|
|
|
EXPECT_EQ((f1 ? *f1 : nullptr), &r1->getFileEntry());
|
|
|
|
EXPECT_EQ((f2 ? *f2 : nullptr), &r2->getFileEntry());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(FileManagerTest, getFileRefReturnsCorrectNameForDifferentStatPath) {
|
|
|
|
// Inject files with the same inode, but where some files have a stat that
|
|
|
|
// gives a different name. This is adding coverage for weird stat behaviour
|
|
|
|
// triggered by the RedirectingFileSystem that FileManager::getFileRef has
|
|
|
|
// special logic for.
|
|
|
|
auto StatCache = std::make_unique<FakeStatCache>();
|
|
|
|
StatCache->InjectDirectory("dir", 40);
|
|
|
|
StatCache->InjectFile("dir/f1.cpp", 41);
|
|
|
|
StatCache->InjectFile("dir/f1-alias.cpp", 41, "dir/f1.cpp");
|
|
|
|
StatCache->InjectFile("dir/f2.cpp", 42);
|
|
|
|
StatCache->InjectFile("dir/f2-alias.cpp", 42, "dir/f2.cpp");
|
2020-10-15 23:39:07 +08:00
|
|
|
|
|
|
|
// This unintuitive rename-the-file-on-stat behaviour supports how the
|
|
|
|
// RedirectingFileSystem VFS layer responds to stats. However, even if you
|
|
|
|
// have two layers, you should only get a single filename back. As such the
|
|
|
|
// following stat cache behaviour is not supported (the correct stat entry
|
|
|
|
// for a double-redirection would be "dir/f1.cpp") and the getFileRef below
|
|
|
|
// should assert.
|
|
|
|
StatCache->InjectFile("dir/f1-alias-alias.cpp", 41, "dir/f1-alias.cpp");
|
|
|
|
|
2020-10-15 22:36:00 +08:00
|
|
|
manager.setStatCache(std::move(StatCache));
|
|
|
|
|
2020-10-15 23:39:07 +08:00
|
|
|
// With F1, test accessing the non-redirected name first.
|
2020-10-15 22:36:00 +08:00
|
|
|
auto F1 = manager.getFileRef("dir/f1.cpp");
|
|
|
|
auto F1Alias = manager.getFileRef("dir/f1-alias.cpp");
|
|
|
|
auto F1Alias2 = manager.getFileRef("dir/f1-alias.cpp");
|
|
|
|
ASSERT_FALSE(!F1);
|
|
|
|
ASSERT_FALSE(!F1Alias);
|
|
|
|
ASSERT_FALSE(!F1Alias2);
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1->getName());
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1->getFileEntry().getName());
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1Alias->getName());
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1Alias2->getName());
|
|
|
|
EXPECT_EQ(&F1->getFileEntry(), &F1Alias->getFileEntry());
|
|
|
|
EXPECT_EQ(&F1->getFileEntry(), &F1Alias2->getFileEntry());
|
|
|
|
|
2020-10-15 23:39:07 +08:00
|
|
|
#if !defined(NDEBUG) && GTEST_HAS_DEATH_TEST
|
|
|
|
EXPECT_DEATH((void)manager.getFileRef("dir/f1-alias-alias.cpp"),
|
|
|
|
"filename redirected to a non-canonical filename?");
|
|
|
|
#endif
|
|
|
|
|
2020-10-15 22:36:00 +08:00
|
|
|
// With F2, test accessing the redirected name first.
|
|
|
|
auto F2Alias = manager.getFileRef("dir/f2-alias.cpp");
|
|
|
|
auto F2 = manager.getFileRef("dir/f2.cpp");
|
|
|
|
auto F2Alias2 = manager.getFileRef("dir/f2-alias.cpp");
|
|
|
|
ASSERT_FALSE(!F2);
|
|
|
|
ASSERT_FALSE(!F2Alias);
|
|
|
|
ASSERT_FALSE(!F2Alias2);
|
|
|
|
EXPECT_EQ("dir/f2.cpp", F2->getName());
|
|
|
|
EXPECT_EQ("dir/f2.cpp", F2->getFileEntry().getName());
|
|
|
|
EXPECT_EQ("dir/f2.cpp", F2Alias->getName());
|
|
|
|
EXPECT_EQ("dir/f2.cpp", F2Alias2->getName());
|
|
|
|
EXPECT_EQ(&F2->getFileEntry(), &F2Alias->getFileEntry());
|
|
|
|
EXPECT_EQ(&F2->getFileEntry(), &F2Alias2->getFileEntry());
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// getFile() returns the same FileEntry for virtual files that have
|
|
|
|
// corresponding real files that are aliases.
|
|
|
|
TEST_F(FileManagerTest, getFileReturnsSameFileEntryForAliasedVirtualFiles) {
|
|
|
|
// Inject two real files with the same inode.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2011-02-12 02:44:49 +08:00
|
|
|
statCache->InjectDirectory("abc", 41);
|
|
|
|
statCache->InjectFile("abc/foo.cpp", 42);
|
|
|
|
statCache->InjectFile("abc/bar.cpp", 42);
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2017-03-28 17:18:05 +08:00
|
|
|
ASSERT_TRUE(manager.getVirtualFile("abc/foo.cpp", 100, 0)->isValid());
|
|
|
|
ASSERT_TRUE(manager.getVirtualFile("abc/bar.cpp", 200, 0)->isValid());
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2019-08-02 05:31:49 +08:00
|
|
|
auto f1 = manager.getFile("abc/foo.cpp");
|
|
|
|
auto f2 = manager.getFile("abc/bar.cpp");
|
|
|
|
|
|
|
|
EXPECT_EQ(f1 ? *f1 : nullptr,
|
|
|
|
f2 ? *f2 : nullptr);
|
2011-02-12 02:44:49 +08:00
|
|
|
}
|
|
|
|
|
Reapply "FileManager: Improve the FileEntryRef API and customize its OptionalStorage"
This reverts commit 940d0a310dca31ae97080b068cef92eadfee6367,
effectively reapplying 84e8257937ec6a332aa0b688f4dce57016516ffd, after
working around the compile errors on the bots that I wasn't seeing
locally. I removed the `constexpr` from `OptionalStorage<FileEntryRef>`
that I had cargo-culted from the generic version, since `FileEntryRef`
isn't relevant in `constexpr` contexts anyway.
The original commit message follows:
Make a few changes to the `FileEntryRef` API in preparation for
propagating it enough to remove `FileEntry::getName()`.
- Allow `FileEntryRef` to degrade implicitly to `const FileEntry*`. This
allows functions currently returning `const FileEntry *` to be updated
to return `FileEntryRef` without requiring all callers to be updated
in the same patch. This helps avoid both (a) massive patches where
many fields and locals are updated simultaneously and (b) noisy
incremental patches where the first patch adds `getFileEntry()` at
call sites and the second patch removes it. (Once `FileEntryRef` is
everywhere, we should remove this API.)
- Change `operator==` to compare the underlying `FileEntry*`, ignoring
any difference in the spelling of the filename. There were 0 users of
the existing function because it's not useful. In case comparing the
exact named reference becomes important, add/test `isSameRef`.
- Add `==` comparisons between `FileEntryRef` and `const FileEntry *`
(compares the `FileEntry*`).
- Customize `OptionalStorage<FileEntryRef>` to be pointer-sized. Add
a private constructor that initializes with `nullptr` and specialize
`OptionalStorage` to use it. This unblocks updating fields in
size-sensitive data structures that currently use `const FileEntry *`.
- Add `OptionalFileEntryRefDegradesToFileEntryPtr`, a wrapper around
`Optional<FileEntryRef>` that degrades to `const FileEntry*`. This
facilitates future incremental patches, like the same operator on
`FileEntryRef`. (Once `FileEntryRef` is everywhere, we should remove
this class.)
- Remove the unncessary `const` from the by-value return of
`FileEntryRef::getName`.
- Delete the unused function `FileEntry::isOpenForTests`.
Note that there are still `FileEntry` APIs that aren't wrapped and I
plan to deal with these separately / incrementally, as they are needed.
Differential Revision: https://reviews.llvm.org/D89834
2020-10-31 02:08:19 +08:00
|
|
|
TEST_F(FileManagerTest, getFileRefEquality) {
|
|
|
|
auto StatCache = std::make_unique<FakeStatCache>();
|
|
|
|
StatCache->InjectDirectory("dir", 40);
|
|
|
|
StatCache->InjectFile("dir/f1.cpp", 41);
|
|
|
|
StatCache->InjectFile("dir/f1-also.cpp", 41);
|
|
|
|
StatCache->InjectFile("dir/f1-redirect.cpp", 41, "dir/f1.cpp");
|
|
|
|
StatCache->InjectFile("dir/f2.cpp", 42);
|
|
|
|
manager.setStatCache(std::move(StatCache));
|
|
|
|
|
|
|
|
auto F1 = manager.getFileRef("dir/f1.cpp");
|
|
|
|
auto F1Again = manager.getFileRef("dir/f1.cpp");
|
|
|
|
auto F1Also = manager.getFileRef("dir/f1-also.cpp");
|
|
|
|
auto F1Redirect = manager.getFileRef("dir/f1-redirect.cpp");
|
|
|
|
auto F2 = manager.getFileRef("dir/f2.cpp");
|
|
|
|
|
|
|
|
// Check Expected<FileEntryRef> for error.
|
|
|
|
ASSERT_FALSE(!F1);
|
|
|
|
ASSERT_FALSE(!F1Also);
|
|
|
|
ASSERT_FALSE(!F1Again);
|
|
|
|
ASSERT_FALSE(!F1Redirect);
|
|
|
|
ASSERT_FALSE(!F2);
|
|
|
|
|
|
|
|
// Check names.
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1->getName());
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1Again->getName());
|
|
|
|
EXPECT_EQ("dir/f1-also.cpp", F1Also->getName());
|
|
|
|
EXPECT_EQ("dir/f1.cpp", F1Redirect->getName());
|
|
|
|
EXPECT_EQ("dir/f2.cpp", F2->getName());
|
|
|
|
|
|
|
|
// Compare against FileEntry*.
|
|
|
|
EXPECT_EQ(&F1->getFileEntry(), *F1);
|
|
|
|
EXPECT_EQ(*F1, &F1->getFileEntry());
|
|
|
|
EXPECT_NE(&F2->getFileEntry(), *F1);
|
|
|
|
EXPECT_NE(*F1, &F2->getFileEntry());
|
|
|
|
|
|
|
|
// Compare using ==.
|
|
|
|
EXPECT_EQ(*F1, *F1Also);
|
|
|
|
EXPECT_EQ(*F1, *F1Again);
|
|
|
|
EXPECT_EQ(*F1, *F1Redirect);
|
|
|
|
EXPECT_EQ(*F1Also, *F1Redirect);
|
|
|
|
EXPECT_NE(*F2, *F1);
|
|
|
|
EXPECT_NE(*F2, *F1Also);
|
|
|
|
EXPECT_NE(*F2, *F1Again);
|
|
|
|
EXPECT_NE(*F2, *F1Redirect);
|
|
|
|
|
|
|
|
// Compare using isSameRef.
|
|
|
|
EXPECT_TRUE(F1->isSameRef(*F1Again));
|
|
|
|
EXPECT_TRUE(F1->isSameRef(*F1Redirect));
|
|
|
|
EXPECT_FALSE(F1->isSameRef(*F1Also));
|
|
|
|
EXPECT_FALSE(F1->isSameRef(*F2));
|
|
|
|
}
|
|
|
|
|
2017-03-28 17:18:05 +08:00
|
|
|
// getFile() Should return the same entry as getVirtualFile if the file actually
|
|
|
|
// is a virtual file, even if the name is not exactly the same (but is after
|
|
|
|
// normalisation done by the file system, like on Windows). This can be checked
|
2018-07-24 11:34:15 +08:00
|
|
|
// here by checking the size.
|
2017-03-28 17:18:05 +08:00
|
|
|
TEST_F(FileManagerTest, getVirtualFileWithDifferentName) {
|
|
|
|
// Inject fake files into the file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2017-03-28 17:18:05 +08:00
|
|
|
statCache->InjectDirectory("c:\\tmp", 42);
|
|
|
|
statCache->InjectFile("c:\\tmp\\test", 43);
|
|
|
|
|
2018-12-22 03:33:09 +08:00
|
|
|
manager.setStatCache(std::move(statCache));
|
2017-03-28 17:18:05 +08:00
|
|
|
|
|
|
|
// Inject the virtual file:
|
|
|
|
const FileEntry *file1 = manager.getVirtualFile("c:\\tmp\\test", 123, 1);
|
|
|
|
ASSERT_TRUE(file1 != nullptr);
|
|
|
|
ASSERT_TRUE(file1->isValid());
|
|
|
|
EXPECT_EQ(43U, file1->getUniqueID().getFile());
|
|
|
|
EXPECT_EQ(123, file1->getSize());
|
|
|
|
|
|
|
|
// Lookup the virtual file with a different name:
|
2019-08-02 05:31:49 +08:00
|
|
|
auto file2 = manager.getFile("c:/tmp/test", 100, 1);
|
|
|
|
ASSERT_TRUE(file2);
|
|
|
|
ASSERT_TRUE((*file2)->isValid());
|
2017-03-28 17:18:05 +08:00
|
|
|
// Check that it's the same UFE:
|
2019-08-02 05:31:49 +08:00
|
|
|
EXPECT_EQ(file1, *file2);
|
|
|
|
EXPECT_EQ(43U, (*file2)->getUniqueID().getFile());
|
2017-03-28 17:18:05 +08:00
|
|
|
// Check that the contents of the UFE are not overwritten by the entry in the
|
|
|
|
// filesystem:
|
2019-08-02 05:31:49 +08:00
|
|
|
EXPECT_EQ(123, (*file2)->getSize());
|
2017-03-28 17:18:05 +08:00
|
|
|
}
|
|
|
|
|
2018-04-28 03:11:14 +08:00
|
|
|
#endif // !_WIN32
|
2011-02-12 02:44:49 +08:00
|
|
|
|
2017-08-02 15:25:24 +08:00
|
|
|
TEST_F(FileManagerTest, makeAbsoluteUsesVFS) {
|
|
|
|
SmallString<64> CustomWorkingDir;
|
2018-04-28 03:11:14 +08:00
|
|
|
#ifdef _WIN32
|
2017-08-02 15:25:24 +08:00
|
|
|
CustomWorkingDir = "C:";
|
|
|
|
#else
|
|
|
|
CustomWorkingDir = "/";
|
|
|
|
#endif
|
|
|
|
llvm::sys::path::append(CustomWorkingDir, "some", "weird", "path");
|
|
|
|
|
2018-10-10 21:27:25 +08:00
|
|
|
auto FS = IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem>(
|
|
|
|
new llvm::vfs::InMemoryFileSystem);
|
2017-08-02 15:25:24 +08:00
|
|
|
// setCurrentworkingdirectory must finish without error.
|
|
|
|
ASSERT_TRUE(!FS->setCurrentWorkingDirectory(CustomWorkingDir));
|
|
|
|
|
|
|
|
FileSystemOptions Opts;
|
|
|
|
FileManager Manager(Opts, FS);
|
|
|
|
|
|
|
|
SmallString<64> Path("a/foo.cpp");
|
|
|
|
|
|
|
|
SmallString<64> ExpectedResult(CustomWorkingDir);
|
|
|
|
llvm::sys::path::append(ExpectedResult, Path);
|
|
|
|
|
|
|
|
ASSERT_TRUE(Manager.makeAbsolutePath(Path));
|
|
|
|
EXPECT_EQ(Path, ExpectedResult);
|
|
|
|
}
|
|
|
|
|
2018-12-01 01:10:11 +08:00
|
|
|
// getVirtualFile should always fill the real path.
|
|
|
|
TEST_F(FileManagerTest, getVirtualFileFillsRealPathName) {
|
2018-12-08 07:50:05 +08:00
|
|
|
SmallString<64> CustomWorkingDir;
|
|
|
|
#ifdef _WIN32
|
|
|
|
CustomWorkingDir = "C:/";
|
|
|
|
#else
|
|
|
|
CustomWorkingDir = "/";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
auto FS = IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem>(
|
|
|
|
new llvm::vfs::InMemoryFileSystem);
|
|
|
|
// setCurrentworkingdirectory must finish without error.
|
|
|
|
ASSERT_TRUE(!FS->setCurrentWorkingDirectory(CustomWorkingDir));
|
|
|
|
|
|
|
|
FileSystemOptions Opts;
|
|
|
|
FileManager Manager(Opts, FS);
|
|
|
|
|
2018-12-01 01:10:11 +08:00
|
|
|
// Inject fake files into the file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2018-12-01 01:10:11 +08:00
|
|
|
statCache->InjectDirectory("/tmp", 42);
|
|
|
|
statCache->InjectFile("/tmp/test", 43);
|
2018-12-08 07:50:05 +08:00
|
|
|
|
2018-12-22 03:33:09 +08:00
|
|
|
Manager.setStatCache(std::move(statCache));
|
2018-12-01 01:10:11 +08:00
|
|
|
|
|
|
|
// Check for real path.
|
2018-12-08 07:50:05 +08:00
|
|
|
const FileEntry *file = Manager.getVirtualFile("/tmp/test", 123, 1);
|
2018-12-01 01:10:11 +08:00
|
|
|
ASSERT_TRUE(file != nullptr);
|
|
|
|
ASSERT_TRUE(file->isValid());
|
2018-12-08 07:50:05 +08:00
|
|
|
SmallString<64> ExpectedResult = CustomWorkingDir;
|
|
|
|
|
2018-12-01 02:36:31 +08:00
|
|
|
llvm::sys::path::append(ExpectedResult, "tmp", "test");
|
|
|
|
EXPECT_EQ(file->tryGetRealPathName(), ExpectedResult);
|
2018-12-01 01:10:11 +08:00
|
|
|
}
|
|
|
|
|
2019-02-19 06:33:40 +08:00
|
|
|
TEST_F(FileManagerTest, getFileDontOpenRealPath) {
|
|
|
|
SmallString<64> CustomWorkingDir;
|
|
|
|
#ifdef _WIN32
|
|
|
|
CustomWorkingDir = "C:/";
|
|
|
|
#else
|
|
|
|
CustomWorkingDir = "/";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
auto FS = IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem>(
|
|
|
|
new llvm::vfs::InMemoryFileSystem);
|
|
|
|
// setCurrentworkingdirectory must finish without error.
|
|
|
|
ASSERT_TRUE(!FS->setCurrentWorkingDirectory(CustomWorkingDir));
|
|
|
|
|
|
|
|
FileSystemOptions Opts;
|
|
|
|
FileManager Manager(Opts, FS);
|
|
|
|
|
2019-02-19 07:12:29 +08:00
|
|
|
// Inject fake files into the file system.
|
2019-08-15 07:04:18 +08:00
|
|
|
auto statCache = std::make_unique<FakeStatCache>();
|
2019-02-19 07:12:29 +08:00
|
|
|
statCache->InjectDirectory("/tmp", 42);
|
|
|
|
statCache->InjectFile("/tmp/test", 43);
|
2019-02-19 06:33:40 +08:00
|
|
|
|
2019-02-19 07:12:29 +08:00
|
|
|
Manager.setStatCache(std::move(statCache));
|
2019-02-19 06:33:40 +08:00
|
|
|
|
2019-02-19 07:12:29 +08:00
|
|
|
// Check for real path.
|
2019-08-02 05:31:56 +08:00
|
|
|
auto file = Manager.getFile("/tmp/test", /*OpenFile=*/false);
|
|
|
|
ASSERT_TRUE(file);
|
|
|
|
ASSERT_TRUE((*file)->isValid());
|
2019-02-19 07:12:29 +08:00
|
|
|
SmallString<64> ExpectedResult = CustomWorkingDir;
|
2019-02-19 06:33:40 +08:00
|
|
|
|
2019-02-19 07:12:29 +08:00
|
|
|
llvm::sys::path::append(ExpectedResult, "tmp", "test");
|
2019-08-02 05:31:56 +08:00
|
|
|
EXPECT_EQ((*file)->tryGetRealPathName(), ExpectedResult);
|
2019-02-19 06:33:40 +08:00
|
|
|
}
|
|
|
|
|
2019-08-31 06:59:25 +08:00
|
|
|
TEST_F(FileManagerTest, getBypassFile) {
|
|
|
|
SmallString<64> CustomWorkingDir;
|
|
|
|
#ifdef _WIN32
|
|
|
|
CustomWorkingDir = "C:/";
|
|
|
|
#else
|
|
|
|
CustomWorkingDir = "/";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
auto FS = IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem>(
|
|
|
|
new llvm::vfs::InMemoryFileSystem);
|
|
|
|
// setCurrentworkingdirectory must finish without error.
|
|
|
|
ASSERT_TRUE(!FS->setCurrentWorkingDirectory(CustomWorkingDir));
|
|
|
|
|
|
|
|
FileSystemOptions Opts;
|
|
|
|
FileManager Manager(Opts, FS);
|
|
|
|
|
|
|
|
// Inject fake files into the file system.
|
|
|
|
auto Cache = std::make_unique<FakeStatCache>();
|
|
|
|
Cache->InjectDirectory("/tmp", 42);
|
|
|
|
Cache->InjectFile("/tmp/test", 43);
|
|
|
|
Manager.setStatCache(std::move(Cache));
|
|
|
|
|
|
|
|
// Set up a virtual file with a different size than FakeStatCache uses.
|
|
|
|
const FileEntry *File = Manager.getVirtualFile("/tmp/test", /*Size=*/10, 0);
|
|
|
|
ASSERT_TRUE(File);
|
2020-10-15 23:39:07 +08:00
|
|
|
const FileEntry &FE = *File;
|
|
|
|
EXPECT_TRUE(FE.isValid());
|
|
|
|
EXPECT_EQ(FE.getSize(), 10);
|
2019-08-31 06:59:25 +08:00
|
|
|
|
|
|
|
// Calling a second time should not affect the UID or size.
|
2020-10-15 23:39:07 +08:00
|
|
|
unsigned VirtualUID = FE.getUID();
|
|
|
|
EXPECT_EQ(
|
|
|
|
&FE,
|
|
|
|
&expectedToOptional(Manager.getFileRef("/tmp/test"))->getFileEntry());
|
|
|
|
EXPECT_EQ(FE.getUID(), VirtualUID);
|
|
|
|
EXPECT_EQ(FE.getSize(), 10);
|
2019-08-31 06:59:25 +08:00
|
|
|
|
|
|
|
// Bypass the file.
|
2020-10-15 23:39:07 +08:00
|
|
|
llvm::Optional<FileEntryRef> BypassRef =
|
|
|
|
Manager.getBypassFile(File->getLastRef());
|
2019-08-31 06:59:25 +08:00
|
|
|
ASSERT_TRUE(BypassRef);
|
|
|
|
EXPECT_TRUE(BypassRef->isValid());
|
2020-10-15 23:39:07 +08:00
|
|
|
EXPECT_EQ("/tmp/test", BypassRef->getName());
|
2019-08-31 06:59:25 +08:00
|
|
|
|
|
|
|
// Check that it's different in the right ways.
|
|
|
|
EXPECT_NE(&BypassRef->getFileEntry(), File);
|
|
|
|
EXPECT_NE(BypassRef->getUID(), VirtualUID);
|
2020-10-15 23:39:07 +08:00
|
|
|
EXPECT_NE(BypassRef->getSize(), FE.getSize());
|
2019-08-31 06:59:25 +08:00
|
|
|
|
|
|
|
// The virtual file should still be returned when searching.
|
2020-10-15 23:39:07 +08:00
|
|
|
EXPECT_EQ(
|
|
|
|
&FE,
|
|
|
|
&expectedToOptional(Manager.getFileRef("/tmp/test"))->getFileEntry());
|
2019-08-31 06:59:25 +08:00
|
|
|
}
|
|
|
|
|
2011-02-12 02:44:49 +08:00
|
|
|
} // anonymous namespace
|