When setting the globally-visible declarations for a particular

identifier, we may have a Sema object but no translation unit scope
(because parsing is finished). In this case, we still need to update
the IdResolver, which might still be used when writing a PCH
containing another PCH (without chaining). This bug manifested as a
failure with precompiled preambles.

Also, add a little environment-variable-sensitive logging for
libclang.


git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@114774 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Douglas Gregor 2010-09-24 23:29:12 +00:00
Родитель 37bf9d2bb7
Коммит 76dc8890b6
5 изменённых файлов: 52 добавлений и 7 удалений

Просмотреть файл

@ -749,9 +749,20 @@ bool ASTUnit::Parse(llvm::MemoryBuffer *OverrideMainBuffer) {
getSourceManager()); getSourceManager());
StoredDiagnostics[I].setLocation(Loc); StoredDiagnostics[I].setLocation(Loc);
} }
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: using precompiled preamble for \"%s\" at "
"\"%s\"\n",
OriginalSourceFile.c_str(),
PreambleFile.c_str());
} else { } else {
PreprocessorOpts.PrecompiledPreambleBytes.first = 0; PreprocessorOpts.PrecompiledPreambleBytes.first = 0;
PreprocessorOpts.PrecompiledPreambleBytes.second = false; PreprocessorOpts.PrecompiledPreambleBytes.second = false;
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: not using precompiled preamble for \"%s\"\n",
OriginalSourceFile.c_str());
} }
llvm::OwningPtr<TopLevelDeclTrackerAction> Act; llvm::OwningPtr<TopLevelDeclTrackerAction> Act;
@ -1215,6 +1226,13 @@ llvm::MemoryBuffer *ASTUnit::getMainBufferWithPrecompiledPreamble(
// Create the source manager. // Create the source manager.
Clang.setSourceManager(new SourceManager(getDiagnostics())); Clang.setSourceManager(new SourceManager(getDiagnostics()));
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: creating precompiled preamble for \"%s\" at "
"\"%s\" (%u bytes)\n",
OriginalSourceFile.c_str(),
PreamblePCHPath.c_str(),
(unsigned)Preamble.size());
llvm::OwningPtr<PrecompilePreambleAction> Act; llvm::OwningPtr<PrecompilePreambleAction> Act;
Act.reset(new PrecompilePreambleAction(*this)); Act.reset(new PrecompilePreambleAction(*this));
if (!Act->BeginSourceFile(Clang, Clang.getFrontendOpts().Inputs[0].second, if (!Act->BeginSourceFile(Clang, Clang.getFrontendOpts().Inputs[0].second,
@ -1229,6 +1247,11 @@ llvm::MemoryBuffer *ASTUnit::getMainBufferWithPrecompiledPreamble(
PreambleRebuildCounter = DefaultPreambleRebuildInterval; PreambleRebuildCounter = DefaultPreambleRebuildInterval;
PreprocessorOpts.eraseRemappedFile( PreprocessorOpts.eraseRemappedFile(
PreprocessorOpts.remapped_file_buffer_end() - 1); PreprocessorOpts.remapped_file_buffer_end() - 1);
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: precompiled preamble compilation for \"%s\" "
"failed\n", OriginalSourceFile.c_str());
return 0; return 0;
} }
@ -1239,7 +1262,10 @@ llvm::MemoryBuffer *ASTUnit::getMainBufferWithPrecompiledPreamble(
if (Diagnostics->hasErrorOccurred()) { if (Diagnostics->hasErrorOccurred()) {
// There were errors parsing the preamble, so no precompiled header was // There were errors parsing the preamble, so no precompiled header was
// generated. Forget that we even tried. // generated. Forget that we even tried.
// FIXME: Should we leave a note for ourselves to try again?
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: precompiled preamble compilation for \"%s\" "
"failed\n", OriginalSourceFile.c_str());
llvm::sys::Path(FrontendOpts.OutputFile).eraseFromDisk(); llvm::sys::Path(FrontendOpts.OutputFile).eraseFromDisk();
Preamble.clear(); Preamble.clear();
if (CreatedPreambleBuffer) if (CreatedPreambleBuffer)
@ -1278,7 +1304,11 @@ llvm::MemoryBuffer *ASTUnit::getMainBufferWithPrecompiledPreamble(
if (PreambleTimer) if (PreambleTimer)
PreambleTimer->stopTimer(); PreambleTimer->stopTimer();
if (getenv("LIBCLANG_LOGGING"))
fprintf(stderr, "libclang: precompiled preamble for \"%s\" completed\n",
OriginalSourceFile.c_str());
PreambleRebuildCounter = 1; PreambleRebuildCounter = 1;
PreprocessorOpts.eraseRemappedFile( PreprocessorOpts.eraseRemappedFile(
PreprocessorOpts.remapped_file_buffer_end() - 1); PreprocessorOpts.remapped_file_buffer_end() - 1);

Просмотреть файл

@ -3362,11 +3362,11 @@ void ASTReader::InitializeSema(Sema &S) {
// Makes sure any declarations that were deserialized "too early" // Makes sure any declarations that were deserialized "too early"
// still get added to the identifier's declaration chains. // still get added to the identifier's declaration chains.
if (SemaObj->TUScope) { for (unsigned I = 0, N = PreloadedDecls.size(); I != N; ++I) {
for (unsigned I = 0, N = PreloadedDecls.size(); I != N; ++I) { if (SemaObj->TUScope)
SemaObj->TUScope->AddDecl(PreloadedDecls[I]); SemaObj->TUScope->AddDecl(PreloadedDecls[I]);
SemaObj->IdResolver.AddDecl(PreloadedDecls[I]);
} SemaObj->IdResolver.AddDecl(PreloadedDecls[I]);
} }
PreloadedDecls.clear(); PreloadedDecls.clear();
@ -3564,8 +3564,8 @@ ASTReader::SetGloballyVisibleDecls(IdentifierInfo *II,
// and add it to the declaration chain for this identifier, so // and add it to the declaration chain for this identifier, so
// that (unqualified) name lookup will find it. // that (unqualified) name lookup will find it.
SemaObj->TUScope->AddDecl(D); SemaObj->TUScope->AddDecl(D);
SemaObj->IdResolver.AddDeclToIdentifierChain(II, D);
} }
SemaObj->IdResolver.AddDeclToIdentifierChain(II, D);
} else { } else {
// Queue this declaration so that it will be added to the // Queue this declaration so that it will be added to the
// translation unit scope and identifier's declaration chain // translation unit scope and identifier's declaration chain

4
test/Index/Inputs/a.h Normal file
Просмотреть файл

@ -0,0 +1,4 @@
#ifndef A_H
#define A_H
typedef int A;
#endif

1
test/Index/Inputs/b.h Normal file
Просмотреть файл

@ -0,0 +1 @@
typedef float B;

Просмотреть файл

@ -0,0 +1,10 @@
// RUN: c-index-test -write-pch %t.h.pch -x c-header %S/Inputs/a.h
// RUN: env CINDEXTEST_EDITING=1 c-index-test -test-load-source-reparse 5 all -I%S/Inputs -include %t.h %s | FileCheck %s
#include "a.h"
#include "b.h"
A a;
B b;
// CHECK: a.h:3:13: TypedefDecl=A:3:13 (Definition) Extent=[3:13 - 3:14]
// CHECK: b.h:1:15: TypedefDecl=B:1:15 (Definition) Extent=[1:15 - 1:16]