blob: 3bc41e52544f7838e2659bddf14dd5f0107eea83 [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.
Ken Dyck4230d522011-03-24 01:21:01 +0000141 LayoutVTT(BaseSubobject(MostDerivedClass, CharUnits::Zero()),
142 /*BaseIsVirtual=*/false);
Anders Carlsson19f191f2010-03-26 04:10:39 +0000143}
144
Anders Carlsson2c822f12010-03-26 03:56:54 +0000145llvm::Constant *
146VTTBuilder::GetAddrOfVTable(BaseSubobject Base, bool BaseIsVirtual,
147 AddressPointsMapTy& AddressPoints) {
148 if (!GenerateDefinition)
149 return 0;
Anders Carlssonc1246c82010-03-26 00:35:45 +0000150
Anders Carlsson2c822f12010-03-26 03:56:54 +0000151 if (Base.getBase() == MostDerivedClass) {
Ken Dyck4230d522011-03-24 01:21:01 +0000152 assert(Base.getBaseOffset().isZero() &&
Anders Carlsson2c822f12010-03-26 03:56:54 +0000153 "Most derived class vtable must have a zero offset!");
154 // This is a regular vtable.
155 return CGM.getVTables().GetAddrOfVTable(MostDerivedClass);
156 }
157
158 return CGM.getVTables().GenerateConstructionVTable(MostDerivedClass,
159 Base, BaseIsVirtual,
160 AddressPoints);
161}
162
163void VTTBuilder::AddVTablePointer(BaseSubobject Base, llvm::Constant *VTable,
164 const CXXRecordDecl *VTableClass,
165 const AddressPointsMapTy& AddressPoints) {
Anders Carlssonf6da6a02010-03-29 01:04:16 +0000166 // Store the vtable pointer index if we're generating the primary VTT.
Anders Carlsson80faf692010-03-29 01:12:13 +0000167 if (VTableClass == MostDerivedClass) {
Anders Carlssonf6da6a02010-03-29 01:04:16 +0000168 assert(!SecondaryVirtualPointerIndices.count(Base) &&
169 "A virtual pointer index already exists for this base subobject!");
170 SecondaryVirtualPointerIndices[Base] = VTTComponents.size();
171 }
172
Anders Carlsson2c822f12010-03-26 03:56:54 +0000173 if (!GenerateDefinition) {
Anders Carlsson19f191f2010-03-26 04:10:39 +0000174 VTTComponents.push_back(0);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000175 return;
176 }
177
178 uint64_t AddressPoint;
179 if (VTableClass != MostDerivedClass) {
180 // The vtable is a construction vtable, look in the construction vtable
181 // address points.
182 AddressPoint = AddressPoints.lookup(Base);
Anders Carlsson836d9dd2010-04-11 20:23:06 +0000183 assert(AddressPoint != 0 && "Did not find ctor vtable address point!");
Anders Carlsson2c822f12010-03-26 03:56:54 +0000184 } else {
Anders Carlsson0a4a2fd2010-03-29 02:14:35 +0000185 // Just get the address point for the regular vtable.
186 AddressPoint = CGM.getVTables().getAddressPoint(Base, VTableClass);
Anders Carlsson836d9dd2010-04-11 20:23:06 +0000187 assert(AddressPoint != 0 && "Did not find vtable address point!");
Anders Carlsson2c822f12010-03-26 03:56:54 +0000188 }
189
190 if (!AddressPoint) AddressPoint = 0;
Anders Carlsson2c822f12010-03-26 03:56:54 +0000191
192 llvm::Value *Idxs[] = {
193 llvm::ConstantInt::get(llvm::Type::getInt64Ty(CGM.getLLVMContext()), 0),
194 llvm::ConstantInt::get(llvm::Type::getInt64Ty(CGM.getLLVMContext()),
195 AddressPoint)
196 };
197
198 llvm::Constant *Init =
199 llvm::ConstantExpr::getInBoundsGetElementPtr(VTable, Idxs, 2);
200
201 const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
202 Init = llvm::ConstantExpr::getBitCast(Init, Int8PtrTy);
203
Anders Carlsson19f191f2010-03-26 04:10:39 +0000204 VTTComponents.push_back(Init);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000205}
206
Anders Carlssonc1246c82010-03-26 00:35:45 +0000207void VTTBuilder::LayoutSecondaryVTTs(BaseSubobject Base) {
208 const CXXRecordDecl *RD = Base.getBase();
209
210 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
211 E = RD->bases_end(); I != E; ++I) {
212
213 // Don't layout virtual bases.
214 if (I->isVirtual())
215 continue;
216
217 const CXXRecordDecl *BaseDecl =
218 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
219
220 const ASTRecordLayout &Layout = CGM.getContext().getASTRecordLayout(RD);
Ken Dyck4230d522011-03-24 01:21:01 +0000221 CharUnits BaseOffset = Base.getBaseOffset() +
222 Layout.getBaseClassOffset(BaseDecl);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000223
224 // Layout the VTT for this base.
225 LayoutVTT(BaseSubobject(BaseDecl, BaseOffset), /*BaseIsVirtual=*/false);
226 }
227}
228
Anders Carlsson2c822f12010-03-26 03:56:54 +0000229void
230VTTBuilder::LayoutSecondaryVirtualPointers(BaseSubobject Base,
231 bool BaseIsMorallyVirtual,
232 llvm::Constant *VTable,
233 const CXXRecordDecl *VTableClass,
234 const AddressPointsMapTy& AddressPoints,
235 VisitedVirtualBasesSetTy &VBases) {
236 const CXXRecordDecl *RD = Base.getBase();
237
238 // We're not interested in bases that don't have virtual bases, and not
239 // morally virtual bases.
240 if (!RD->getNumVBases() && !BaseIsMorallyVirtual)
241 return;
242
243 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
244 E = RD->bases_end(); I != E; ++I) {
245 const CXXRecordDecl *BaseDecl =
246 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
247
248 // Itanium C++ ABI 2.6.2:
249 // Secondary virtual pointers are present for all bases with either
250 // virtual bases or virtual function declarations overridden along a
251 // virtual path.
252 //
253 // If the base class is not dynamic, we don't want to add it, nor any
254 // of its base classes.
255 if (!BaseDecl->isDynamicClass())
256 continue;
257
258 bool BaseDeclIsMorallyVirtual = BaseIsMorallyVirtual;
259 bool BaseDeclIsNonVirtualPrimaryBase = false;
Ken Dyck4230d522011-03-24 01:21:01 +0000260 CharUnits BaseOffset;
Anders Carlsson2c822f12010-03-26 03:56:54 +0000261 if (I->isVirtual()) {
262 // Ignore virtual bases that we've already visited.
263 if (!VBases.insert(BaseDecl))
264 continue;
265
Ken Dyck4230d522011-03-24 01:21:01 +0000266 BaseOffset = MostDerivedClassLayout.getVBaseClassOffset(BaseDecl);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000267 BaseDeclIsMorallyVirtual = true;
268 } else {
269 const ASTRecordLayout &Layout = CGM.getContext().getASTRecordLayout(RD);
270
Ken Dyck4230d522011-03-24 01:21:01 +0000271 BaseOffset = Base.getBaseOffset() +
272 Layout.getBaseClassOffset(BaseDecl);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000273
Anders Carlssonc9e814b2010-11-24 23:12:57 +0000274 if (!Layout.isPrimaryBaseVirtual() &&
Anders Carlsson2c822f12010-03-26 03:56:54 +0000275 Layout.getPrimaryBase() == BaseDecl)
276 BaseDeclIsNonVirtualPrimaryBase = true;
277 }
278
279 // Itanium C++ ABI 2.6.2:
280 // Secondary virtual pointers: for each base class X which (a) has virtual
281 // bases or is reachable along a virtual path from D, and (b) is not a
282 // non-virtual primary base, the address of the virtual table for X-in-D
283 // or an appropriate construction virtual table.
284 if (!BaseDeclIsNonVirtualPrimaryBase &&
285 (BaseDecl->getNumVBases() || BaseDeclIsMorallyVirtual)) {
286 // Add the vtable pointer.
Ken Dyck4230d522011-03-24 01:21:01 +0000287 AddVTablePointer(BaseSubobject(BaseDecl, BaseOffset), VTable,
288 VTableClass, AddressPoints);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000289 }
290
291 // And lay out the secondary virtual pointers for the base class.
292 LayoutSecondaryVirtualPointers(BaseSubobject(BaseDecl, BaseOffset),
293 BaseDeclIsMorallyVirtual, VTable,
294 VTableClass, AddressPoints, VBases);
295 }
296}
297
298void
299VTTBuilder::LayoutSecondaryVirtualPointers(BaseSubobject Base,
300 llvm::Constant *VTable,
301 const AddressPointsMapTy& AddressPoints) {
302 VisitedVirtualBasesSetTy VBases;
303 LayoutSecondaryVirtualPointers(Base, /*BaseIsMorallyVirtual=*/false,
304 VTable, Base.getBase(), AddressPoints, VBases);
305}
306
Anders Carlsson9f17d412010-03-26 00:50:17 +0000307void VTTBuilder::LayoutVirtualVTTs(const CXXRecordDecl *RD,
308 VisitedVirtualBasesSetTy &VBases) {
309 for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
310 E = RD->bases_end(); I != E; ++I) {
311 const CXXRecordDecl *BaseDecl =
312 cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
313
314 // Check if this is a virtual base.
315 if (I->isVirtual()) {
316 // Check if we've seen this base before.
317 if (!VBases.insert(BaseDecl))
318 continue;
319
Ken Dyck4230d522011-03-24 01:21:01 +0000320 CharUnits BaseOffset =
321 MostDerivedClassLayout.getVBaseClassOffset(BaseDecl);
Anders Carlsson9f17d412010-03-26 00:50:17 +0000322
323 LayoutVTT(BaseSubobject(BaseDecl, BaseOffset), /*BaseIsVirtual=*/true);
324 }
325
326 // We only need to layout virtual VTTs for this base if it actually has
327 // virtual bases.
328 if (BaseDecl->getNumVBases())
329 LayoutVirtualVTTs(BaseDecl, VBases);
330 }
331}
332
Anders Carlssonc1246c82010-03-26 00:35:45 +0000333void VTTBuilder::LayoutVTT(BaseSubobject Base, bool BaseIsVirtual) {
334 const CXXRecordDecl *RD = Base.getBase();
335
336 // Itanium C++ ABI 2.6.2:
337 // An array of virtual table addresses, called the VTT, is declared for
338 // each class type that has indirect or direct virtual base classes.
339 if (RD->getNumVBases() == 0)
340 return;
Anders Carlsson19f191f2010-03-26 04:10:39 +0000341
342 bool IsPrimaryVTT = Base.getBase() == MostDerivedClass;
343
344 if (!IsPrimaryVTT) {
345 // Remember the sub-VTT index.
Anders Carlsson3855a072010-05-03 00:55:11 +0000346 SubVTTIndicies[Base] = VTTComponents.size();
Anders Carlsson19f191f2010-03-26 04:10:39 +0000347 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000348
Anders Carlsson2c822f12010-03-26 03:56:54 +0000349 AddressPointsMapTy AddressPoints;
350 llvm::Constant *VTable = GetAddrOfVTable(Base, BaseIsVirtual, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000351
Anders Carlsson2c822f12010-03-26 03:56:54 +0000352 // Add the primary vtable pointer.
353 AddVTablePointer(Base, VTable, RD, AddressPoints);
Anders Carlssonc1246c82010-03-26 00:35:45 +0000354
Anders Carlsson2c822f12010-03-26 03:56:54 +0000355 // Add the secondary VTTs.
Anders Carlssonc1246c82010-03-26 00:35:45 +0000356 LayoutSecondaryVTTs(Base);
Anders Carlsson2c822f12010-03-26 03:56:54 +0000357
358 // Add the secondary virtual pointers.
359 LayoutSecondaryVirtualPointers(Base, VTable, AddressPoints);
360
361 // If this is the primary VTT, we want to lay out virtual VTTs as well.
Anders Carlsson19f191f2010-03-26 04:10:39 +0000362 if (IsPrimaryVTT) {
Anders Carlsson2c822f12010-03-26 03:56:54 +0000363 VisitedVirtualBasesSetTy VBases;
364 LayoutVirtualVTTs(Base.getBase(), VBases);
365 }
Anders Carlssonc1246c82010-03-26 00:35:45 +0000366}
367
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000368}
369
Anders Carlsson1cbce122011-01-29 19:16:51 +0000370void
371CodeGenVTables::EmitVTTDefinition(llvm::GlobalVariable *VTT,
372 llvm::GlobalVariable::LinkageTypes Linkage,
373 const CXXRecordDecl *RD) {
374 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/true);
375
376 const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
377 const llvm::ArrayType *ArrayType =
378 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
379
380 llvm::Constant *Init =
381 llvm::ConstantArray::get(ArrayType, Builder.getVTTComponents().data(),
382 Builder.getVTTComponents().size());
383
384 VTT->setInitializer(Init);
385
386 // Set the correct linkage.
387 VTT->setLinkage(Linkage);
Anders Carlsson691222d2011-01-29 19:34:19 +0000388
389 // Set the right visibility.
Anders Carlssonfa2e99f2011-01-29 20:24:48 +0000390 CGM.setTypeVisibility(VTT, RD, CodeGenModule::TVK_ForVTT);
Anders Carlsson1cbce122011-01-29 19:16:51 +0000391}
392
393llvm::GlobalVariable *CodeGenVTables::GetAddrOfVTT(const CXXRecordDecl *RD) {
394 assert(RD->getNumVBases() && "Only classes with virtual bases need a VTT");
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000395
396 llvm::SmallString<256> OutName;
Rafael Espindolaf0be9792011-02-11 02:52:17 +0000397 llvm::raw_svector_ostream Out(OutName);
398 CGM.getCXXABI().getMangleContext().mangleCXXVTT(RD, Out);
399 Out.flush();
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000400 llvm::StringRef Name = OutName.str();
401
Anders Carlsson1cbce122011-01-29 19:16:51 +0000402 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000403
Anders Carlsson1cbce122011-01-29 19:16:51 +0000404 const llvm::Type *Int8PtrTy =
405 llvm::Type::getInt8PtrTy(CGM.getLLVMContext());
406 const llvm::ArrayType *ArrayType =
407 llvm::ArrayType::get(Int8PtrTy, Builder.getVTTComponents().size());
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000408
Anders Carlsson1cbce122011-01-29 19:16:51 +0000409 llvm::GlobalVariable *GV =
410 CGM.CreateOrReplaceCXXRuntimeVariable(Name, ArrayType,
411 llvm::GlobalValue::ExternalLinkage);
412 GV->setUnnamedAddr(true);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000413 return GV;
414}
415
Anders Carlssonaf440352010-03-23 04:11:45 +0000416bool CodeGenVTables::needsVTTParameter(GlobalDecl GD) {
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000417 const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
418
419 // We don't have any virtual bases, just return early.
420 if (!MD->getParent()->getNumVBases())
421 return false;
422
423 // Check if we have a base constructor.
424 if (isa<CXXConstructorDecl>(MD) && GD.getCtorType() == Ctor_Base)
425 return true;
426
427 // Check if we have a base destructor.
428 if (isa<CXXDestructorDecl>(MD) && GD.getDtorType() == Dtor_Base)
429 return true;
430
431 return false;
432}
433
Anders Carlssonaf440352010-03-23 04:11:45 +0000434uint64_t CodeGenVTables::getSubVTTIndex(const CXXRecordDecl *RD,
Anders Carlssonc11bb212010-05-02 23:53:25 +0000435 BaseSubobject Base) {
Anders Carlsson3855a072010-05-03 00:55:11 +0000436 BaseSubobjectPairTy ClassSubobjectPair(RD, Base);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000437
Anders Carlsson3855a072010-05-03 00:55:11 +0000438 SubVTTIndiciesMapTy::iterator I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000439 if (I != SubVTTIndicies.end())
440 return I->second;
441
Anders Carlsson19f191f2010-03-26 04:10:39 +0000442 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000443
Anders Carlsson3855a072010-05-03 00:55:11 +0000444 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000445 Builder.getSubVTTIndicies().begin(),
446 E = Builder.getSubVTTIndicies().end(); I != E; ++I) {
447 // Insert all indices.
Anders Carlsson3855a072010-05-03 00:55:11 +0000448 BaseSubobjectPairTy ClassSubobjectPair(RD, I->first);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000449
Anders Carlsson3855a072010-05-03 00:55:11 +0000450 SubVTTIndicies.insert(std::make_pair(ClassSubobjectPair, I->second));
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000451 }
452
Anders Carlsson3855a072010-05-03 00:55:11 +0000453 I = SubVTTIndicies.find(ClassSubobjectPair);
Anders Carlsson58b7eee2010-01-21 16:50:45 +0000454 assert(I != SubVTTIndicies.end() && "Did not find index!");
455
456 return I->second;
457}
Anders Carlssone1dcc222010-03-26 04:23:58 +0000458
459uint64_t
460CodeGenVTables::getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
461 BaseSubobject Base) {
462 SecondaryVirtualPointerIndicesMapTy::iterator I =
463 SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
464
465 if (I != SecondaryVirtualPointerIndices.end())
466 return I->second;
467
468 VTTBuilder Builder(CGM, RD, /*GenerateDefinition=*/false);
469
470 // Insert all secondary vpointer indices.
471 for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
472 Builder.getSecondaryVirtualPointerIndices().begin(),
473 E = Builder.getSecondaryVirtualPointerIndices().end(); I != E; ++I) {
474 std::pair<const CXXRecordDecl *, BaseSubobject> Pair =
475 std::make_pair(RD, I->first);
476
477 SecondaryVirtualPointerIndices.insert(std::make_pair(Pair, I->second));
478 }
479
480 I = SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
481 assert(I != SecondaryVirtualPointerIndices.end() && "Did not find index!");
482
483 return I->second;
484}
485