diff --git a/clang/include/clang/Tooling/DependencyScanning/DependencyScanningFilesystem.h b/clang/include/clang/Tooling/DependencyScanning/DependencyScanningFilesystem.h --- a/clang/include/clang/Tooling/DependencyScanning/DependencyScanningFilesystem.h +++ b/clang/include/clang/Tooling/DependencyScanning/DependencyScanningFilesystem.h @@ -269,6 +269,33 @@ } }; +enum class ScanFile { Yes, No }; +enum class CacheStatFailure { Yes, No }; + +struct PathPolicy { + unsigned Enable : 1; // Implies caching of all open and stat results. + unsigned ScanFile : 1; + unsigned CacheStatFailure : 1; // Explicitly disables stat failure caching. + + static PathPolicy fallThrough() { + PathPolicy Policy; + Policy.Enable = false; + return Policy; + } + + static PathPolicy cache(enum ScanFile SF, + enum CacheStatFailure CSF = CacheStatFailure::Yes) { + PathPolicy Policy; + Policy.Enable = true; + Policy.ScanFile = SF == ScanFile::Yes; + Policy.CacheStatFailure = CSF == CacheStatFailure::Yes; + return Policy; + } +}; + +/// Determine caching and scanning behavior based on file extension. +PathPolicy getPolicy(StringRef Filename); + /// A virtual file system optimized for the dependency discovery. /// /// It is primarily designed to work with source files whose contents was @@ -293,24 +320,25 @@ /// /// Attempts to use the local and shared caches first, then falls back to /// using the underlying filesystem. - llvm::ErrorOr - getOrCreateFileSystemEntry(StringRef Filename, - bool DisableDirectivesScanning = false); + llvm::ErrorOr getOrCreateFileSystemEntry(StringRef Filename) { + return getOrCreateFileSystemEntry(Filename, getPolicy(Filename)); + } private: - /// Check whether the file should be scanned for preprocessor directives. - bool shouldScanForDirectives(StringRef Filename); + /// Same as the public version, but with explicit PathPolicy parameter. + llvm::ErrorOr getOrCreateFileSystemEntry(StringRef Filename, + PathPolicy Policy); /// For a filename that's not yet associated with any entry in the caches, /// uses the underlying filesystem to either look up the entry based in the /// shared cache indexed by unique ID, or creates new entry from scratch. llvm::ErrorOr - computeAndStoreResult(StringRef Filename); + computeAndStoreResult(StringRef Filename, PathPolicy Policy); /// Scan for preprocessor directives for the given entry if necessary and /// returns a wrapper object with reference semantics. EntryRef scanForDirectivesIfNecessary(const CachedFileSystemEntry &Entry, - StringRef Filename, bool Disable); + StringRef Filename, PathPolicy Policy); /// Represents a filesystem entry that has been stat-ed (and potentially read) /// and that's about to be inserted into the cache as `CachedFileSystemEntry`. diff --git a/clang/lib/Tooling/DependencyScanning/DependencyScanningFilesystem.cpp b/clang/lib/Tooling/DependencyScanning/DependencyScanningFilesystem.cpp --- a/clang/lib/Tooling/DependencyScanning/DependencyScanningFilesystem.cpp +++ b/clang/lib/Tooling/DependencyScanning/DependencyScanningFilesystem.cpp @@ -42,9 +42,8 @@ } EntryRef DependencyScanningWorkerFilesystem::scanForDirectivesIfNecessary( - const CachedFileSystemEntry &Entry, StringRef Filename, bool Disable) { - if (Entry.isError() || Entry.isDirectory() || Disable || - !shouldScanForDirectives(Filename)) + const CachedFileSystemEntry &Entry, StringRef Filename, PathPolicy Policy) { + if (Entry.isError() || Entry.isDirectory() || !Policy.ScanFile) return EntryRef(Filename, Entry); CachedFileContents *Contents = Entry.getCachedContents(); @@ -159,39 +158,22 @@ return *EntriesByFilename.insert({Filename, &Entry}).first->getValue(); } -/// Whitelist file extensions that should be minimized, treating no extension as -/// a source file that should be minimized. -/// -/// This is kinda hacky, it would be better if we knew what kind of file Clang -/// was expecting instead. -static bool shouldScanForDirectivesBasedOnExtension(StringRef Filename) { +PathPolicy clang::tooling::dependencies::getPolicy(StringRef Filename) { StringRef Ext = llvm::sys::path::extension(Filename); if (Ext.empty()) - return true; // C++ standard library - return llvm::StringSwitch(Ext) - .CasesLower(".c", ".cc", ".cpp", ".c++", ".cxx", true) - .CasesLower(".h", ".hh", ".hpp", ".h++", ".hxx", true) - .CasesLower(".m", ".mm", true) - .CasesLower(".i", ".ii", ".mi", ".mmi", true) - .CasesLower(".def", ".inc", true) - .Default(false); -} - -static bool shouldCacheStatFailures(StringRef Filename) { - StringRef Ext = llvm::sys::path::extension(Filename); - if (Ext.empty()) - return false; // This may be the module cache directory. - // Only cache stat failures on files that are not expected to change during - // the build. - StringRef FName = llvm::sys::path::filename(Filename); - if (FName == "module.modulemap" || FName == "module.map") - return true; - return shouldScanForDirectivesBasedOnExtension(Filename); -} - -bool DependencyScanningWorkerFilesystem::shouldScanForDirectives( - StringRef Filename) { - return shouldScanForDirectivesBasedOnExtension(Filename); + return PathPolicy::cache(ScanFile::Yes, CacheStatFailure::No); + // clang-format off + return llvm::StringSwitch(Ext) + .CasesLower(".c", ".cc", ".cpp", ".c++", ".cxx", PathPolicy::cache(ScanFile::Yes)) + .CasesLower(".h", ".hh", ".hpp", ".h++", ".hxx", PathPolicy::cache(ScanFile::Yes)) + .CasesLower(".m", ".mm", PathPolicy::cache(ScanFile::Yes)) + .CasesLower(".i", ".ii", ".mi", ".mmi", PathPolicy::cache(ScanFile::Yes)) + .CasesLower(".def", ".inc", PathPolicy::cache(ScanFile::Yes)) + .CasesLower(".modulemap", ".map", PathPolicy::cache(ScanFile::No)) + .CasesLower(".framework", ".apinotes", PathPolicy::cache(ScanFile::No)) + .CasesLower(".yaml", ".json", ".hmap", PathPolicy::cache(ScanFile::No)) + .Default(PathPolicy::fallThrough()); + // clang-format on } const CachedFileSystemEntry & @@ -215,10 +197,11 @@ } llvm::ErrorOr -DependencyScanningWorkerFilesystem::computeAndStoreResult(StringRef Filename) { +DependencyScanningWorkerFilesystem::computeAndStoreResult(StringRef Filename, + PathPolicy Policy) { llvm::ErrorOr Stat = getUnderlyingFS().status(Filename); if (!Stat) { - if (!shouldCacheStatFailures(Filename)) + if (!Policy.CacheStatFailure) return Stat.getError(); const auto &Entry = getOrEmplaceSharedEntryForFilename(Filename, Stat.getError()); @@ -244,16 +227,13 @@ llvm::ErrorOr DependencyScanningWorkerFilesystem::getOrCreateFileSystemEntry( - StringRef Filename, bool DisableDirectivesScanning) { + StringRef Filename, PathPolicy Policy) { if (const auto *Entry = findEntryByFilenameWithWriteThrough(Filename)) - return scanForDirectivesIfNecessary(*Entry, Filename, - DisableDirectivesScanning) - .unwrapError(); - auto MaybeEntry = computeAndStoreResult(Filename); + return scanForDirectivesIfNecessary(*Entry, Filename, Policy).unwrapError(); + auto MaybeEntry = computeAndStoreResult(Filename, Policy); if (!MaybeEntry) return MaybeEntry.getError(); - return scanForDirectivesIfNecessary(*MaybeEntry, Filename, - DisableDirectivesScanning) + return scanForDirectivesIfNecessary(*MaybeEntry, Filename, Policy) .unwrapError(); } @@ -261,8 +241,11 @@ DependencyScanningWorkerFilesystem::status(const Twine &Path) { SmallString<256> OwnedFilename; StringRef Filename = Path.toStringRef(OwnedFilename); + PathPolicy Policy = getPolicy(Filename); + if (!Policy.Enable) + return getUnderlyingFS().status(Path); - llvm::ErrorOr Result = getOrCreateFileSystemEntry(Filename); + llvm::ErrorOr Result = getOrCreateFileSystemEntry(Filename, Policy); if (!Result) return Result.getError(); return Result->getStatus(); @@ -318,8 +301,11 @@ DependencyScanningWorkerFilesystem::openFileForRead(const Twine &Path) { SmallString<256> OwnedFilename; StringRef Filename = Path.toStringRef(OwnedFilename); + PathPolicy Policy = getPolicy(Filename); + if (!Policy.Enable) + return getUnderlyingFS().openFileForRead(Path); - llvm::ErrorOr Result = getOrCreateFileSystemEntry(Filename); + llvm::ErrorOr Result = getOrCreateFileSystemEntry(Filename, Policy); if (!Result) return Result.getError(); return DepScanFile::create(Result.get()); diff --git a/clang/unittests/Tooling/DependencyScannerTest.cpp b/clang/unittests/Tooling/DependencyScannerTest.cpp --- a/clang/unittests/Tooling/DependencyScannerTest.cpp +++ b/clang/unittests/Tooling/DependencyScannerTest.cpp @@ -239,3 +239,130 @@ EXPECT_EQ(convert_to_slash(DepFile), "test.cpp.o: /root/test.cpp /root/header.h\n"); } + +// Note: We want to test caching in DependencyScanningWorkerFilesystem. To do +// that, we need to be able to mutate the underlying file system. However, +// InMemoryFileSystem does not allow changing the contents of a file after it's +// been created. +// To simulate the behavior, we create two separate in-memory file systems, each +// containing different version of the same file. We pass those to two scanning +// file systems that share the same cache. + +TEST(DependencyScanningFileSystemTest, CacheFileContentsEnabled) { + DependencyScanningFilesystemSharedCache SharedCache; + + StringRef Path = "/root/source.c"; + auto Contents1 = llvm::MemoryBuffer::getMemBuffer("contents1"); + auto Contents2 = llvm::MemoryBuffer::getMemBuffer("contents2"); + + { + auto InMemoryFS = + llvm::makeIntrusiveRefCnt(); + ASSERT_TRUE(InMemoryFS->addFile(Path, 0, std::move(Contents1))); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + auto File = ScanningFS.openFileForRead(Path); + ASSERT_TRUE(File); + auto Buffer = (*File)->getBuffer("Buffer for /root/source.c."); + ASSERT_TRUE(Buffer); + auto Contents = (*Buffer)->getBuffer(); + EXPECT_EQ(Contents, "contents1"); + } + + { + auto InMemoryFS = + llvm::makeIntrusiveRefCnt(); + ASSERT_TRUE(InMemoryFS->addFile(Path, 0, std::move(Contents2))); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + auto File = ScanningFS.openFileForRead(Path); + ASSERT_TRUE(File); + auto Buffer = (*File)->getBuffer("Buffer for /root/source.c."); + ASSERT_TRUE(Buffer); + auto Contents = (*Buffer)->getBuffer(); + EXPECT_EQ(Contents, "contents1"); + } +} + +TEST(DependencyScanningFileSystemTest, CacheFileContentsDisabled) { + DependencyScanningFilesystemSharedCache SharedCache; + + StringRef Path = "/root/module.pcm"; + auto Contents1 = llvm::MemoryBuffer::getMemBuffer("contents1"); + auto Contents2 = llvm::MemoryBuffer::getMemBuffer("contents2"); + + { + auto InMemoryFS = + llvm::makeIntrusiveRefCnt(); + ASSERT_TRUE(InMemoryFS->addFile(Path, 0, std::move(Contents1))); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + auto File = ScanningFS.openFileForRead(Path); + ASSERT_TRUE(File); + auto Buffer = (*File)->getBuffer("Buffer for /root/module.pcm."); + ASSERT_TRUE(Buffer); + auto Contents = (*Buffer)->getBuffer(); + EXPECT_EQ(Contents, "contents1"); + } + + { + auto InMemoryFS = + llvm::makeIntrusiveRefCnt(); + ASSERT_TRUE(InMemoryFS->addFile(Path, 0, std::move(Contents2))); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + auto File = ScanningFS.openFileForRead(Path); + ASSERT_TRUE(File); + auto Buffer = (*File)->getBuffer("Buffer for /root/module.pcm."); + ASSERT_TRUE(Buffer); + auto Contents = (*Buffer)->getBuffer(); + EXPECT_EQ(Contents, "contents2"); + } +} + +TEST(DependencyScanningFileSystemTest, CacheStatFailureEnabled) { + DependencyScanningFilesystemSharedCache SharedCache; + auto InMemoryFS = llvm::makeIntrusiveRefCnt(); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + + StringRef Path = "/root/source.c"; + + auto Stat1 = ScanningFS.status(Path); + EXPECT_FALSE(Stat1); + + auto Contents = llvm::MemoryBuffer::getMemBuffer("contents"); + InMemoryFS->addFile(Path, 0, std::move(Contents)); + + auto Stat2 = ScanningFS.status(Path); + EXPECT_FALSE(Stat2); +} + +TEST(DependencyScanningFileSystemTest, CacheStatFailureDisabledFile) { + DependencyScanningFilesystemSharedCache SharedCache; + auto InMemoryFS = llvm::makeIntrusiveRefCnt(); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + + StringRef Path = "/root/vector"; + + auto Stat1 = ScanningFS.status(Path); + EXPECT_FALSE(Stat1); + + auto Contents = llvm::MemoryBuffer::getMemBuffer("contents"); + InMemoryFS->addFile(Path, 0, std::move(Contents)); + + auto Stat2 = ScanningFS.status(Path); + EXPECT_TRUE(Stat2); +} + +TEST(DependencyScanningFileSystemTest, CacheStatFailureDisabledDirectory) { + DependencyScanningFilesystemSharedCache SharedCache; + auto InMemoryFS = llvm::makeIntrusiveRefCnt(); + DependencyScanningWorkerFilesystem ScanningFS(SharedCache, InMemoryFS); + + StringRef Path = "/root/dir"; + + auto Stat1 = ScanningFS.status(Path); + EXPECT_FALSE(Stat1); + + auto Contents = llvm::MemoryBuffer::getMemBuffer("contents"); + InMemoryFS->addFile("/root/dir/file", 0, std::move(Contents)); + + auto Stat2 = ScanningFS.status(Path); + EXPECT_TRUE(Stat2); +}