Index: include/clang/Basic/FileManager.h =================================================================== --- include/clang/Basic/FileManager.h +++ include/clang/Basic/FileManager.h @@ -186,6 +186,9 @@ /// \brief Removes the specified FileSystemStatCache object from the manager. void removeStatCache(FileSystemStatCache *statCache); + /// \brief Removes all FileSystemStatCache objects from the manager. + void clearStatCaches(); + /// \brief Lookup, cache, and verify the specified directory (real or /// virtual). /// Index: lib/Basic/FileManager.cpp =================================================================== --- lib/Basic/FileManager.cpp +++ lib/Basic/FileManager.cpp @@ -223,6 +223,10 @@ PrevCache->setNextStatCache(statCache->getNextStatCache()); } +void FileManager::clearStatCaches() { + StatCache.reset(0); +} + /// \brief Retrieve the directory that the given file name resides in. /// Filename can point to either a real file or a virtual file. static const DirectoryEntry *getDirectoryFromFile(FileManager &FileMgr, Index: lib/Tooling/Tooling.cpp =================================================================== --- lib/Tooling/Tooling.cpp +++ lib/Tooling/Tooling.cpp @@ -212,6 +212,7 @@ const bool Success = Compiler.ExecuteAction(*ScopedToolAction); Compiler.resetAndLeakFileManager(); + Files->clearStatCaches(); return Success; } Index: test/Tooling/pch.cpp =================================================================== --- /dev/null +++ test/Tooling/pch.cpp @@ -0,0 +1,21 @@ +// This is a regression test for handling of stat caches within the tooling +// infrastructure. This test reproduces the problem under valgrind: + +// First, create a pch that we can later load. Loading the pch will insert +// a stat cache into the FileManager: +// RUN: %clang -x c++-header %S/Inputs/pch.h -o %t1 + +// Use the generated pch and enforce a subsequent stat miss by by using +// the test file with an unrelated include as second translation unit: +// Do not directly pipe into FileCheck, as that would hide errors from +// valgrind due to pipefail not being set in lit. +// RUN: clang-check "%S/Inputs/pch.cpp" "%s" -- -include-pch %t1 -I "%S" -c >%t2 2>&1 +// RUN: FileCheck %s < %t2 + +#include "Inputs/pch-fail.h" + +// CHECK: Processing + +// FIXME: This is incompatible to -fms-compatibility. +// XFAIL: win32 +