More improvements to setting the vtable pointer. We now no longer set the vtable pointer for non-virtual primary bases. We also do a pre-order traversal of the class hierarchy; this is necessary in order to get the right vbase offset offsets in base ctors/dtors.

llvm-svn: 99775
This commit is contained in:
Anders Carlsson 2010-03-28 20:34:31 +00:00
parent 03719af41d
commit 74eab715ae
2 changed files with 65 additions and 56 deletions

View File

@ -690,7 +690,7 @@ CodeGenFunction::SynthesizeCXXCopyConstructor(const FunctionArgList &Args) {
}
}
InitializeVtablePtrs(ClassDecl);
InitializeVTablePointers(ClassDecl);
}
/// SynthesizeCXXCopyAssignment - Implicitly define copy assignment operator.
@ -1010,7 +1010,7 @@ void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD,
MemberInitializers.push_back(Member);
}
InitializeVtablePtrs(ClassDecl);
InitializeVTablePointers(ClassDecl);
for (unsigned I = 0, E = MemberInitializers.size(); I != E; ++I) {
assert(LiveTemporaries.empty() &&
@ -1060,7 +1060,7 @@ void CodeGenFunction::EmitDestructorBody(FunctionArgList &Args) {
// Otherwise, we're in the base variant, so we need to ensure the
// vtable ptrs are right before emitting the body.
} else {
InitializeVtablePtrs(Dtor->getParent());
InitializeVTablePointers(Dtor->getParent());
}
// Emit the body of the statement.
@ -1584,59 +1584,66 @@ CodeGenFunction::InitializeVTablePointer(BaseSubobject Base,
Builder.CreateStore(VTableAddressPoint, VTableField);
}
void CodeGenFunction::InitializeVtablePtrs(const CXXRecordDecl *RD) {
void
CodeGenFunction::InitializeVTablePointers(BaseSubobject Base,
bool BaseIsMorallyVirtual,
bool BaseIsNonVirtualPrimaryBase,
llvm::Constant *VTable,
const CXXRecordDecl *VTableClass,
VisitedVirtualBasesSetTy& VBases) {
// If this base is a non-virtual primary base the address point has already
// been set.
if (true || !BaseIsNonVirtualPrimaryBase) {
// Initialize the vtable pointer for this base.
InitializeVTablePointer(Base, BaseIsMorallyVirtual, VTable, VTableClass);
}
const CXXRecordDecl *RD = Base.getBase();
// Traverse bases.
for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
E = RD->bases_end(); I != E; ++I) {
CXXRecordDecl *BaseDecl
= cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
uint64_t BaseOffset;
if (I->isVirtual()) {
// Check if we've visited this virtual base before.
if (!VBases.insert(BaseDecl))
continue;
const ASTRecordLayout &Layout =
getContext().getASTRecordLayout(VTableClass);
BaseIsMorallyVirtual = true;
BaseIsNonVirtualPrimaryBase = false;
BaseOffset = Layout.getVBaseClassOffset(BaseDecl);
} else {
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(RD);
BaseOffset = Base.getBaseOffset() + Layout.getBaseClassOffset(BaseDecl);
BaseIsNonVirtualPrimaryBase = Layout.getPrimaryBase() == BaseDecl;
}
InitializeVTablePointers(BaseSubobject(BaseDecl, BaseOffset),
BaseIsMorallyVirtual, BaseIsNonVirtualPrimaryBase,
VTable, VTableClass, VBases);
}
}
void CodeGenFunction::InitializeVTablePointers(const CXXRecordDecl *RD) {
if (!RD->isDynamicClass())
return;
// Get the VTable.
llvm::Constant *VTable = CGM.getVTables().GetAddrOfVTable(RD);
// Store address points for the current class and its non-virtual bases.
InitializeVtablePtrs(BaseSubobject(RD, 0), VTable, RD);
if (!RD->getNumVBases())
return;
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(RD);
// Store address points for virtual basess.
for (CXXRecordDecl::base_class_const_iterator I = RD->vbases_begin(),
E = RD->vbases_end(); I != E; ++I) {
CXXRecordDecl *BaseDecl
= cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
uint64_t BaseOffset = Layout.getVBaseClassOffset(BaseDecl);
InitializeVtablePtrs(BaseSubobject(BaseDecl, BaseOffset), VTable, RD);
}
}
void CodeGenFunction::InitializeVtablePtrs(BaseSubobject Base,
llvm::Constant *VTable,
const CXXRecordDecl *VTableClass) {
const CXXRecordDecl *RD = Base.getBase();
// Ignore classes without a vtable pointer.
if (!RD->isDynamicClass())
return;
const ASTRecordLayout &Layout = getContext().getASTRecordLayout(RD);
// Store address points for non-virtual bases.
for (CXXRecordDecl::base_class_const_iterator I = RD->bases_begin(),
E = RD->bases_end(); I != E; ++I) {
if (I->isVirtual())
continue;
CXXRecordDecl *BaseDecl
= cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl());
uint64_t BaseOffset = Base.getBaseOffset() +
Layout.getBaseClassOffset(BaseDecl);
InitializeVtablePtrs(BaseSubobject(BaseDecl, BaseOffset),
VTable, VTableClass);
}
// FIXME: BaseIsMorallyVirtual is not correct here.
InitializeVTablePointer(Base, /*BaseIsMorallyVirtual=*/false, VTable,
VTableClass);
// Initialize the vtable pointers for this class and all of its bases.
VisitedVirtualBasesSetTy VBases;
InitializeVTablePointers(BaseSubobject(RD, 0),
/*BaseIsMorallyVirtual=*/false,
/*BaseIsNonVirtualPrimaryBase=*/false,
VTable, RD, VBases);
}

View File

@ -519,11 +519,13 @@ public:
const CXXRecordDecl *VTableClass);
typedef llvm::SmallPtrSet<const CXXRecordDecl *, 4> VisitedVirtualBasesSetTy;
void InitializeVTablePointers(BaseSubobject Base, bool BaseIsMorallyVirtual,
bool BaseIsNonVirtualPrimaryBase,
llvm::Constant *VTable,
const CXXRecordDecl *VTableClass,
VisitedVirtualBasesSetTy& VBases);
void InitializeVtablePtrs(BaseSubobject Base, llvm::Constant *VTable,
const CXXRecordDecl *VTableClass);
void InitializeVtablePtrs(const CXXRecordDecl *ClassDecl);
void InitializeVTablePointers(const CXXRecordDecl *ClassDecl);
void SynthesizeCXXCopyConstructor(const FunctionArgList &Args);