diff options
author | Hiroshi Yamauchi <yamauchi@google.com> | 2019-07-09 15:57:29 +0000 |
---|---|---|
committer | Hiroshi Yamauchi <yamauchi@google.com> | 2019-07-09 15:57:29 +0000 |
commit | d088720edad9c29ee0d622b5d69092e18a9ac0bd (patch) | |
tree | d0901bccd81be6183de39837fe06c4bc8c436455 | |
parent | e3892d84e0c1bf6b0e0c41e243726f3c06a1f772 (diff) | |
download | bcm5719-llvm-d088720edad9c29ee0d622b5d69092e18a9ac0bd.tar.gz bcm5719-llvm-d088720edad9c29ee0d622b5d69092e18a9ac0bd.zip |
Revert Revert Devirtualize destructor of final class.
Revert r364359 and recommit r364100.
r364100 was reverted as r364359 due to an internal test failure, but it was a
false alarm.
llvm-svn: 365509
-rw-r--r-- | clang/lib/CodeGen/CGExprCXX.cpp | 30 | ||||
-rw-r--r-- | clang/test/CodeGenCXX/devirtualize-dtor-final.cpp | 23 |
2 files changed, 50 insertions, 3 deletions
diff --git a/clang/lib/CodeGen/CGExprCXX.cpp b/clang/lib/CodeGen/CGExprCXX.cpp index 25b0abbc030..fdca7699888 100644 --- a/clang/lib/CodeGen/CGExprCXX.cpp +++ b/clang/lib/CodeGen/CGExprCXX.cpp @@ -1865,9 +1865,33 @@ static void EmitObjectDelete(CodeGenFunction &CGF, Dtor = RD->getDestructor(); if (Dtor->isVirtual()) { - CGF.CGM.getCXXABI().emitVirtualObjectDelete(CGF, DE, Ptr, ElementType, - Dtor); - return; + bool UseVirtualCall = true; + const Expr *Base = DE->getArgument(); + if (auto *DevirtualizedDtor = + dyn_cast_or_null<const CXXDestructorDecl>( + Dtor->getDevirtualizedMethod( + Base, CGF.CGM.getLangOpts().AppleKext))) { + UseVirtualCall = false; + const CXXRecordDecl *DevirtualizedClass = + DevirtualizedDtor->getParent(); + if (declaresSameEntity(getCXXRecord(Base), DevirtualizedClass)) { + // Devirtualized to the class of the base type (the type of the + // whole expression). + Dtor = DevirtualizedDtor; + } else { + // Devirtualized to some other type. Would need to cast the this + // pointer to that type but we don't have support for that yet, so + // do a virtual call. FIXME: handle the case where it is + // devirtualized to the derived type (the type of the inner + // expression) as in EmitCXXMemberOrOperatorMemberCallExpr. + UseVirtualCall = true; + } + } + if (UseVirtualCall) { + CGF.CGM.getCXXABI().emitVirtualObjectDelete(CGF, DE, Ptr, ElementType, + Dtor); + return; + } } } } diff --git a/clang/test/CodeGenCXX/devirtualize-dtor-final.cpp b/clang/test/CodeGenCXX/devirtualize-dtor-final.cpp new file mode 100644 index 00000000000..ea11be773f2 --- /dev/null +++ b/clang/test/CodeGenCXX/devirtualize-dtor-final.cpp @@ -0,0 +1,23 @@ +// RUN: %clang_cc1 -triple i386-unknown-unknown -std=c++11 %s -emit-llvm -o - | FileCheck %s + +namespace Test1 { + struct A { virtual ~A() {} }; + struct B final : A {}; + struct C : A { virtual ~C() final {} }; + struct D { virtual ~D() final = 0; }; + // CHECK-LABEL: define void @_ZN5Test13fooEPNS_1BE + void foo(B *b) { + // CHECK: call void @_ZN5Test11BD1Ev + delete b; + } + // CHECK-LABEL: define void @_ZN5Test14foo2EPNS_1CE + void foo2(C *c) { + // CHECK: call void @_ZN5Test11CD1Ev + delete c; + } + // CHECK-LABEL: define void @_ZN5Test14evilEPNS_1DE + void evil(D *p) { + // CHECK-NOT: call void @_ZN5Test11DD1Ev + delete p; + } +} |