blob: ba83580cb69c980240f7b72f62401237b097269e [file] [log] [blame]
Nico Weber5aa74af2011-01-25 20:34:14 +00001//===- MultiplexConsumer.cpp - AST Consumer for PCH Generation --*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file defines the MultiplexConsumer class. It also declares and defines
11// MultiplexASTDeserializationListener and MultiplexASTMutationListener, which
12// are implementation details of MultiplexConsumer.
13//
14//===----------------------------------------------------------------------===//
15
16#include "clang/Frontend/MultiplexConsumer.h"
Nico Weber5aa74af2011-01-25 20:34:14 +000017#include "clang/AST/ASTMutationListener.h"
18#include "clang/AST/DeclGroup.h"
19#include "clang/Serialization/ASTDeserializationListener.h"
20
21using namespace clang;
22
23namespace clang {
24
25// This ASTDeserializationListener forwards its notifications to a set of
26// child listeners.
27class MultiplexASTDeserializationListener
28 : public ASTDeserializationListener {
29public:
30 // Does NOT take ownership of the elements in L.
31 MultiplexASTDeserializationListener(
32 const std::vector<ASTDeserializationListener*>& L);
33 virtual void ReaderInitialized(ASTReader *Reader);
34 virtual void IdentifierRead(serialization::IdentID ID,
35 IdentifierInfo *II);
36 virtual void TypeRead(serialization::TypeIdx Idx, QualType T);
37 virtual void DeclRead(serialization::DeclID ID, const Decl *D);
38 virtual void SelectorRead(serialization::SelectorID iD, Selector Sel);
Argyrios Kyrtzidise24692b2011-09-15 18:02:56 +000039 virtual void MacroDefinitionRead(serialization::PreprocessedEntityID,
Nico Weber5aa74af2011-01-25 20:34:14 +000040 MacroDefinition *MD);
41private:
42 std::vector<ASTDeserializationListener*> Listeners;
43};
44
45MultiplexASTDeserializationListener::MultiplexASTDeserializationListener(
46 const std::vector<ASTDeserializationListener*>& L)
47 : Listeners(L) {
48}
49
50void MultiplexASTDeserializationListener::ReaderInitialized(
51 ASTReader *Reader) {
52 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
53 Listeners[i]->ReaderInitialized(Reader);
54}
55
56void MultiplexASTDeserializationListener::IdentifierRead(
57 serialization::IdentID ID, IdentifierInfo *II) {
58 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
59 Listeners[i]->IdentifierRead(ID, II);
60}
61
62void MultiplexASTDeserializationListener::TypeRead(
63 serialization::TypeIdx Idx, QualType T) {
64 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
65 Listeners[i]->TypeRead(Idx, T);
66}
67
68void MultiplexASTDeserializationListener::DeclRead(
69 serialization::DeclID ID, const Decl *D) {
70 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
71 Listeners[i]->DeclRead(ID, D);
72}
73
74void MultiplexASTDeserializationListener::SelectorRead(
75 serialization::SelectorID ID, Selector Sel) {
76 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
77 Listeners[i]->SelectorRead(ID, Sel);
78}
79
80void MultiplexASTDeserializationListener::MacroDefinitionRead(
Argyrios Kyrtzidise24692b2011-09-15 18:02:56 +000081 serialization::PreprocessedEntityID ID, MacroDefinition *MD) {
Nico Weber5aa74af2011-01-25 20:34:14 +000082 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
83 Listeners[i]->MacroDefinitionRead(ID, MD);
84}
85
86// This ASTMutationListener forwards its notifications to a set of
87// child listeners.
88class MultiplexASTMutationListener : public ASTMutationListener {
89public:
90 // Does NOT take ownership of the elements in L.
Argyrios Kyrtzidis409e2452012-02-10 20:10:38 +000091 MultiplexASTMutationListener(ArrayRef<ASTMutationListener*> L);
Nico Weber5aa74af2011-01-25 20:34:14 +000092 virtual void CompletedTagDefinition(const TagDecl *D);
93 virtual void AddedVisibleDecl(const DeclContext *DC, const Decl *D);
94 virtual void AddedCXXImplicitMember(const CXXRecordDecl *RD, const Decl *D);
95 virtual void AddedCXXTemplateSpecialization(const ClassTemplateDecl *TD,
96 const ClassTemplateSpecializationDecl *D);
Sebastian Redl5bbcdbf2011-04-14 14:07:59 +000097 virtual void AddedCXXTemplateSpecialization(const FunctionTemplateDecl *TD,
98 const FunctionDecl *D);
Sebastian Redl58a2cd82011-04-24 16:28:06 +000099 virtual void CompletedImplicitDefinition(const FunctionDecl *D);
Sebastian Redlf79a7192011-04-29 08:19:30 +0000100 virtual void StaticDataMemberInstantiated(const VarDecl *D);
Argyrios Kyrtzidis1da95db2012-02-10 20:10:36 +0000101 virtual void AddedObjCCategoryToInterface(const ObjCCategoryDecl *CatD,
102 const ObjCInterfaceDecl *IFD);
103 virtual void AddedObjCPropertyInClassExtension(const ObjCPropertyDecl *Prop,
104 const ObjCPropertyDecl *OrigProp,
105 const ObjCCategoryDecl *ClassExt);
Nico Weber5aa74af2011-01-25 20:34:14 +0000106private:
107 std::vector<ASTMutationListener*> Listeners;
108};
109
110MultiplexASTMutationListener::MultiplexASTMutationListener(
Argyrios Kyrtzidis409e2452012-02-10 20:10:38 +0000111 ArrayRef<ASTMutationListener*> L)
112 : Listeners(L.begin(), L.end()) {
Nico Weber5aa74af2011-01-25 20:34:14 +0000113}
114
115void MultiplexASTMutationListener::CompletedTagDefinition(const TagDecl *D) {
116 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
117 Listeners[i]->CompletedTagDefinition(D);
118}
119
120void MultiplexASTMutationListener::AddedVisibleDecl(
121 const DeclContext *DC, const Decl *D) {
122 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
123 Listeners[i]->AddedVisibleDecl(DC, D);
124}
125
126void MultiplexASTMutationListener::AddedCXXImplicitMember(
127 const CXXRecordDecl *RD, const Decl *D) {
128 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
129 Listeners[i]->AddedCXXImplicitMember(RD, D);
130}
131void MultiplexASTMutationListener::AddedCXXTemplateSpecialization(
132 const ClassTemplateDecl *TD, const ClassTemplateSpecializationDecl *D) {
133 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
134 Listeners[i]->AddedCXXTemplateSpecialization(TD, D);
135}
Sebastian Redl5bbcdbf2011-04-14 14:07:59 +0000136void MultiplexASTMutationListener::AddedCXXTemplateSpecialization(
137 const FunctionTemplateDecl *TD, const FunctionDecl *D) {
138 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
139 Listeners[i]->AddedCXXTemplateSpecialization(TD, D);
140}
Sebastian Redl58a2cd82011-04-24 16:28:06 +0000141void MultiplexASTMutationListener::CompletedImplicitDefinition(
142 const FunctionDecl *D) {
143 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
144 Listeners[i]->CompletedImplicitDefinition(D);
145}
Sebastian Redlf79a7192011-04-29 08:19:30 +0000146void MultiplexASTMutationListener::StaticDataMemberInstantiated(
147 const VarDecl *D) {
148 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
149 Listeners[i]->StaticDataMemberInstantiated(D);
150}
Argyrios Kyrtzidis1da95db2012-02-10 20:10:36 +0000151void MultiplexASTMutationListener::AddedObjCCategoryToInterface(
152 const ObjCCategoryDecl *CatD,
153 const ObjCInterfaceDecl *IFD) {
154 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
155 Listeners[i]->AddedObjCCategoryToInterface(CatD, IFD);
156}
157void MultiplexASTMutationListener::AddedObjCPropertyInClassExtension(
158 const ObjCPropertyDecl *Prop,
159 const ObjCPropertyDecl *OrigProp,
160 const ObjCCategoryDecl *ClassExt) {
161 for (size_t i = 0, e = Listeners.size(); i != e; ++i)
162 Listeners[i]->AddedObjCPropertyInClassExtension(Prop, OrigProp, ClassExt);
163}
Nico Weber5aa74af2011-01-25 20:34:14 +0000164
165} // end namespace clang
166
167
Argyrios Kyrtzidis9ec9ce12012-02-04 05:23:00 +0000168MultiplexConsumer::MultiplexConsumer(ArrayRef<ASTConsumer*> C)
169 : Consumers(C.begin(), C.end()),
170 MutationListener(0), DeserializationListener(0) {
Nico Weber5aa74af2011-01-25 20:34:14 +0000171 // Collect the mutation listeners and deserialization listeners of all
172 // children, and create a multiplex listener each if so.
173 std::vector<ASTMutationListener*> mutationListeners;
174 std::vector<ASTDeserializationListener*> serializationListeners;
175 for (size_t i = 0, e = Consumers.size(); i != e; ++i) {
176 ASTMutationListener* mutationListener =
177 Consumers[i]->GetASTMutationListener();
178 if (mutationListener)
179 mutationListeners.push_back(mutationListener);
180 ASTDeserializationListener* serializationListener =
181 Consumers[i]->GetASTDeserializationListener();
182 if (serializationListener)
183 serializationListeners.push_back(serializationListener);
184 }
185 if (mutationListeners.size()) {
186 MutationListener.reset(new MultiplexASTMutationListener(mutationListeners));
187 }
188 if (serializationListeners.size()) {
189 DeserializationListener.reset(
190 new MultiplexASTDeserializationListener(serializationListeners));
191 }
192}
193
194MultiplexConsumer::~MultiplexConsumer() {
195 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
196 delete Consumers[i];
197}
198
199void MultiplexConsumer::Initialize(ASTContext &Context) {
200 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
201 Consumers[i]->Initialize(Context);
202}
203
Argyrios Kyrtzidis88c25962011-11-18 00:26:59 +0000204bool MultiplexConsumer::HandleTopLevelDecl(DeclGroupRef D) {
Argyrios Kyrtzidis6f3ce972011-11-28 04:56:00 +0000205 bool Continue = true;
Nico Weber5aa74af2011-01-25 20:34:14 +0000206 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
Argyrios Kyrtzidis6f3ce972011-11-28 04:56:00 +0000207 Continue = Continue && Consumers[i]->HandleTopLevelDecl(D);
208 return Continue;
Nico Weber5aa74af2011-01-25 20:34:14 +0000209}
210
Rafael Espindola02503932012-03-08 15:51:03 +0000211void MultiplexConsumer::HandleCXXStaticMemberVarInstantiation(VarDecl *VD) {
Rafael Espindola234fe652012-03-05 10:54:55 +0000212 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
Rafael Espindola02503932012-03-08 15:51:03 +0000213 Consumers[i]->HandleCXXStaticMemberVarInstantiation(VD);
Rafael Espindola234fe652012-03-05 10:54:55 +0000214}
215
Nico Weber5aa74af2011-01-25 20:34:14 +0000216void MultiplexConsumer::HandleInterestingDecl(DeclGroupRef D) {
217 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
218 Consumers[i]->HandleInterestingDecl(D);
219}
220
221void MultiplexConsumer::HandleTranslationUnit(ASTContext &Ctx) {
222 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
223 Consumers[i]->HandleTranslationUnit(Ctx);
224}
225
226void MultiplexConsumer::HandleTagDeclDefinition(TagDecl *D) {
227 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
228 Consumers[i]->HandleTagDeclDefinition(D);
229}
230
Argyrios Kyrtzidis6d968362012-02-10 20:10:44 +0000231void MultiplexConsumer::HandleCXXImplicitFunctionInstantiation(FunctionDecl *D){
232 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
233 Consumers[i]->HandleCXXImplicitFunctionInstantiation(D);
234}
235
Argyrios Kyrtzidis6f3ce972011-11-28 04:56:00 +0000236void MultiplexConsumer::HandleTopLevelDeclInObjCContainer(DeclGroupRef D) {
237 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
238 Consumers[i]->HandleTopLevelDeclInObjCContainer(D);
239}
240
Nico Weber5aa74af2011-01-25 20:34:14 +0000241void MultiplexConsumer::CompleteTentativeDefinition(VarDecl *D) {
242 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
243 Consumers[i]->CompleteTentativeDefinition(D);
244}
245
246void MultiplexConsumer::HandleVTable(
247 CXXRecordDecl *RD, bool DefinitionRequired) {
248 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
249 Consumers[i]->HandleVTable(RD, DefinitionRequired);
250}
251
252ASTMutationListener *MultiplexConsumer::GetASTMutationListener() {
253 return MutationListener.get();
254}
255
256ASTDeserializationListener *MultiplexConsumer::GetASTDeserializationListener() {
257 return DeserializationListener.get();
258}
259
260void MultiplexConsumer::PrintStats() {
261 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
262 Consumers[i]->PrintStats();
263}
264
265void MultiplexConsumer::InitializeSema(Sema &S) {
266 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
267 if (SemaConsumer *SC = dyn_cast<SemaConsumer>(Consumers[i]))
268 SC->InitializeSema(S);
269}
270
271void MultiplexConsumer::ForgetSema() {
272 for (size_t i = 0, e = Consumers.size(); i != e; ++i)
273 if (SemaConsumer *SC = dyn_cast<SemaConsumer>(Consumers[i]))
274 SC->ForgetSema();
275}