Teach CompilerInstance to create and hold on to the Sema object used

for parsing, so that it can persist beyond the lifetime of the parsing
call.

llvm-svn: 110978
This commit is contained in:
Douglas Gregor 2010-08-12 23:31:19 +00:00
parent 384cae6e06
commit 0e93f017e5
3 changed files with 48 additions and 6 deletions

View File

@ -36,6 +36,7 @@ class FileManager;
class FrontendAction;
class PCHReader;
class Preprocessor;
class Sema;
class SourceManager;
class TargetInfo;
@ -91,6 +92,9 @@ class CompilerInstance {
/// The code completion consumer.
llvm::OwningPtr<CodeCompleteConsumer> CompletionConsumer;
/// \brief The semantic analysis object.
llvm::OwningPtr<Sema> TheSema;
/// The frontend timer
llvm::OwningPtr<llvm::Timer> FrontendTimer;
@ -369,6 +373,10 @@ public:
/// takes ownership of \arg Value.
void setASTContext(ASTContext *Value);
/// \brief Replace the current Sema; the compiler instance takes ownership
/// of S.
void setSema(Sema *S);
/// }
/// @name ASTConsumer
/// {
@ -388,6 +396,18 @@ public:
/// takes ownership of \arg Value.
void setASTConsumer(ASTConsumer *Value);
/// }
/// @name Semantic analysis
/// {
bool hasSema() const { return TheSema != 0; }
Sema &getSema() const {
assert(TheSema && "Compiler instance has no Sema object!");
return *TheSema;
}
Sema *takeSema() { return TheSema.take(); }
/// }
/// @name Code Completion
/// {
@ -526,6 +546,10 @@ public:
bool UseDebugPrinter, bool ShowMacros,
bool ShowCodePatterns, llvm::raw_ostream &OS);
/// \brief Create the Sema object to be used for parsing.
void createSema(bool CompleteTranslationUnit,
CodeCompleteConsumer *CompletionConsumer);
/// Create the frontend timer and replace any existing one with it.
void createFrontendTimer();

View File

@ -8,6 +8,7 @@
//===----------------------------------------------------------------------===//
#include "clang/Frontend/CompilerInstance.h"
#include "clang/Sema/Sema.h"
#include "clang/AST/ASTConsumer.h"
#include "clang/AST/ASTContext.h"
#include "clang/Basic/Diagnostic.h"
@ -41,6 +42,7 @@ CompilerInstance::CompilerInstance()
}
CompilerInstance::~CompilerInstance() {
TheSema.reset();
}
void CompilerInstance::setLLVMContext(llvm::LLVMContext *Value) {
@ -79,6 +81,10 @@ void CompilerInstance::setASTContext(ASTContext *Value) {
Context.reset(Value);
}
void CompilerInstance::setSema(Sema *S) {
TheSema.reset(S);
}
void CompilerInstance::setASTConsumer(ASTConsumer *Value) {
Consumer.reset(Value);
}
@ -362,6 +368,12 @@ CompilerInstance::createCodeCompletionConsumer(Preprocessor &PP,
return new CIndexCodeCompleteConsumer(ShowMacros, ShowCodePatterns, OS);
}
void CompilerInstance::createSema(bool CompleteTranslationUnit,
CodeCompleteConsumer *CompletionConsumer) {
TheSema.reset(new Sema(getPreprocessor(), getASTContext(), getASTConsumer(),
CompleteTranslationUnit, CompletionConsumer));
}
// Output Files
void CompilerInstance::addOutputFile(llvm::StringRef Path,

View File

@ -196,12 +196,16 @@ void FrontendAction::EndSourceFile() {
// FIXME: There is more per-file stuff we could just drop here?
if (CI.getFrontendOpts().DisableFree) {
CI.takeASTConsumer();
if (!isCurrentFileAST())
if (!isCurrentFileAST()) {
CI.takeSema();
CI.takeASTContext();
}
} else {
CI.setASTConsumer(0);
if (!isCurrentFileAST())
if (!isCurrentFileAST()) {
CI.setSema(0);
CI.setASTContext(0);
}
CI.setASTConsumer(0);
}
// Inform the preprocessor we are done.
@ -225,6 +229,7 @@ void FrontendAction::EndSourceFile() {
CI.getDiagnosticClient().EndSourceFile();
if (isCurrentFileAST()) {
CI.takeSema();
CI.takeASTContext();
CI.takePreprocessor();
CI.takeSourceManager();
@ -253,9 +258,10 @@ void ASTFrontendAction::ExecuteAction() {
if (CI.hasCodeCompletionConsumer())
CompletionConsumer = &CI.getCodeCompletionConsumer();
ParseAST(CI.getPreprocessor(), &CI.getASTConsumer(), CI.getASTContext(),
CI.getFrontendOpts().ShowStats,
usesCompleteTranslationUnit(), CompletionConsumer);
if (!CI.hasSema())
CI.createSema(usesCompleteTranslationUnit(), CompletionConsumer);
ParseAST(CI.getSema(), CI.getFrontendOpts().ShowStats);
}
ASTConsumer *