blob: 56acfc84802e3f845317cd9789b3de0f36118157 [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"
John McCall4c40d982010-08-31 07:33:07 +000015#include "CGCXXABI.h"
Anders Carlsson58b7eee2010-01-21 16:50:45 +000016#include "clang/AST/RecordLayout.h"
17using namespace clang;
18using namespace CodeGen;
19
20#define D1(x)
21
22namespace {
Anders Carlsson50a2b422010-03-26 00:58:21 +000023
24/// VTT builder - Class for building VTT layout information.
Anders Carlsson58b7eee2010-01-21 16:50:45 +000025class VTTBuilder {
Anders Carlsson19f191f2010-03-26 04:10:39 +000026
27 CodeGenModule &CGM;
28
Anders Carlsson50a2b422010-03-26 00:58:21 +000029 /// MostDerivedClass - The most derived class for which we're building this
30 /// vtable.
31 const CXXRecordDecl *MostDerivedClass;
32
Anders Carlsson19f191f2010-03-26 04:10:39 +000033 typedef llvm::SmallVector<llvm::Constant *, 64> VTTComponentsVectorTy;
34
35 /// VTTComponents - The VTT components.
36 VTTComponentsVectorTy VTTComponents;
Anders Carlsson50a2b422010-03-26 00:58:21 +000037
38 /// MostDerivedClassLayout - the AST record layout of the most derived class.
39 const ASTRecordLayout &MostDerivedClassLayout;
40
Anders Carlsson9f17d412010-03-26 00:50:17 +000041 typedef llvm::SmallPtrSet<const CXXRecordDecl *, 4> VisitedVirtualBasesSetTy;
42
Anders Carlsson2c822f12010-03-26 03:56:54 +000043 typedef llvm::DenseMap<BaseSubobject, uint64_t> AddressPointsMapTy;
44
Anders Carlsson19f191f2010-03-26 04:10:39 +000045 /// SubVTTIndicies - The sub-VTT indices for the bases of the most derived
Anders Carlssone1dcc222010-03-26 04:23:58 +000046 /// class.
Anders Carlsson3855a072010-05-03 00:55:11 +000047 llvm::DenseMap<BaseSubobject, uint64_t> SubVTTIndicies;
Anders Carlssone1dcc222010-03-26 04:23:58 +000048
49 /// SecondaryVirtualPointerIndices - The secondary virtual pointer indices of
50 /// all subobjects of the most derived class.
51 llvm::DenseMap<BaseSubobject, uint64_t> SecondaryVirtualPointerIndices;
52
Anders Carlsson19f191f2010-03-26 04:10:39 +000053 /// GenerateDefinition - Whether the VTT builder should generate LLVM IR for
54 /// the VTT.
Anders Carlsson58b7eee2010-01-21 16:50:45 +000055 bool GenerateDefinition;
Anders Carlsson58b7eee2010-01-21 16:50:45 +000056
Anders Carlsson2c822f12010-03-26 03:56:54 +000057 /// GetAddrOfVTable - Returns the address of the vtable for the base class in
58 /// the given vtable class.
59 ///
60 /// \param AddressPoints - If the returned vtable is a construction vtable,
61 /// this will hold the address points for it.
62 llvm::Constant *GetAddrOfVTable(BaseSubobject Base, bool BaseIsVirtual,
63 AddressPointsMapTy& AddressPoints);
64
65 /// AddVTablePointer - Add a vtable pointer to the VTT currently being built.
66 ///
67 /// \param AddressPoints - If the vtable is a construction vtable, this has
68 /// the address points for it.
69 void AddVTablePointer(BaseSubobject Base, llvm::Constant *VTable,
70 const CXXRecordDecl *VTableClass,
71 const AddressPointsMapTy& AddressPoints);
72
Anders Carlssonc1246c82010-03-26 00:35:45 +000073 /// LayoutSecondaryVTTs - Lay out the secondary VTTs of the given base
74 /// subobject.
75 void LayoutSecondaryVTTs(BaseSubobject Base);
76
Anders Carlsson2c822f12010-03-26 03:56:54 +000077 /// LayoutSecondaryVirtualPointers - Lay out the secondary virtual pointers
78 /// for the given base subobject.
79 ///
80 /// \param BaseIsMorallyVirtual whether the base subobject is a virtual base
81 /// or a direct or indirect base of a virtual base.
82 ///
83 /// \param AddressPoints - If the vtable is a construction vtable, this has
84 /// the address points for it.
85 void LayoutSecondaryVirtualPointers(BaseSubobject Base,
86 bool BaseIsMorallyVirtual,
87 llvm::Constant *VTable,
88 const CXXRecordDecl *VTableClass,
89 const AddressPointsMapTy& AddressPoints,
90 VisitedVirtualBasesSetTy &VBases);
91
92 /// LayoutSecondaryVirtualPointers - Lay out the secondary virtual pointers
93 /// for the given base subobject.
94 ///
95 /// \param AddressPoints - If the vtable is a construction vtable, this has
96 /// the address points for it.
97 void LayoutSecondaryVirtualPointers(BaseSubobject Base,
98 llvm::Constant *VTable,
99 const AddressPointsMapTy& AddressPoints);
100
Anders Carlsson9f17d412010-03-26 00:50:17 +0000101 /// LayoutVirtualVTTs - Lay out the VTTs for the virtual base classes of the
102 /// given record decl.
103 void LayoutVirtualVTTs(const CXXRecordDecl *RD,
104 VisitedVirtualBasesSetTy &VBases);
105
106 /// LayoutVTT - Will lay out the VTT for the given subobject, including any
107 /// secondary VTTs, secondary virtual pointers and virtual VTTs.
108 void LayoutVTT(BaseSubobject Base, bool BaseIsVirtual);
109
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000110public:
Anders Carlsson19f191f2010-03-26 04:10:39 +0000111 VTTBuilder(CodeGenModule &CGM, const CXXRecordDecl *MostDerivedClass,
112 bool GenerateDefinition);
Anders Carlssone1dcc222010-03-26 04:23:58 +0000113
Anders Carlsson19f191f2010-03-26 04:10:39 +0000114 // getVTTComponents - Returns a reference to the VTT components.
115 const VTTComponentsVectorTy &getVTTComponents() const {
116 return VTTComponents;
117 }
Anders Carlssone1dcc222010-03-26 04:23:58 +0000118
119 /// getSubVTTIndicies - Returns a reference to the sub-VTT indices.
Anders Carlsson3855a072010-05-03 00:55:11 +0000120 const llvm::DenseMap<BaseSubobject, uint64_t> &getSubVTTIndicies() const {
Anders Carlssone1dcc222010-03-26 04:23:58 +0000121 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.
Anders Carlsson3855a072010-05-03 00:55:11 +0000344 SubVTTIndicies[Base] = VTTComponents.size();
Anders Carlsson19f191f2010-03-26 04:10:39 +0000345 }
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;
John McCall4c40d982010-08-31 07:33:07 +0000377 CGM.getCXXABI().getMangleContext().mangleCXXVTT(RD, OutName);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000378 llvm::StringRef Name = OutName.str();
379
380 D1(printf("vtt %s\n", RD->getNameAsCString()));
381
Douglas Gregorf4aac112010-05-06 22:18:21 +0000382 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name, true);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000383 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,
Anders Carlssonc11bb212010-05-02 23:53:25 +0000438 BaseSubobject Base) {
Anders Carlsson3855a072010-05-03 00:55:11 +0000439 BaseSubobjectPairTy ClassSubobjectPair(RD, Base);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000440
Anders Carlsson3855a072010-05-03 00:55:11 +0000441 SubVTTIndiciesMapTy::iterator I = SubVTTIndicies.find(ClassSubobjectPair);
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 Carlsson3855a072010-05-03 00:55:11 +0000447 for (llvm::DenseMap<BaseSubobject, 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.
Anders Carlsson3855a072010-05-03 00:55:11 +0000451 BaseSubobjectPairTy ClassSubobjectPair(RD, I->first);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000452
Anders Carlsson3855a072010-05-03 00:55:11 +0000453 SubVTTIndicies.insert(std::make_pair(ClassSubobjectPair, I->second));
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000454 }
455
Anders Carlsson3855a072010-05-03 00:55:11 +0000456 I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000457 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