blob: 8c642dc8162eb358769075ee43880002c1f17f5f [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
369llvm::GlobalVariable *
Anders Carlssonaf440352010-03-23 04:11:45 +0000370CodeGenVTables::GenerateVTT(llvm::GlobalVariable::LinkageTypes Linkage,
371 bool GenerateDefinition,
372 const CXXRecordDecl *RD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000373 // Only classes that have virtual bases need a VTT.
374 if (RD->getNumVBases() == 0)
375 return 0;
376
377 llvm::SmallString<256> OutName;
John McCall4c40d982010-08-31 07:33:07 +0000378 CGM.getCXXABI().getMangleContext().mangleCXXVTT(RD, OutName);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000379 llvm::StringRef Name = OutName.str();
380
381 D1(printf("vtt %s\n", RD->getNameAsCString()));
382
Douglas Gregorf4aac112010-05-06 22:18:21 +0000383 llvm::GlobalVariable *GV = CGM.getModule().getGlobalVariable(Name, true);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000384 if (GV == 0 || GV->isDeclaration()) {
385 const llvm::Type *Int8PtrTy =
386 llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
387
Anders Carlsson19f191f2010-03-26 04:10:39 +0000388 VTTBuilder Builder(CGM, RD, GenerateDefinition);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000389
Anders Carlsson19f191f2010-03-26 04:10:39 +0000390 const llvm::ArrayType *Type =
391 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
392
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000393 llvm::Constant *Init = 0;
394 if (GenerateDefinition)
Anders Carlsson19f191f2010-03-26 04:10:39 +0000395 Init = llvm::ConstantArray::get(Type, Builder.getVTTComponents().data(),
396 Builder.getVTTComponents().size());
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000397
398 llvm::GlobalVariable *OldGV = GV;
399 GV = new llvm::GlobalVariable(CGM.getModule(), Type, /*isConstant=*/true,
400 Linkage, Init, Name);
John McCallaf146032010-10-30 11:50:40 +0000401 CGM.setGlobalVisibility(GV, RD, /*ForDefinition*/ GenerateDefinition);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000402
403 if (OldGV) {
404 GV->takeName(OldGV);
405 llvm::Constant *NewPtr =
406 llvm::ConstantExpr::getBitCast(GV, OldGV->getType());
407 OldGV->replaceAllUsesWith(NewPtr);
408 OldGV->eraseFromParent();
409 }
410 }
411
412 return GV;
413}
414
Anders Carlssonaf440352010-03-23 04:11:45 +0000415llvm::GlobalVariable *CodeGenVTables::getVTT(const CXXRecordDecl *RD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000416 return GenerateVTT(llvm::GlobalValue::ExternalLinkage,
417 /*GenerateDefinition=*/false, RD);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000418}
419
Anders Carlssonaf440352010-03-23 04:11:45 +0000420bool CodeGenVTables::needsVTTParameter(GlobalDecl GD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000421 const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
422
423 // We don't have any virtual bases, just return early.
424 if (!MD->getParent()->getNumVBases())
425 return false;
426
427 // Check if we have a base constructor.
428 if (isa<CXXConstructorDecl>(MD) && GD.getCtorType() == Ctor_Base)
429 return true;
430
431 // Check if we have a base destructor.
432 if (isa<CXXDestructorDecl>(MD) && GD.getDtorType() == Dtor_Base)
433 return true;
434
435 return false;
436}
437
Anders Carlssonaf440352010-03-23 04:11:45 +0000438uint64_t CodeGenVTables::getSubVTTIndex(const CXXRecordDecl *RD,
Anders Carlssonc11bb212010-05-02 23:53:25 +0000439 BaseSubobject Base) {
Anders Carlsson3855a072010-05-03 00:55:11 +0000440 BaseSubobjectPairTy ClassSubobjectPair(RD, Base);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000441
Anders Carlsson3855a072010-05-03 00:55:11 +0000442 SubVTTIndiciesMapTy::iterator I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000443 if (I != SubVTTIndicies.end())
444 return I->second;
445
Anders Carlsson19f191f2010-03-26 04:10:39 +0000446 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000447
Anders Carlsson3855a072010-05-03 00:55:11 +0000448 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000449 Builder.getSubVTTIndicies().begin(),
450 E = Builder.getSubVTTIndicies().end(); I != E; ++I) {
451 // Insert all indices.
Anders Carlsson3855a072010-05-03 00:55:11 +0000452 BaseSubobjectPairTy ClassSubobjectPair(RD, I->first);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000453
Anders Carlsson3855a072010-05-03 00:55:11 +0000454 SubVTTIndicies.insert(std::make_pair(ClassSubobjectPair, I->second));
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000455 }
456
Anders Carlsson3855a072010-05-03 00:55:11 +0000457 I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000458 assert(I != SubVTTIndicies.end() && "Did not find index!");
459
460 return I->second;
461}
Anders Carlssone1dcc222010-03-26 04:23:58 +0000462
463uint64_t
464CodeGenVTables::getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
465 BaseSubobject Base) {
466 SecondaryVirtualPointerIndicesMapTy::iterator I =
467 SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
468
469 if (I != SecondaryVirtualPointerIndices.end())
470 return I->second;
471
472 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
473
474 // Insert all secondary vpointer indices.
475 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
476 Builder.getSecondaryVirtualPointerIndices().begin(),
477 E = Builder.getSecondaryVirtualPointerIndices().end(); I != E; ++I) {
478 std::pair<const CXXRecordDecl *, BaseSubobject> Pair =
479 std::make_pair(RD, I->first);
480
481 SecondaryVirtualPointerIndices.insert(std::make_pair(Pair, I->second));
482 }
483
484 I = SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
485 assert(I != SecondaryVirtualPointerIndices.end() && "Did not find index!");
486
487 return I->second;
488}
489