diff options
author | Anders Carlsson <andersca@mac.com> | 2010-02-14 17:05:59 +0000 |
---|---|---|
committer | Anders Carlsson <andersca@mac.com> | 2010-02-14 17:05:59 +0000 |
commit | b828afa89a3d50c4370199a7c6c04facfd843b47 (patch) | |
tree | dac88d3630a8dd4e788d73b1610af1c7e2fd40ac | |
parent | d999b3736ce2646ae0711416b421d906298764db (diff) |
Don't compute final overriders or build vtables for bases that don't need a vtable.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@96171 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r-- | lib/CodeGen/CGVtable.cpp | 20 | ||||
-rw-r--r-- | test/CodeGenCXX/vtable-layout.cpp | 19 |
2 files changed, 36 insertions, 3 deletions
diff --git a/lib/CodeGen/CGVtable.cpp b/lib/CodeGen/CGVtable.cpp index 878e563d84..2f952c54b6 100644 --- a/lib/CodeGen/CGVtable.cpp +++ b/lib/CodeGen/CGVtable.cpp @@ -498,6 +498,10 @@ void FinalOverriders::ComputeFinalOverriders(BaseSubobject Base, const CXXRecordDecl *BaseDecl = cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl()); + // Ignore bases that don't have any virtual member functions. + if (!BaseDecl->isPolymorphic()) + continue; + uint64_t BaseOffset; if (I->isVirtual()) { BaseOffset = MostDerivedClassLayout.getVBaseClassOffset(BaseDecl); @@ -528,6 +532,10 @@ void FinalOverriders::dump(llvm::raw_ostream &Out, BaseSubobject Base) { const CXXRecordDecl *BaseDecl = cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl()); + // Ignore bases that don't have any virtual member functions. + if (!BaseDecl->isPolymorphic()) + continue; + uint64_t BaseOffset; if (I->isVirtual()) { if (!VisitedVirtualBases.insert(BaseDecl)) { @@ -949,7 +957,9 @@ VtableBuilder::AddMethods(BaseSubobject Base, PrimaryBasesSetTy &PrimaryBases) { void VtableBuilder::layoutVtable(BaseSubobject Base) { const CXXRecordDecl *RD = Base.getBase(); - + + assert(RD->isDynamicClass() && "class does not have a vtable!"); + // First, add the offset to top. // FIXME: This is not going to be right for construction vtables. // FIXME: We should not use / 8 here. @@ -987,11 +997,15 @@ void VtableBuilder::layoutVtable(BaseSubobject Base) { E = RD->bases_end(); I != E; ++I) { const CXXRecordDecl *BaseDecl = cast<CXXRecordDecl>(I->getType()->getAs<RecordType>()->getDecl()); - + // Ignore the primary base. if (BaseDecl == PrimaryBase) continue; - + + // Ignore bases that don't have a vtable. + if (!BaseDecl->isDynamicClass()) + continue; + assert(!I->isVirtual() && "FIXME: Handle virtual bases"); // Get the base offset of this base. diff --git a/test/CodeGenCXX/vtable-layout.cpp b/test/CodeGenCXX/vtable-layout.cpp index e151de14ed..a41aad41f9 100644 --- a/test/CodeGenCXX/vtable-layout.cpp +++ b/test/CodeGenCXX/vtable-layout.cpp @@ -321,3 +321,22 @@ struct D : C, B1, B2 { void D::f() { } } + +namespace Test8 { + +// Test that we don't try to layout vtables for classes that don't have +// virtual bases or virtual member functions. + +struct A { }; + +// CHECK: Vtable for 'Test8::B' (3 entries). +// CHECK-NEXT: 0 | offset_to_top (0) +// CHECK-NEXT: 1 | Test8::B RTTI +// CHECK-NEXT: -- (Test8::B, 0) vtable address -- +// CHECK-NEXT: 2 | void Test8::B::f() +struct B : A { + virtual void f(); +}; +void B::f() { } + +} |