blob: 68bb655fcf005c47df7f8d37aad0a250ed25fdaa [file] [log] [blame]
Anders Carlsson58b7eee2010-01-21 16:50:45 +00001//===--- CGVTT.cpp - Emit LLVM Code for C++ VTTs --------------------------===//
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 contains code dealing with C++ code generation of VTTs (vtable tables).
11//
12//===----------------------------------------------------------------------===//
13
14#include "CodeGenModule.h"
15#include "clang/AST/RecordLayout.h"
16using namespace clang;
17using namespace CodeGen;
18
19#define D1(x)
20
21namespace {
Anders Carlsson50a2b422010-03-26 00:58:21 +000022
23/// VTT builder - Class for building VTT layout information.
Anders Carlsson58b7eee2010-01-21 16:50:45 +000024class VTTBuilder {
Anders Carlsson19f191f2010-03-26 04:10:39 +000025
26 CodeGenModule &CGM;
27
Anders Carlsson50a2b422010-03-26 00:58:21 +000028 /// MostDerivedClass - The most derived class for which we're building this
29 /// vtable.
30 const CXXRecordDecl *MostDerivedClass;
31
Anders Carlsson19f191f2010-03-26 04:10:39 +000032 typedef llvm::SmallVector<llvm::Constant *, 64> VTTComponentsVectorTy;
33
34 /// VTTComponents - The VTT components.
35 VTTComponentsVectorTy VTTComponents;
Anders Carlsson50a2b422010-03-26 00:58:21 +000036
37 /// MostDerivedClassLayout - the AST record layout of the most derived class.
38 const ASTRecordLayout &MostDerivedClassLayout;
39
Anders Carlsson9f17d412010-03-26 00:50:17 +000040 typedef llvm::SmallPtrSet<const CXXRecordDecl *, 4> VisitedVirtualBasesSetTy;
41
Anders Carlsson2c822f12010-03-26 03:56:54 +000042 typedef llvm::DenseMap<BaseSubobject, uint64_t> AddressPointsMapTy;
43
Anders Carlsson19f191f2010-03-26 04:10:39 +000044 /// SubVTTIndicies - The sub-VTT indices for the bases of the most derived
Anders Carlssone1dcc222010-03-26 04:23:58 +000045 /// class.
Anders Carlsson58b7eee2010-01-21 16:50:45 +000046 llvm::DenseMap<const CXXRecordDecl *, uint64_t> SubVTTIndicies;
Anders Carlssone1dcc222010-03-26 04:23:58 +000047
48 /// SecondaryVirtualPointerIndices - The secondary virtual pointer indices of
49 /// all subobjects of the most derived class.
50 llvm::DenseMap<BaseSubobject, uint64_t> SecondaryVirtualPointerIndices;
51
Anders Carlsson19f191f2010-03-26 04:10:39 +000052 /// GenerateDefinition - Whether the VTT builder should generate LLVM IR for
53 /// the VTT.
Anders Carlsson58b7eee2010-01-21 16:50:45 +000054 bool GenerateDefinition;
Anders Carlsson58b7eee2010-01-21 16:50:45 +000055
Anders Carlsson2c822f12010-03-26 03:56:54 +000056 /// GetAddrOfVTable - Returns the address of the vtable for the base class in
57 /// the given vtable class.
58 ///
59 /// \param AddressPoints - If the returned vtable is a construction vtable,
60 /// this will hold the address points for it.
61 llvm::Constant *GetAddrOfVTable(BaseSubobject Base, bool BaseIsVirtual,
62 AddressPointsMapTy& AddressPoints);
63
64 /// AddVTablePointer - Add a vtable pointer to the VTT currently being built.
65 ///
66 /// \param AddressPoints - If the vtable is a construction vtable, this has
67 /// the address points for it.
68 void AddVTablePointer(BaseSubobject Base, llvm::Constant *VTable,
69 const CXXRecordDecl *VTableClass,
70 const AddressPointsMapTy& AddressPoints);
71
Anders Carlssonc1246c82010-03-26 00:35:45 +000072 /// LayoutSecondaryVTTs - Lay out the secondary VTTs of the given base
73 /// subobject.
74 void LayoutSecondaryVTTs(BaseSubobject Base);
75
Anders Carlsson2c822f12010-03-26 03:56:54 +000076 /// LayoutSecondaryVirtualPointers - Lay out the secondary virtual pointers
77 /// for the given base subobject.
78 ///
79 /// \param BaseIsMorallyVirtual whether the base subobject is a virtual base
80 /// or a direct or indirect base of a virtual base.
81 ///
82 /// \param AddressPoints - If the vtable is a construction vtable, this has
83 /// the address points for it.
84 void LayoutSecondaryVirtualPointers(BaseSubobject Base,
85 bool BaseIsMorallyVirtual,
86 llvm::Constant *VTable,
87 const CXXRecordDecl *VTableClass,
88 const AddressPointsMapTy& AddressPoints,
89 VisitedVirtualBasesSetTy &VBases);
90
91 /// LayoutSecondaryVirtualPointers - Lay out the secondary virtual pointers
92 /// for the given base subobject.
93 ///
94 /// \param AddressPoints - If the vtable is a construction vtable, this has
95 /// the address points for it.
96 void LayoutSecondaryVirtualPointers(BaseSubobject Base,
97 llvm::Constant *VTable,
98 const AddressPointsMapTy& AddressPoints);
99
Anders Carlsson9f17d412010-03-26 00:50:17 +0000100 /// LayoutVirtualVTTs - Lay out the VTTs for the virtual base classes of the
101 /// given record decl.
102 void LayoutVirtualVTTs(const CXXRecordDecl *RD,
103 VisitedVirtualBasesSetTy &VBases);
104
105 /// LayoutVTT - Will lay out the VTT for the given subobject, including any
106 /// secondary VTTs, secondary virtual pointers and virtual VTTs.
107 void LayoutVTT(BaseSubobject Base, bool BaseIsVirtual);
108
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000109public:
Anders Carlsson19f191f2010-03-26 04:10:39 +0000110 VTTBuilder(CodeGenModule &CGM, const CXXRecordDecl *MostDerivedClass,
111 bool GenerateDefinition);
Anders Carlssone1dcc222010-03-26 04:23:58 +0000112
Anders Carlsson19f191f2010-03-26 04:10:39 +0000113 // getVTTComponents - Returns a reference to the VTT components.
114 const VTTComponentsVectorTy &getVTTComponents() const {
115 return VTTComponents;
116 }
Anders Carlssone1dcc222010-03-26 04:23:58 +0000117
118 /// getSubVTTIndicies - Returns a reference to the sub-VTT indices.
119 const llvm::DenseMap<const CXXRecordDecl *, uint64_t> &
120 getSubVTTIndicies() const {
121 return SubVTTIndicies;
122 }
123
124 /// getSecondaryVirtualPointerIndices - Returns a reference to the secondary
125 /// virtual pointer indices.
126 const llvm::DenseMap<BaseSubobject, uint64_t> &
127 getSecondaryVirtualPointerIndices() const {
128 return SecondaryVirtualPointerIndices;
129 }
130
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000131};
Anders Carlsson2c822f12010-03-26 03:56:54 +0000132
Anders Carlsson19f191f2010-03-26 04:10:39 +0000133VTTBuilder::VTTBuilder(CodeGenModule &CGM,
134 const CXXRecordDecl *MostDerivedClass,
135 bool GenerateDefinition)
136 : CGM(CGM), MostDerivedClass(MostDerivedClass),
137 MostDerivedClassLayout(CGM.getContext().getASTRecordLayout(MostDerivedClass)),
Anders Carlsson19f191f2010-03-26 04:10:39 +0000138 GenerateDefinition(GenerateDefinition) {
139
140 // Lay out this VTT.
141 LayoutVTT(BaseSubobject(MostDerivedClass, 0), /*BaseIsVirtual=*/false);
142}
143
Anders Carlsson2c822f12010-03-26 03:56:54 +0000144llvm::Constant *
145VTTBuilder::GetAddrOfVTable(BaseSubobject Base, bool BaseIsVirtual,
146 AddressPointsMapTy& AddressPoints) {
147 if (!GenerateDefinition)
148 return 0;
Anders Carlssonc1246c82010-03-26 00:35:45 +0000149
Anders Carlsson2c822f12010-03-26 03:56:54 +0000150 if (Base.getBase() == MostDerivedClass) {
151 assert(Base.getBaseOffset() == 0 &&
152 "Most derived class vtable must have a zero offset!");
153 // This is a regular vtable.
154 return CGM.getVTables().GetAddrOfVTable(MostDerivedClass);
155 }
156
157 return CGM.getVTables().GenerateConstructionVTable(MostDerivedClass,
158 Base, BaseIsVirtual,
159 AddressPoints);
160}
161
162void VTTBuilder::AddVTablePointer(BaseSubobject Base, llvm::Constant *VTable,
163 const CXXRecordDecl *VTableClass,
164 const AddressPointsMapTy& AddressPoints) {
Anders Carlssonf6da6a02010-03-29 01:04:16 +0000165 // Store the vtable pointer index if we're generating the primary VTT.
Anders Carlsson80faf692010-03-29 01:12:13 +0000166 if (VTableClass == MostDerivedClass) {
Anders Carlssonf6da6a02010-03-29 01:04:16 +0000167 assert(!SecondaryVirtualPointerIndices.count(Base) &&
168 "A virtual pointer index already exists for this base subobject!");
169 SecondaryVirtualPointerIndices[Base] = VTTComponents.size();
170 }
171
Anders Carlsson2c822f12010-03-26 03:56:54 +0000172 if (!GenerateDefinition) {
Anders Carlsson19f191f2010-03-26 04:10:39 +0000173 VTTComponents.push_back(0);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000174 return;
175 }
176
177 uint64_t AddressPoint;
178 if (VTableClass != MostDerivedClass) {
179 // The vtable is a construction vtable, look in the construction vtable
180 // address points.
181 AddressPoint = AddressPoints.lookup(Base);
Anders Carlsson836d9dd2010-04-11 20:23:06 +0000182 assert(AddressPoint != 0 && "Did not find ctor vtable address point!");
Anders Carlsson2c822f12010-03-26 03:56:54 +0000183 } else {
Anders Carlsson0a4a2fd2010-03-29 02:14:35 +0000184 // Just get the address point for the regular vtable.
185 AddressPoint = CGM.getVTables().getAddressPoint(Base, VTableClass);
Anders Carlsson836d9dd2010-04-11 20:23:06 +0000186 assert(AddressPoint != 0 && "Did not find vtable address point!");
Anders Carlsson2c822f12010-03-26 03:56:54 +0000187 }
188
189 if (!AddressPoint) AddressPoint = 0;
Anders Carlsson2c822f12010-03-26 03:56:54 +0000190
191 llvm::Value *Idxs[] = {
192 llvm::ConstantInt::get(llvm::Type::getInt64Ty(CGM.getLLVMContext()), 0),
193 llvm::ConstantInt::get(llvm::Type::getInt64Ty(CGM.getLLVMContext()),
194 AddressPoint)
195 };
196
197 llvm::Constant *Init =
198 llvm::ConstantExpr::getInBoundsGetElementPtr(VTable, Idxs, 2);
199
200 const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
201 Init = llvm::ConstantExpr::getBitCast(Init, Int8PtrTy);
202
Anders Carlsson19f191f2010-03-26 04:10:39 +0000203 VTTComponents.push_back(Init);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000204}
205
Anders Carlssonc1246c82010-03-26 00:35:45 +0000206void VTTBuilder::LayoutSecondaryVTTs(BaseSubobject Base) {
207 const CXXRecordDecl *RD = Base.getBase();
208
209 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
210 E = RD->bases_end(); I != E; ++I) {
211
212 // Don't layout virtual bases.
213 if (I->isVirtual())
214 continue;
215
216 const CXXRecordDecl *BaseDecl =
217 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
218
219 const ASTRecordLayout &Layout = CGM.getContext().getASTRecordLayout(RD);
220 uint64_t BaseOffset = Base.getBaseOffset() +
221 Layout.getBaseClassOffset(BaseDecl);
222
223 // Layout the VTT for this base.
224 LayoutVTT(BaseSubobject(BaseDecl, BaseOffset), /*BaseIsVirtual=*/false);
225 }
226}
227
Anders Carlsson2c822f12010-03-26 03:56:54 +0000228void
229VTTBuilder::LayoutSecondaryVirtualPointers(BaseSubobject Base,
230 bool BaseIsMorallyVirtual,
231 llvm::Constant *VTable,
232 const CXXRecordDecl *VTableClass,
233 const AddressPointsMapTy& AddressPoints,
234 VisitedVirtualBasesSetTy &VBases) {
235 const CXXRecordDecl *RD = Base.getBase();
236
237 // We're not interested in bases that don't have virtual bases, and not
238 // morally virtual bases.
239 if (!RD->getNumVBases() && !BaseIsMorallyVirtual)
240 return;
241
242 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
243 E = RD->bases_end(); I != E; ++I) {
244 const CXXRecordDecl *BaseDecl =
245 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
246
247 // Itanium C++ ABI 2.6.2:
248 // Secondary virtual pointers are present for all bases with either
249 // virtual bases or virtual function declarations overridden along a
250 // virtual path.
251 //
252 // If the base class is not dynamic, we don't want to add it, nor any
253 // of its base classes.
254 if (!BaseDecl->isDynamicClass())
255 continue;
256
257 bool BaseDeclIsMorallyVirtual = BaseIsMorallyVirtual;
258 bool BaseDeclIsNonVirtualPrimaryBase = false;
259 uint64_t BaseOffset;
260 if (I->isVirtual()) {
261 // Ignore virtual bases that we've already visited.
262 if (!VBases.insert(BaseDecl))
263 continue;
264
265 BaseOffset = MostDerivedClassLayout.getVBaseClassOffset(BaseDecl);
266 BaseDeclIsMorallyVirtual = true;
267 } else {
268 const ASTRecordLayout &Layout = CGM.getContext().getASTRecordLayout(RD);
269
270 BaseOffset = Base.getBaseOffset() + Layout.getBaseClassOffset(BaseDecl);
271
272 if (!Layout.getPrimaryBaseWasVirtual() &&
273 Layout.getPrimaryBase() == BaseDecl)
274 BaseDeclIsNonVirtualPrimaryBase = true;
275 }
276
277 // Itanium C++ ABI 2.6.2:
278 // Secondary virtual pointers: for each base class X which (a) has virtual
279 // bases or is reachable along a virtual path from D, and (b) is not a
280 // non-virtual primary base, the address of the virtual table for X-in-D
281 // or an appropriate construction virtual table.
282 if (!BaseDeclIsNonVirtualPrimaryBase &&
283 (BaseDecl->getNumVBases() || BaseDeclIsMorallyVirtual)) {
284 // Add the vtable pointer.
285 AddVTablePointer(BaseSubobject(BaseDecl, BaseOffset), VTable, VTableClass,
286 AddressPoints);
287 }
288
289 // And lay out the secondary virtual pointers for the base class.
290 LayoutSecondaryVirtualPointers(BaseSubobject(BaseDecl, BaseOffset),
291 BaseDeclIsMorallyVirtual, VTable,
292 VTableClass, AddressPoints, VBases);
293 }
294}
295
296void
297VTTBuilder::LayoutSecondaryVirtualPointers(BaseSubobject Base,
298 llvm::Constant *VTable,
299 const AddressPointsMapTy& AddressPoints) {
300 VisitedVirtualBasesSetTy VBases;
301 LayoutSecondaryVirtualPointers(Base, /*BaseIsMorallyVirtual=*/false,
302 VTable, Base.getBase(), AddressPoints, VBases);
303}
304
Anders Carlsson9f17d412010-03-26 00:50:17 +0000305void VTTBuilder::LayoutVirtualVTTs(const CXXRecordDecl *RD,
306 VisitedVirtualBasesSetTy &VBases) {
307 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
308 E = RD->bases_end(); I != E; ++I) {
309 const CXXRecordDecl *BaseDecl =
310 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
311
312 // Check if this is a virtual base.
313 if (I->isVirtual()) {
314 // Check if we've seen this base before.
315 if (!VBases.insert(BaseDecl))
316 continue;
317
Anders Carlsson9f17d412010-03-26 00:50:17 +0000318 uint64_t BaseOffset =
319 MostDerivedClassLayout.getVBaseClassOffset(BaseDecl);
320
321 LayoutVTT(BaseSubobject(BaseDecl, BaseOffset), /*BaseIsVirtual=*/true);
322 }
323
324 // We only need to layout virtual VTTs for this base if it actually has
325 // virtual bases.
326 if (BaseDecl->getNumVBases())
327 LayoutVirtualVTTs(BaseDecl, VBases);
328 }
329}
330
Anders Carlssonc1246c82010-03-26 00:35:45 +0000331void VTTBuilder::LayoutVTT(BaseSubobject Base, bool BaseIsVirtual) {
332 const CXXRecordDecl *RD = Base.getBase();
333
334 // Itanium C++ ABI 2.6.2:
335 // An array of virtual table addresses, called the VTT, is declared for
336 // each class type that has indirect or direct virtual base classes.
337 if (RD->getNumVBases() == 0)
338 return;
Anders Carlsson19f191f2010-03-26 04:10:39 +0000339
340 bool IsPrimaryVTT = Base.getBase() == MostDerivedClass;
341
342 if (!IsPrimaryVTT) {
343 // Remember the sub-VTT index.
344 SubVTTIndicies[RD] = VTTComponents.size();
345 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000346
Anders Carlsson2c822f12010-03-26 03:56:54 +0000347 AddressPointsMapTy AddressPoints;
348 llvm::Constant *VTable = GetAddrOfVTable(Base, BaseIsVirtual, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000349
Anders Carlsson2c822f12010-03-26 03:56:54 +0000350 // Add the primary vtable pointer.
351 AddVTablePointer(Base, VTable, RD, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000352
Anders Carlsson2c822f12010-03-26 03:56:54 +0000353 // Add the secondary VTTs.
Anders Carlssonc1246c82010-03-26 00:35:45 +0000354 LayoutSecondaryVTTs(Base);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000355
356 // Add the secondary virtual pointers.
357 LayoutSecondaryVirtualPointers(Base, VTable, AddressPoints);
358
359 // If this is the primary VTT, we want to lay out virtual VTTs as well.
Anders Carlsson19f191f2010-03-26 04:10:39 +0000360 if (IsPrimaryVTT) {
Anders Carlsson2c822f12010-03-26 03:56:54 +0000361 VisitedVirtualBasesSetTy VBases;
362 LayoutVirtualVTTs(Base.getBase(), VBases);
363 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000364}
365
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000366}
367
368llvm::GlobalVariable *
Anders Carlssonaf440352010-03-23 04:11:45 +0000369CodeGenVTables::GenerateVTT(llvm::GlobalVariable::LinkageTypes Linkage,
370 bool GenerateDefinition,
371 const CXXRecordDecl *RD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000372 // Only classes that have virtual bases need a VTT.
373 if (RD->getNumVBases() == 0)
374 return 0;
375
376 llvm::SmallString<256> OutName;
377 CGM.getMangleContext().mangleCXXVTT(RD, OutName);
378 llvm::StringRef Name = OutName.str();
379
380 D1(printf("vtt %s\n", RD->getNameAsCString()));
381
382 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name);
383 if (GV == 0 || GV->isDeclaration()) {
384 const llvm::Type *Int8PtrTy =
385 llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
386
Anders Carlsson19f191f2010-03-26 04:10:39 +0000387 VTTBuilder Builder(CGM, RD, GenerateDefinition);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000388
Anders Carlsson19f191f2010-03-26 04:10:39 +0000389 const llvm::ArrayType *Type =
390 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
391
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000392 llvm::Constant *Init = 0;
393 if (GenerateDefinition)
Anders Carlsson19f191f2010-03-26 04:10:39 +0000394 Init = llvm::ConstantArray::get(Type, Builder.getVTTComponents().data(),
395 Builder.getVTTComponents().size());
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000396
397 llvm::GlobalVariable *OldGV = GV;
398 GV = new llvm::GlobalVariable(CGM.getModule(), Type, /*isConstant=*/true,
399 Linkage, Init, Name);
400 CGM.setGlobalVisibility(GV, RD);
401
402 if (OldGV) {
403 GV->takeName(OldGV);
404 llvm::Constant *NewPtr =
405 llvm::ConstantExpr::getBitCast(GV, OldGV->getType());
406 OldGV->replaceAllUsesWith(NewPtr);
407 OldGV->eraseFromParent();
408 }
409 }
410
411 return GV;
412}
413
Anders Carlssonaf440352010-03-23 04:11:45 +0000414llvm::GlobalVariable *CodeGenVTables::getVTT(const CXXRecordDecl *RD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000415 return GenerateVTT(llvm::GlobalValue::ExternalLinkage,
416 /*GenerateDefinition=*/false, RD);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000417}
418
Anders Carlssonaf440352010-03-23 04:11:45 +0000419bool CodeGenVTables::needsVTTParameter(GlobalDecl GD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000420 const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
421
422 // We don't have any virtual bases, just return early.
423 if (!MD->getParent()->getNumVBases())
424 return false;
425
426 // Check if we have a base constructor.
427 if (isa<CXXConstructorDecl>(MD) && GD.getCtorType() == Ctor_Base)
428 return true;
429
430 // Check if we have a base destructor.
431 if (isa<CXXDestructorDecl>(MD) && GD.getDtorType() == Dtor_Base)
432 return true;
433
434 return false;
435}
436
Anders Carlssonaf440352010-03-23 04:11:45 +0000437uint64_t CodeGenVTables::getSubVTTIndex(const CXXRecordDecl *RD,
438 const CXXRecordDecl *Base) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000439 ClassPairTy ClassPair(RD, Base);
440
Anders Carlssone1dcc222010-03-26 04:23:58 +0000441 SubVTTIndiciesMapTy::iterator I = SubVTTIndicies.find(ClassPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000442 if (I != SubVTTIndicies.end())
443 return I->second;
444
Anders Carlsson19f191f2010-03-26 04:10:39 +0000445 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000446
Anders Carlssone1dcc222010-03-26 04:23:58 +0000447 for (llvm::DenseMap<const CXXRecordDecl *, uint64_t>::const_iterator I =
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000448 Builder.getSubVTTIndicies().begin(),
449 E = Builder.getSubVTTIndicies().end(); I != E; ++I) {
450 // Insert all indices.
451 ClassPairTy ClassPair(RD, I->first);
452
453 SubVTTIndicies.insert(std::make_pair(ClassPair, I->second));
454 }
455
456 I = SubVTTIndicies.find(ClassPair);
457 assert(I != SubVTTIndicies.end() && "Did not find index!");
458
459 return I->second;
460}
Anders Carlssone1dcc222010-03-26 04:23:58 +0000461
462uint64_t
463CodeGenVTables::getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
464 BaseSubobject Base) {
465 SecondaryVirtualPointerIndicesMapTy::iterator I =
466 SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
467
468 if (I != SecondaryVirtualPointerIndices.end())
469 return I->second;
470
471 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
472
473 // Insert all secondary vpointer indices.
474 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
475 Builder.getSecondaryVirtualPointerIndices().begin(),
476 E = Builder.getSecondaryVirtualPointerIndices().end(); I != E; ++I) {
477 std::pair<const CXXRecordDecl *, BaseSubobject> Pair =
478 std::make_pair(RD, I->first);
479
480 SecondaryVirtualPointerIndices.insert(std::make_pair(Pair, I->second));
481 }
482
483 I = SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
484 assert(I != SecondaryVirtualPointerIndices.end() && "Did not find index!");
485
486 return I->second;
487}
488