blob: fd8f576a0441dfcccb4bb7e01b47a452b0622716 [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() +
Anders Carlssona14f5972010-10-31 23:22:37 +0000221 Layout.getBaseClassOffsetInBits(BaseDecl);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000222
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
Anders Carlssona14f5972010-10-31 23:22:37 +0000265 BaseOffset = MostDerivedClassLayout.getVBaseClassOffsetInBits(BaseDecl);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000266 BaseDeclIsMorallyVirtual = true;
267 } else {
268 const ASTRecordLayout &Layout = CGM.getContext().getASTRecordLayout(RD);
269
Anders Carlssona14f5972010-10-31 23:22:37 +0000270 BaseOffset =
271 Base.getBaseOffset() + Layout.getBaseClassOffsetInBits(BaseDecl);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000272
Anders Carlssonc9e814b2010-11-24 23:12:57 +0000273 if (!Layout.isPrimaryBaseVirtual() &&
Anders Carlsson2c822f12010-03-26 03:56:54 +0000274 Layout.getPrimaryBase() == BaseDecl)
275 BaseDeclIsNonVirtualPrimaryBase = true;
276 }
277
278 // Itanium C++ ABI 2.6.2:
279 // Secondary virtual pointers: for each base class X which (a) has virtual
280 // bases or is reachable along a virtual path from D, and (b) is not a
281 // non-virtual primary base, the address of the virtual table for X-in-D
282 // or an appropriate construction virtual table.
283 if (!BaseDeclIsNonVirtualPrimaryBase &&
284 (BaseDecl->getNumVBases() || BaseDeclIsMorallyVirtual)) {
285 // Add the vtable pointer.
286 AddVTablePointer(BaseSubobject(BaseDecl, BaseOffset), VTable, VTableClass,
287 AddressPoints);
288 }
289
290 // And lay out the secondary virtual pointers for the base class.
291 LayoutSecondaryVirtualPointers(BaseSubobject(BaseDecl, BaseOffset),
292 BaseDeclIsMorallyVirtual, VTable,
293 VTableClass, AddressPoints, VBases);
294 }
295}
296
297void
298VTTBuilder::LayoutSecondaryVirtualPointers(BaseSubobject Base,
299 llvm::Constant *VTable,
300 const AddressPointsMapTy& AddressPoints) {
301 VisitedVirtualBasesSetTy VBases;
302 LayoutSecondaryVirtualPointers(Base, /*BaseIsMorallyVirtual=*/false,
303 VTable, Base.getBase(), AddressPoints, VBases);
304}
305
Anders Carlsson9f17d412010-03-26 00:50:17 +0000306void VTTBuilder::LayoutVirtualVTTs(const CXXRecordDecl *RD,
307 VisitedVirtualBasesSetTy &VBases) {
308 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
309 E = RD->bases_end(); I != E; ++I) {
310 const CXXRecordDecl *BaseDecl =
311 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
312
313 // Check if this is a virtual base.
314 if (I->isVirtual()) {
315 // Check if we've seen this base before.
316 if (!VBases.insert(BaseDecl))
317 continue;
318
Anders Carlsson9f17d412010-03-26 00:50:17 +0000319 uint64_t BaseOffset =
Anders Carlssona14f5972010-10-31 23:22:37 +0000320 MostDerivedClassLayout.getVBaseClassOffsetInBits(BaseDecl);
Anders Carlsson9f17d412010-03-26 00:50:17 +0000321
322 LayoutVTT(BaseSubobject(BaseDecl, BaseOffset), /*BaseIsVirtual=*/true);
323 }
324
325 // We only need to layout virtual VTTs for this base if it actually has
326 // virtual bases.
327 if (BaseDecl->getNumVBases())
328 LayoutVirtualVTTs(BaseDecl, VBases);
329 }
330}
331
Anders Carlssonc1246c82010-03-26 00:35:45 +0000332void VTTBuilder::LayoutVTT(BaseSubobject Base, bool BaseIsVirtual) {
333 const CXXRecordDecl *RD = Base.getBase();
334
335 // Itanium C++ ABI 2.6.2:
336 // An array of virtual table addresses, called the VTT, is declared for
337 // each class type that has indirect or direct virtual base classes.
338 if (RD->getNumVBases() == 0)
339 return;
Anders Carlsson19f191f2010-03-26 04:10:39 +0000340
341 bool IsPrimaryVTT = Base.getBase() == MostDerivedClass;
342
343 if (!IsPrimaryVTT) {
344 // Remember the sub-VTT index.
Anders Carlsson3855a072010-05-03 00:55:11 +0000345 SubVTTIndicies[Base] = VTTComponents.size();
Anders Carlsson19f191f2010-03-26 04:10:39 +0000346 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000347
Anders Carlsson2c822f12010-03-26 03:56:54 +0000348 AddressPointsMapTy AddressPoints;
349 llvm::Constant *VTable = GetAddrOfVTable(Base, BaseIsVirtual, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000350
Anders Carlsson2c822f12010-03-26 03:56:54 +0000351 // Add the primary vtable pointer.
352 AddVTablePointer(Base, VTable, RD, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000353
Anders Carlsson2c822f12010-03-26 03:56:54 +0000354 // Add the secondary VTTs.
Anders Carlssonc1246c82010-03-26 00:35:45 +0000355 LayoutSecondaryVTTs(Base);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000356
357 // Add the secondary virtual pointers.
358 LayoutSecondaryVirtualPointers(Base, VTable, AddressPoints);
359
360 // If this is the primary VTT, we want to lay out virtual VTTs as well.
Anders Carlsson19f191f2010-03-26 04:10:39 +0000361 if (IsPrimaryVTT) {
Anders Carlsson2c822f12010-03-26 03:56:54 +0000362 VisitedVirtualBasesSetTy VBases;
363 LayoutVirtualVTTs(Base.getBase(), VBases);
364 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000365}
366
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000367}
368
Anders Carlsson1cbce122011-01-29 19:16:51 +0000369void
370CodeGenVTables::EmitVTTDefinition(llvm::GlobalVariable *VTT,
371 llvm::GlobalVariable::LinkageTypes Linkage,
372 const CXXRecordDecl *RD) {
373 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/true);
374
375 const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
376 const llvm::ArrayType *ArrayType =
377 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
378
379 llvm::Constant *Init =
380 llvm::ConstantArray::get(ArrayType, Builder.getVTTComponents().data(),
381 Builder.getVTTComponents().size());
382
383 VTT->setInitializer(Init);
384
385 // Set the correct linkage.
386 VTT->setLinkage(Linkage);
Anders Carlsson691222d2011-01-29 19:34:19 +0000387
388 // Set the right visibility.
389 CGM.setTypeVisibility(VTT, RD, /*ForRTTI=*/false);
Anders Carlsson1cbce122011-01-29 19:16:51 +0000390}
391
392llvm::GlobalVariable *CodeGenVTables::GetAddrOfVTT(const CXXRecordDecl *RD) {
393 assert(RD->getNumVBases() && "Only classes with virtual bases need a VTT");
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000394
395 llvm::SmallString<256> OutName;
John McCall4c40d982010-08-31 07:33:07 +0000396 CGM.getCXXABI().getMangleContext().mangleCXXVTT(RD, OutName);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000397 llvm::StringRef Name = OutName.str();
398
Anders Carlsson1cbce122011-01-29 19:16:51 +0000399 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000400
Anders Carlsson1cbce122011-01-29 19:16:51 +0000401 const llvm::Type *Int8PtrTy =
402 llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
403 const llvm::ArrayType *ArrayType =
404 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000405
Anders Carlsson1cbce122011-01-29 19:16:51 +0000406 llvm::GlobalVariable *GV =
407 CGM.CreateOrReplaceCXXRuntimeVariable(Name, ArrayType,
408 llvm::GlobalValue::ExternalLinkage);
409 GV->setUnnamedAddr(true);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000410 return GV;
411}
412
Anders Carlssonaf440352010-03-23 04:11:45 +0000413bool CodeGenVTables::needsVTTParameter(GlobalDecl GD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000414 const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
415
416 // We don't have any virtual bases, just return early.
417 if (!MD->getParent()->getNumVBases())
418 return false;
419
420 // Check if we have a base constructor.
421 if (isa<CXXConstructorDecl>(MD) && GD.getCtorType() == Ctor_Base)
422 return true;
423
424 // Check if we have a base destructor.
425 if (isa<CXXDestructorDecl>(MD) && GD.getDtorType() == Dtor_Base)
426 return true;
427
428 return false;
429}
430
Anders Carlssonaf440352010-03-23 04:11:45 +0000431uint64_t CodeGenVTables::getSubVTTIndex(const CXXRecordDecl *RD,
Anders Carlssonc11bb212010-05-02 23:53:25 +0000432 BaseSubobject Base) {
Anders Carlsson3855a072010-05-03 00:55:11 +0000433 BaseSubobjectPairTy ClassSubobjectPair(RD, Base);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000434
Anders Carlsson3855a072010-05-03 00:55:11 +0000435 SubVTTIndiciesMapTy::iterator I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000436 if (I != SubVTTIndicies.end())
437 return I->second;
438
Anders Carlsson19f191f2010-03-26 04:10:39 +0000439 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000440
Anders Carlsson3855a072010-05-03 00:55:11 +0000441 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000442 Builder.getSubVTTIndicies().begin(),
443 E = Builder.getSubVTTIndicies().end(); I != E; ++I) {
444 // Insert all indices.
Anders Carlsson3855a072010-05-03 00:55:11 +0000445 BaseSubobjectPairTy ClassSubobjectPair(RD, I->first);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000446
Anders Carlsson3855a072010-05-03 00:55:11 +0000447 SubVTTIndicies.insert(std::make_pair(ClassSubobjectPair, I->second));
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000448 }
449
Anders Carlsson3855a072010-05-03 00:55:11 +0000450 I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000451 assert(I != SubVTTIndicies.end() && "Did not find index!");
452
453 return I->second;
454}
Anders Carlssone1dcc222010-03-26 04:23:58 +0000455
456uint64_t
457CodeGenVTables::getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
458 BaseSubobject Base) {
459 SecondaryVirtualPointerIndicesMapTy::iterator I =
460 SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
461
462 if (I != SecondaryVirtualPointerIndices.end())
463 return I->second;
464
465 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
466
467 // Insert all secondary vpointer indices.
468 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
469 Builder.getSecondaryVirtualPointerIndices().begin(),
470 E = Builder.getSecondaryVirtualPointerIndices().end(); I != E; ++I) {
471 std::pair<const CXXRecordDecl *, BaseSubobject> Pair =
472 std::make_pair(RD, I->first);
473
474 SecondaryVirtualPointerIndices.insert(std::make_pair(Pair, I->second));
475 }
476
477 I = SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
478 assert(I != SecondaryVirtualPointerIndices.end() && "Did not find index!");
479
480 return I->second;
481}
482