forked from OSchip/llvm-project
Improve our uniquing of file entries when files are re-saved or are
overridden via remapping. Thus, when we create a "virtual" file in the file manager, we still stat() the real file that lives behind it so that we can provide proper uniquing based on inodes. This helps keep the file manager much more consistent. To take advantage of this when reparsing files in libclang, we disable the use of the stat() cache when reparsing or performing code completion, since the stat() cache is very likely to be out of date in this use case. llvm-svn: 124971
This commit is contained in:
parent
0890502f44
commit
606c4ac325
|
@ -537,6 +537,7 @@ public:
|
||||||
/// context.
|
/// context.
|
||||||
void createPCHExternalASTSource(llvm::StringRef Path,
|
void createPCHExternalASTSource(llvm::StringRef Path,
|
||||||
bool DisablePCHValidation,
|
bool DisablePCHValidation,
|
||||||
|
bool DisableStatCache,
|
||||||
void *DeserializationListener);
|
void *DeserializationListener);
|
||||||
|
|
||||||
/// Create an external AST source to read a PCH file.
|
/// Create an external AST source to read a PCH file.
|
||||||
|
@ -545,6 +546,7 @@ public:
|
||||||
static ExternalASTSource *
|
static ExternalASTSource *
|
||||||
createPCHExternalASTSource(llvm::StringRef Path, const std::string &Sysroot,
|
createPCHExternalASTSource(llvm::StringRef Path, const std::string &Sysroot,
|
||||||
bool DisablePCHValidation,
|
bool DisablePCHValidation,
|
||||||
|
bool DisableStatCache,
|
||||||
Preprocessor &PP, ASTContext &Context,
|
Preprocessor &PP, ASTContext &Context,
|
||||||
void *DeserializationListener, bool Preamble);
|
void *DeserializationListener, bool Preamble);
|
||||||
|
|
||||||
|
|
|
@ -48,6 +48,10 @@ public:
|
||||||
/// precompiled headers.
|
/// precompiled headers.
|
||||||
bool DisablePCHValidation;
|
bool DisablePCHValidation;
|
||||||
|
|
||||||
|
/// \brief When true, disables the use of the stat cache within a
|
||||||
|
/// precompiled header or AST file.
|
||||||
|
bool DisableStatCache;
|
||||||
|
|
||||||
/// \brief Dump declarations that are deserialized from PCH, for testing.
|
/// \brief Dump declarations that are deserialized from PCH, for testing.
|
||||||
bool DumpDeserializedPCHDecls;
|
bool DumpDeserializedPCHDecls;
|
||||||
|
|
||||||
|
@ -125,7 +129,7 @@ public:
|
||||||
|
|
||||||
public:
|
public:
|
||||||
PreprocessorOptions() : UsePredefines(true), DetailedRecord(false),
|
PreprocessorOptions() : UsePredefines(true), DetailedRecord(false),
|
||||||
DisablePCHValidation(false),
|
DisablePCHValidation(false), DisableStatCache(false),
|
||||||
DumpDeserializedPCHDecls(false),
|
DumpDeserializedPCHDecls(false),
|
||||||
PrecompiledPreambleBytes(0, true),
|
PrecompiledPreambleBytes(0, true),
|
||||||
RetainRemappedFileBuffers(false) { }
|
RetainRemappedFileBuffers(false) { }
|
||||||
|
|
|
@ -590,6 +590,9 @@ private:
|
||||||
/// headers when they are loaded.
|
/// headers when they are loaded.
|
||||||
bool DisableValidation;
|
bool DisableValidation;
|
||||||
|
|
||||||
|
/// \brief Whether to disable the use of stat caches in AST files.
|
||||||
|
bool DisableStatCache;
|
||||||
|
|
||||||
/// \brief Mapping from switch-case IDs in the chain to switch-case statements
|
/// \brief Mapping from switch-case IDs in the chain to switch-case statements
|
||||||
///
|
///
|
||||||
/// Statements usually don't have IDs, but switch cases need them, so that the
|
/// Statements usually don't have IDs, but switch cases need them, so that the
|
||||||
|
@ -783,8 +786,14 @@ public:
|
||||||
/// \param DisableValidation If true, the AST reader will suppress most
|
/// \param DisableValidation If true, the AST reader will suppress most
|
||||||
/// of its regular consistency checking, allowing the use of precompiled
|
/// of its regular consistency checking, allowing the use of precompiled
|
||||||
/// headers that cannot be determined to be compatible.
|
/// headers that cannot be determined to be compatible.
|
||||||
|
///
|
||||||
|
/// \param DisableStatCache If true, the AST reader will ignore the
|
||||||
|
/// stat cache in the AST files. This performance pessimization can
|
||||||
|
/// help when an AST file is being used in cases where the
|
||||||
|
/// underlying files in the file system may have changed, but
|
||||||
|
/// parsing should still continue.
|
||||||
ASTReader(Preprocessor &PP, ASTContext *Context, const char *isysroot = 0,
|
ASTReader(Preprocessor &PP, ASTContext *Context, const char *isysroot = 0,
|
||||||
bool DisableValidation = false);
|
bool DisableValidation = false, bool DisableStatCache = false);
|
||||||
|
|
||||||
/// \brief Load the AST file without using any pre-initialized Preprocessor.
|
/// \brief Load the AST file without using any pre-initialized Preprocessor.
|
||||||
///
|
///
|
||||||
|
@ -805,9 +814,15 @@ public:
|
||||||
/// \param DisableValidation If true, the AST reader will suppress most
|
/// \param DisableValidation If true, the AST reader will suppress most
|
||||||
/// of its regular consistency checking, allowing the use of precompiled
|
/// of its regular consistency checking, allowing the use of precompiled
|
||||||
/// headers that cannot be determined to be compatible.
|
/// headers that cannot be determined to be compatible.
|
||||||
|
///
|
||||||
|
/// \param DisableStatCache If true, the AST reader will ignore the
|
||||||
|
/// stat cache in the AST files. This performance pessimization can
|
||||||
|
/// help when an AST file is being used in cases where the
|
||||||
|
/// underlying files in the file system may have changed, but
|
||||||
|
/// parsing should still continue.
|
||||||
ASTReader(SourceManager &SourceMgr, FileManager &FileMgr,
|
ASTReader(SourceManager &SourceMgr, FileManager &FileMgr,
|
||||||
Diagnostic &Diags, const char *isysroot = 0,
|
Diagnostic &Diags, const char *isysroot = 0,
|
||||||
bool DisableValidation = false);
|
bool DisableValidation = false, bool DisableStatCache = false);
|
||||||
~ASTReader();
|
~ASTReader();
|
||||||
|
|
||||||
/// \brief Load the precompiled header designated by the given file
|
/// \brief Load the precompiled header designated by the given file
|
||||||
|
|
|
@ -359,12 +359,43 @@ FileManager::getVirtualFile(llvm::StringRef Filename, off_t Size,
|
||||||
NamedFileEnt.setValue(NON_EXISTENT_FILE);
|
NamedFileEnt.setValue(NON_EXISTENT_FILE);
|
||||||
|
|
||||||
// We allow the directory to not exist. If it does exist we store it.
|
// We allow the directory to not exist. If it does exist we store it.
|
||||||
//
|
FileEntry *UFE = 0;
|
||||||
const DirectoryEntry *DirInfo = getDirectoryFromFile(*this, Filename);
|
const DirectoryEntry *DirInfo = getDirectoryFromFile(*this, Filename);
|
||||||
|
if (DirInfo) {
|
||||||
|
// Check to see if the file exists. If so, drop the virtual file
|
||||||
|
int FileDescriptor = -1;
|
||||||
|
struct stat StatBuf;
|
||||||
|
const char *InterndFileName = NamedFileEnt.getKeyData();
|
||||||
|
if (getStatValue(InterndFileName, StatBuf, &FileDescriptor) == 0) {
|
||||||
|
// If the stat process opened the file, close it to avoid a FD leak.
|
||||||
|
if (FileDescriptor != -1)
|
||||||
|
close(FileDescriptor);
|
||||||
|
|
||||||
FileEntry *UFE = new FileEntry();
|
StatBuf.st_size = Size;
|
||||||
VirtualFileEntries.push_back(UFE);
|
StatBuf.st_mtime = ModificationTime;
|
||||||
NamedFileEnt.setValue(UFE);
|
UFE = &UniqueFiles.getFile(InterndFileName, StatBuf);
|
||||||
|
|
||||||
|
NamedFileEnt.setValue(UFE);
|
||||||
|
|
||||||
|
// If we had already opened this file, close it now so we don't
|
||||||
|
// leak the descriptor. We're not going to use the file
|
||||||
|
// descriptor anyway, since this is a virtual file.
|
||||||
|
if (UFE->FD != -1) {
|
||||||
|
close(UFE->FD);
|
||||||
|
UFE->FD = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we already have an entry with this inode, return it.
|
||||||
|
if (UFE->getName())
|
||||||
|
return UFE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!UFE) {
|
||||||
|
UFE = new FileEntry();
|
||||||
|
VirtualFileEntries.push_back(UFE);
|
||||||
|
NamedFileEnt.setValue(UFE);
|
||||||
|
}
|
||||||
|
|
||||||
// Get the null-terminated file name as stored as the key of the
|
// Get the null-terminated file name as stored as the key of the
|
||||||
// FileEntries map.
|
// FileEntries map.
|
||||||
|
@ -375,23 +406,7 @@ FileManager::getVirtualFile(llvm::StringRef Filename, off_t Size,
|
||||||
UFE->ModTime = ModificationTime;
|
UFE->ModTime = ModificationTime;
|
||||||
UFE->Dir = DirInfo;
|
UFE->Dir = DirInfo;
|
||||||
UFE->UID = NextFileUID++;
|
UFE->UID = NextFileUID++;
|
||||||
|
UFE->FD = -1;
|
||||||
// If this virtual file resolves to a file, also map that file to the
|
|
||||||
// newly-created file entry.
|
|
||||||
int FileDescriptor = -1;
|
|
||||||
struct stat StatBuf;
|
|
||||||
if (getStatValue(InterndFileName, StatBuf, &FileDescriptor)) {
|
|
||||||
// If the stat process opened the file, close it to avoid a FD leak.
|
|
||||||
if (FileDescriptor != -1)
|
|
||||||
close(FileDescriptor);
|
|
||||||
|
|
||||||
return UFE;
|
|
||||||
}
|
|
||||||
|
|
||||||
UFE->FD = FileDescriptor;
|
|
||||||
llvm::SmallString<128> FilePath(UFE->Name);
|
|
||||||
llvm::sys::fs::make_absolute(FilePath);
|
|
||||||
FileEntries[FilePath] = UFE;
|
|
||||||
return UFE;
|
return UFE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1580,6 +1580,7 @@ bool ASTUnit::Reparse(RemappedFile *RemappedFiles, unsigned NumRemappedFiles) {
|
||||||
|
|
||||||
// Remap files.
|
// Remap files.
|
||||||
PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
|
PreprocessorOptions &PPOpts = Invocation->getPreprocessorOpts();
|
||||||
|
PPOpts.DisableStatCache = true;
|
||||||
for (PreprocessorOptions::remapped_file_buffer_iterator
|
for (PreprocessorOptions::remapped_file_buffer_iterator
|
||||||
R = PPOpts.remapped_file_buffer_begin(),
|
R = PPOpts.remapped_file_buffer_begin(),
|
||||||
REnd = PPOpts.remapped_file_buffer_end();
|
REnd = PPOpts.remapped_file_buffer_end();
|
||||||
|
@ -1948,6 +1949,7 @@ void ASTUnit::CodeComplete(llvm::StringRef File, unsigned Line, unsigned Column,
|
||||||
|
|
||||||
// If the main file has been overridden due to the use of a preamble,
|
// If the main file has been overridden due to the use of a preamble,
|
||||||
// make that override happen and introduce the preamble.
|
// make that override happen and introduce the preamble.
|
||||||
|
PreprocessorOpts.DisableStatCache = true;
|
||||||
StoredDiagnostics.insert(StoredDiagnostics.end(),
|
StoredDiagnostics.insert(StoredDiagnostics.end(),
|
||||||
this->StoredDiagnostics.begin(),
|
this->StoredDiagnostics.begin(),
|
||||||
this->StoredDiagnostics.begin() + NumStoredDiagnosticsFromDriver);
|
this->StoredDiagnostics.begin() + NumStoredDiagnosticsFromDriver);
|
||||||
|
|
|
@ -234,11 +234,13 @@ void CompilerInstance::createASTContext() {
|
||||||
|
|
||||||
void CompilerInstance::createPCHExternalASTSource(llvm::StringRef Path,
|
void CompilerInstance::createPCHExternalASTSource(llvm::StringRef Path,
|
||||||
bool DisablePCHValidation,
|
bool DisablePCHValidation,
|
||||||
|
bool DisableStatCache,
|
||||||
void *DeserializationListener){
|
void *DeserializationListener){
|
||||||
llvm::OwningPtr<ExternalASTSource> Source;
|
llvm::OwningPtr<ExternalASTSource> Source;
|
||||||
bool Preamble = getPreprocessorOpts().PrecompiledPreambleBytes.first != 0;
|
bool Preamble = getPreprocessorOpts().PrecompiledPreambleBytes.first != 0;
|
||||||
Source.reset(createPCHExternalASTSource(Path, getHeaderSearchOpts().Sysroot,
|
Source.reset(createPCHExternalASTSource(Path, getHeaderSearchOpts().Sysroot,
|
||||||
DisablePCHValidation,
|
DisablePCHValidation,
|
||||||
|
DisableStatCache,
|
||||||
getPreprocessor(), getASTContext(),
|
getPreprocessor(), getASTContext(),
|
||||||
DeserializationListener,
|
DeserializationListener,
|
||||||
Preamble));
|
Preamble));
|
||||||
|
@ -249,6 +251,7 @@ ExternalASTSource *
|
||||||
CompilerInstance::createPCHExternalASTSource(llvm::StringRef Path,
|
CompilerInstance::createPCHExternalASTSource(llvm::StringRef Path,
|
||||||
const std::string &Sysroot,
|
const std::string &Sysroot,
|
||||||
bool DisablePCHValidation,
|
bool DisablePCHValidation,
|
||||||
|
bool DisableStatCache,
|
||||||
Preprocessor &PP,
|
Preprocessor &PP,
|
||||||
ASTContext &Context,
|
ASTContext &Context,
|
||||||
void *DeserializationListener,
|
void *DeserializationListener,
|
||||||
|
@ -256,7 +259,7 @@ CompilerInstance::createPCHExternalASTSource(llvm::StringRef Path,
|
||||||
llvm::OwningPtr<ASTReader> Reader;
|
llvm::OwningPtr<ASTReader> Reader;
|
||||||
Reader.reset(new ASTReader(PP, &Context,
|
Reader.reset(new ASTReader(PP, &Context,
|
||||||
Sysroot.empty() ? 0 : Sysroot.c_str(),
|
Sysroot.empty() ? 0 : Sysroot.c_str(),
|
||||||
DisablePCHValidation));
|
DisablePCHValidation, DisableStatCache));
|
||||||
|
|
||||||
Reader->setDeserializationListener(
|
Reader->setDeserializationListener(
|
||||||
static_cast<ASTDeserializationListener *>(DeserializationListener));
|
static_cast<ASTDeserializationListener *>(DeserializationListener));
|
||||||
|
|
|
@ -224,6 +224,7 @@ bool FrontendAction::BeginSourceFile(CompilerInstance &CI,
|
||||||
CI.createPCHExternalASTSource(
|
CI.createPCHExternalASTSource(
|
||||||
CI.getPreprocessorOpts().ImplicitPCHInclude,
|
CI.getPreprocessorOpts().ImplicitPCHInclude,
|
||||||
CI.getPreprocessorOpts().DisablePCHValidation,
|
CI.getPreprocessorOpts().DisablePCHValidation,
|
||||||
|
CI.getPreprocessorOpts().DisableStatCache,
|
||||||
DeserialListener);
|
DeserialListener);
|
||||||
if (!CI.getASTContext().getExternalSource())
|
if (!CI.getASTContext().getExternalSource())
|
||||||
goto failure;
|
goto failure;
|
||||||
|
|
|
@ -505,8 +505,6 @@ static void maybeSynthesizeBlockSignature(TypeProcessingState &state,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert(!isOnlyParens &&
|
|
||||||
"non-empty abstract-declarator contained only parens!");
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -2034,12 +2034,14 @@ ASTReader::ReadASTBlock(PerFileData &F) {
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case STAT_CACHE: {
|
case STAT_CACHE: {
|
||||||
ASTStatCache *MyStatCache =
|
if (!DisableStatCache) {
|
||||||
new ASTStatCache((const unsigned char *)BlobStart + Record[0],
|
ASTStatCache *MyStatCache =
|
||||||
(const unsigned char *)BlobStart,
|
new ASTStatCache((const unsigned char *)BlobStart + Record[0],
|
||||||
NumStatHits, NumStatMisses);
|
(const unsigned char *)BlobStart,
|
||||||
FileMgr.addStatCache(MyStatCache);
|
NumStatHits, NumStatMisses);
|
||||||
F.StatCache = MyStatCache;
|
FileMgr.addStatCache(MyStatCache);
|
||||||
|
F.StatCache = MyStatCache;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4668,28 +4670,32 @@ void ASTReader::FinishedDeserializing() {
|
||||||
}
|
}
|
||||||
|
|
||||||
ASTReader::ASTReader(Preprocessor &PP, ASTContext *Context,
|
ASTReader::ASTReader(Preprocessor &PP, ASTContext *Context,
|
||||||
const char *isysroot, bool DisableValidation)
|
const char *isysroot, bool DisableValidation,
|
||||||
|
bool DisableStatCache)
|
||||||
: Listener(new PCHValidator(PP, *this)), DeserializationListener(0),
|
: Listener(new PCHValidator(PP, *this)), DeserializationListener(0),
|
||||||
SourceMgr(PP.getSourceManager()), FileMgr(PP.getFileManager()),
|
SourceMgr(PP.getSourceManager()), FileMgr(PP.getFileManager()),
|
||||||
Diags(PP.getDiagnostics()), SemaObj(0), PP(&PP), Context(Context),
|
Diags(PP.getDiagnostics()), SemaObj(0), PP(&PP), Context(Context),
|
||||||
Consumer(0), isysroot(isysroot), DisableValidation(DisableValidation),
|
Consumer(0), isysroot(isysroot), DisableValidation(DisableValidation),
|
||||||
NumStatHits(0), NumStatMisses(0), NumSLocEntriesRead(0),
|
DisableStatCache(DisableStatCache), NumStatHits(0), NumStatMisses(0),
|
||||||
TotalNumSLocEntries(0), NextSLocOffset(0), NumStatementsRead(0),
|
NumSLocEntriesRead(0), TotalNumSLocEntries(0), NextSLocOffset(0),
|
||||||
TotalNumStatements(0), NumMacrosRead(0), TotalNumMacros(0),
|
NumStatementsRead(0), TotalNumStatements(0), NumMacrosRead(0),
|
||||||
NumSelectorsRead(0), NumMethodPoolEntriesRead(0), NumMethodPoolMisses(0),
|
TotalNumMacros(0), NumSelectorsRead(0), NumMethodPoolEntriesRead(0),
|
||||||
TotalNumMethodPoolEntries(0), NumLexicalDeclContextsRead(0),
|
NumMethodPoolMisses(0), TotalNumMethodPoolEntries(0),
|
||||||
TotalLexicalDeclContexts(0), NumVisibleDeclContextsRead(0),
|
NumLexicalDeclContextsRead(0), TotalLexicalDeclContexts(0),
|
||||||
TotalVisibleDeclContexts(0), NumCurrentElementsDeserializing(0) {
|
NumVisibleDeclContextsRead(0), TotalVisibleDeclContexts(0),
|
||||||
|
NumCurrentElementsDeserializing(0)
|
||||||
|
{
|
||||||
RelocatablePCH = false;
|
RelocatablePCH = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASTReader::ASTReader(SourceManager &SourceMgr, FileManager &FileMgr,
|
ASTReader::ASTReader(SourceManager &SourceMgr, FileManager &FileMgr,
|
||||||
Diagnostic &Diags, const char *isysroot,
|
Diagnostic &Diags, const char *isysroot,
|
||||||
bool DisableValidation)
|
bool DisableValidation, bool DisableStatCache)
|
||||||
: DeserializationListener(0), SourceMgr(SourceMgr), FileMgr(FileMgr),
|
: DeserializationListener(0), SourceMgr(SourceMgr), FileMgr(FileMgr),
|
||||||
Diags(Diags), SemaObj(0), PP(0), Context(0), Consumer(0),
|
Diags(Diags), SemaObj(0), PP(0), Context(0), Consumer(0),
|
||||||
isysroot(isysroot), DisableValidation(DisableValidation), NumStatHits(0),
|
isysroot(isysroot), DisableValidation(DisableValidation),
|
||||||
NumStatMisses(0), NumSLocEntriesRead(0), TotalNumSLocEntries(0),
|
DisableStatCache(DisableStatCache), NumStatHits(0), NumStatMisses(0),
|
||||||
|
NumSLocEntriesRead(0), TotalNumSLocEntries(0),
|
||||||
NextSLocOffset(0), NumStatementsRead(0), TotalNumStatements(0),
|
NextSLocOffset(0), NumStatementsRead(0), TotalNumStatements(0),
|
||||||
NumMacrosRead(0), TotalNumMacros(0), NumSelectorsRead(0),
|
NumMacrosRead(0), TotalNumMacros(0), NumSelectorsRead(0),
|
||||||
NumMethodPoolEntriesRead(0), NumMethodPoolMisses(0),
|
NumMethodPoolEntriesRead(0), NumMethodPoolMisses(0),
|
||||||
|
|
Loading…
Reference in New Issue