mirror of
https://github.com/Gericom/teak-llvm.git
synced 2025-06-20 20:15:49 -04:00

When plugins are used the Multiplex(AST)Consumer is employed to dispatch to both the plugin ASTConsumers and the IRGen ASTConsumer. It wasn't dispatching a critical call for debug info, resulting in plugin users having a negative debugging experience. While I'm here, forward a bunch of other missing calls through the consumer that seem like they should be there. To test this, use the example plugin (requires plugins and examples) and split the test case up so that the plugin testing can be done under that requirement while the non-plugin testing will execute even in builds that don't include plugin support or examples. llvm-svn: 213213
324 lines
13 KiB
C++
324 lines
13 KiB
C++
//===- MultiplexConsumer.cpp - AST Consumer for PCH Generation --*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines the MultiplexConsumer class. It also declares and defines
|
|
// MultiplexASTDeserializationListener and MultiplexASTMutationListener, which
|
|
// are implementation details of MultiplexConsumer.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Frontend/MultiplexConsumer.h"
|
|
#include "clang/AST/ASTMutationListener.h"
|
|
#include "clang/AST/DeclGroup.h"
|
|
#include "clang/Serialization/ASTDeserializationListener.h"
|
|
|
|
using namespace clang;
|
|
|
|
namespace clang {
|
|
|
|
// This ASTDeserializationListener forwards its notifications to a set of
|
|
// child listeners.
|
|
class MultiplexASTDeserializationListener
|
|
: public ASTDeserializationListener {
|
|
public:
|
|
// Does NOT take ownership of the elements in L.
|
|
MultiplexASTDeserializationListener(
|
|
const std::vector<ASTDeserializationListener*>& L);
|
|
void ReaderInitialized(ASTReader *Reader) override;
|
|
void IdentifierRead(serialization::IdentID ID,
|
|
IdentifierInfo *II) override;
|
|
void TypeRead(serialization::TypeIdx Idx, QualType T) override;
|
|
void DeclRead(serialization::DeclID ID, const Decl *D) override;
|
|
void SelectorRead(serialization::SelectorID iD, Selector Sel) override;
|
|
void MacroDefinitionRead(serialization::PreprocessedEntityID,
|
|
MacroDefinition *MD) override;
|
|
private:
|
|
std::vector<ASTDeserializationListener*> Listeners;
|
|
};
|
|
|
|
MultiplexASTDeserializationListener::MultiplexASTDeserializationListener(
|
|
const std::vector<ASTDeserializationListener*>& L)
|
|
: Listeners(L) {
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::ReaderInitialized(
|
|
ASTReader *Reader) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->ReaderInitialized(Reader);
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::IdentifierRead(
|
|
serialization::IdentID ID, IdentifierInfo *II) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->IdentifierRead(ID, II);
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::TypeRead(
|
|
serialization::TypeIdx Idx, QualType T) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->TypeRead(Idx, T);
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::DeclRead(
|
|
serialization::DeclID ID, const Decl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->DeclRead(ID, D);
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::SelectorRead(
|
|
serialization::SelectorID ID, Selector Sel) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->SelectorRead(ID, Sel);
|
|
}
|
|
|
|
void MultiplexASTDeserializationListener::MacroDefinitionRead(
|
|
serialization::PreprocessedEntityID ID, MacroDefinition *MD) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->MacroDefinitionRead(ID, MD);
|
|
}
|
|
|
|
// This ASTMutationListener forwards its notifications to a set of
|
|
// child listeners.
|
|
class MultiplexASTMutationListener : public ASTMutationListener {
|
|
public:
|
|
// Does NOT take ownership of the elements in L.
|
|
MultiplexASTMutationListener(ArrayRef<ASTMutationListener*> L);
|
|
void CompletedTagDefinition(const TagDecl *D) override;
|
|
void AddedVisibleDecl(const DeclContext *DC, const Decl *D) override;
|
|
void AddedCXXImplicitMember(const CXXRecordDecl *RD, const Decl *D) override;
|
|
void AddedCXXTemplateSpecialization(const ClassTemplateDecl *TD,
|
|
const ClassTemplateSpecializationDecl *D) override;
|
|
void AddedCXXTemplateSpecialization(const VarTemplateDecl *TD,
|
|
const VarTemplateSpecializationDecl *D) override;
|
|
void AddedCXXTemplateSpecialization(const FunctionTemplateDecl *TD,
|
|
const FunctionDecl *D) override;
|
|
void DeducedReturnType(const FunctionDecl *FD, QualType ReturnType) override;
|
|
void CompletedImplicitDefinition(const FunctionDecl *D) override;
|
|
void StaticDataMemberInstantiated(const VarDecl *D) override;
|
|
void AddedObjCCategoryToInterface(const ObjCCategoryDecl *CatD,
|
|
const ObjCInterfaceDecl *IFD) override;
|
|
void AddedObjCPropertyInClassExtension(const ObjCPropertyDecl *Prop,
|
|
const ObjCPropertyDecl *OrigProp,
|
|
const ObjCCategoryDecl *ClassExt) override;
|
|
void DeclarationMarkedUsed(const Decl *D) override;
|
|
|
|
private:
|
|
std::vector<ASTMutationListener*> Listeners;
|
|
};
|
|
|
|
MultiplexASTMutationListener::MultiplexASTMutationListener(
|
|
ArrayRef<ASTMutationListener*> L)
|
|
: Listeners(L.begin(), L.end()) {
|
|
}
|
|
|
|
void MultiplexASTMutationListener::CompletedTagDefinition(const TagDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->CompletedTagDefinition(D);
|
|
}
|
|
|
|
void MultiplexASTMutationListener::AddedVisibleDecl(
|
|
const DeclContext *DC, const Decl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedVisibleDecl(DC, D);
|
|
}
|
|
|
|
void MultiplexASTMutationListener::AddedCXXImplicitMember(
|
|
const CXXRecordDecl *RD, const Decl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedCXXImplicitMember(RD, D);
|
|
}
|
|
void MultiplexASTMutationListener::AddedCXXTemplateSpecialization(
|
|
const ClassTemplateDecl *TD, const ClassTemplateSpecializationDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedCXXTemplateSpecialization(TD, D);
|
|
}
|
|
void MultiplexASTMutationListener::AddedCXXTemplateSpecialization(
|
|
const VarTemplateDecl *TD, const VarTemplateSpecializationDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedCXXTemplateSpecialization(TD, D);
|
|
}
|
|
void MultiplexASTMutationListener::AddedCXXTemplateSpecialization(
|
|
const FunctionTemplateDecl *TD, const FunctionDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedCXXTemplateSpecialization(TD, D);
|
|
}
|
|
void MultiplexASTMutationListener::DeducedReturnType(const FunctionDecl *FD,
|
|
QualType ReturnType) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->DeducedReturnType(FD, ReturnType);
|
|
}
|
|
void MultiplexASTMutationListener::CompletedImplicitDefinition(
|
|
const FunctionDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->CompletedImplicitDefinition(D);
|
|
}
|
|
void MultiplexASTMutationListener::StaticDataMemberInstantiated(
|
|
const VarDecl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->StaticDataMemberInstantiated(D);
|
|
}
|
|
void MultiplexASTMutationListener::AddedObjCCategoryToInterface(
|
|
const ObjCCategoryDecl *CatD,
|
|
const ObjCInterfaceDecl *IFD) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedObjCCategoryToInterface(CatD, IFD);
|
|
}
|
|
void MultiplexASTMutationListener::AddedObjCPropertyInClassExtension(
|
|
const ObjCPropertyDecl *Prop,
|
|
const ObjCPropertyDecl *OrigProp,
|
|
const ObjCCategoryDecl *ClassExt) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->AddedObjCPropertyInClassExtension(Prop, OrigProp, ClassExt);
|
|
}
|
|
void MultiplexASTMutationListener::DeclarationMarkedUsed(const Decl *D) {
|
|
for (size_t i = 0, e = Listeners.size(); i != e; ++i)
|
|
Listeners[i]->DeclarationMarkedUsed(D);
|
|
}
|
|
|
|
} // end namespace clang
|
|
|
|
MultiplexConsumer::MultiplexConsumer(ArrayRef<ASTConsumer *> C)
|
|
: Consumers(C.begin(), C.end()), MutationListener(),
|
|
DeserializationListener() {
|
|
// Collect the mutation listeners and deserialization listeners of all
|
|
// children, and create a multiplex listener each if so.
|
|
std::vector<ASTMutationListener*> mutationListeners;
|
|
std::vector<ASTDeserializationListener*> serializationListeners;
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i) {
|
|
ASTMutationListener* mutationListener =
|
|
Consumers[i]->GetASTMutationListener();
|
|
if (mutationListener)
|
|
mutationListeners.push_back(mutationListener);
|
|
ASTDeserializationListener* serializationListener =
|
|
Consumers[i]->GetASTDeserializationListener();
|
|
if (serializationListener)
|
|
serializationListeners.push_back(serializationListener);
|
|
}
|
|
if (mutationListeners.size()) {
|
|
MutationListener.reset(new MultiplexASTMutationListener(mutationListeners));
|
|
}
|
|
if (serializationListeners.size()) {
|
|
DeserializationListener.reset(
|
|
new MultiplexASTDeserializationListener(serializationListeners));
|
|
}
|
|
}
|
|
|
|
MultiplexConsumer::~MultiplexConsumer() {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
delete Consumers[i];
|
|
}
|
|
|
|
void MultiplexConsumer::Initialize(ASTContext &Context) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->Initialize(Context);
|
|
}
|
|
|
|
bool MultiplexConsumer::HandleTopLevelDecl(DeclGroupRef D) {
|
|
bool Continue = true;
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Continue = Continue && Consumers[i]->HandleTopLevelDecl(D);
|
|
return Continue;
|
|
}
|
|
|
|
void MultiplexConsumer::HandleInlineMethodDefinition(CXXMethodDecl *D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleInlineMethodDefinition(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleCXXStaticMemberVarInstantiation(VarDecl *VD) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleCXXStaticMemberVarInstantiation(VD);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleInterestingDecl(DeclGroupRef D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleInterestingDecl(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleTranslationUnit(ASTContext &Ctx) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleTranslationUnit(Ctx);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleTagDeclDefinition(TagDecl *D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleTagDeclDefinition(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleTagDeclRequiredDefinition(const TagDecl *D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleTagDeclRequiredDefinition(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleCXXImplicitFunctionInstantiation(FunctionDecl *D){
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleCXXImplicitFunctionInstantiation(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleTopLevelDeclInObjCContainer(DeclGroupRef D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleTopLevelDeclInObjCContainer(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleImplicitImportDecl(ImportDecl *D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleImplicitImportDecl(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleLinkerOptionPragma(llvm::StringRef Opts) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleLinkerOptionPragma(Opts);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleDetectMismatch(llvm::StringRef Name, llvm::StringRef Value) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleDetectMismatch(Name, Value);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleDependentLibrary(llvm::StringRef Lib) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleDependentLibrary(Lib);
|
|
}
|
|
|
|
void MultiplexConsumer::CompleteTentativeDefinition(VarDecl *D) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->CompleteTentativeDefinition(D);
|
|
}
|
|
|
|
void MultiplexConsumer::HandleVTable(
|
|
CXXRecordDecl *RD, bool DefinitionRequired) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->HandleVTable(RD, DefinitionRequired);
|
|
}
|
|
|
|
ASTMutationListener *MultiplexConsumer::GetASTMutationListener() {
|
|
return MutationListener.get();
|
|
}
|
|
|
|
ASTDeserializationListener *MultiplexConsumer::GetASTDeserializationListener() {
|
|
return DeserializationListener.get();
|
|
}
|
|
|
|
void MultiplexConsumer::PrintStats() {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
Consumers[i]->PrintStats();
|
|
}
|
|
|
|
void MultiplexConsumer::InitializeSema(Sema &S) {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
if (SemaConsumer *SC = dyn_cast<SemaConsumer>(Consumers[i]))
|
|
SC->InitializeSema(S);
|
|
}
|
|
|
|
void MultiplexConsumer::ForgetSema() {
|
|
for (size_t i = 0, e = Consumers.size(); i != e; ++i)
|
|
if (SemaConsumer *SC = dyn_cast<SemaConsumer>(Consumers[i]))
|
|
SC->ForgetSema();
|
|
}
|