forked from OSchip/llvm-project
List implicit operator== after implicit destructors in a vtable.
Summary: We previously listed first declared members, then implicit operator=, then implicit operator==, then implicit destructors. Per discussion on https://github.com/itanium-cxx-abi/cxx-abi/issues/88, put the implicit equality comparison operators at the very end, after all special member functions. This reinstatesadd2b7e44a
, reverted in commit89e43f04ba
, with a fix for 32-bit targets. Reviewers: rjmccall Subscribers: cfe-commits Tags: #clang Differential Revision: https://reviews.llvm.org/D72897
This commit is contained in:
parent
fa001767f0
commit
6e73fee780
|
@ -1474,11 +1474,11 @@ void ItaniumVTableBuilder::AddMethods(
|
||||||
llvm_unreachable("Found a duplicate primary base!");
|
llvm_unreachable("Found a duplicate primary base!");
|
||||||
}
|
}
|
||||||
|
|
||||||
const CXXDestructorDecl *ImplicitVirtualDtor = nullptr;
|
|
||||||
|
|
||||||
typedef llvm::SmallVector<const CXXMethodDecl *, 8> NewVirtualFunctionsTy;
|
typedef llvm::SmallVector<const CXXMethodDecl *, 8> NewVirtualFunctionsTy;
|
||||||
NewVirtualFunctionsTy NewVirtualFunctions;
|
NewVirtualFunctionsTy NewVirtualFunctions;
|
||||||
|
|
||||||
|
llvm::SmallVector<const CXXMethodDecl*, 4> NewImplicitVirtualFunctions;
|
||||||
|
|
||||||
// Now go through all virtual member functions and add them.
|
// Now go through all virtual member functions and add them.
|
||||||
for (const auto *MD : RD->methods()) {
|
for (const auto *MD : RD->methods()) {
|
||||||
if (!MD->isVirtual())
|
if (!MD->isVirtual())
|
||||||
|
@ -1542,24 +1542,30 @@ void ItaniumVTableBuilder::AddMethods(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (const CXXDestructorDecl *DD = dyn_cast<CXXDestructorDecl>(MD)) {
|
if (MD->isImplicit())
|
||||||
if (MD->isImplicit()) {
|
NewImplicitVirtualFunctions.push_back(MD);
|
||||||
// Itanium C++ ABI 2.5.2:
|
else
|
||||||
// If a class has an implicitly-defined virtual destructor,
|
NewVirtualFunctions.push_back(MD);
|
||||||
// its entries come after the declared virtual function pointers.
|
|
||||||
|
|
||||||
assert(!ImplicitVirtualDtor &&
|
|
||||||
"Did already see an implicit virtual dtor!");
|
|
||||||
ImplicitVirtualDtor = DD;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
NewVirtualFunctions.push_back(MD);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ImplicitVirtualDtor)
|
std::stable_sort(
|
||||||
NewVirtualFunctions.push_back(ImplicitVirtualDtor);
|
NewImplicitVirtualFunctions.begin(), NewImplicitVirtualFunctions.end(),
|
||||||
|
[](const CXXMethodDecl *A, const CXXMethodDecl *B) {
|
||||||
|
if (A->isCopyAssignmentOperator() != B->isCopyAssignmentOperator())
|
||||||
|
return A->isCopyAssignmentOperator();
|
||||||
|
if (A->isMoveAssignmentOperator() != B->isMoveAssignmentOperator())
|
||||||
|
return A->isMoveAssignmentOperator();
|
||||||
|
if (isa<CXXDestructorDecl>(A) != isa<CXXDestructorDecl>(B))
|
||||||
|
return isa<CXXDestructorDecl>(A);
|
||||||
|
assert(A->getOverloadedOperator() == OO_EqualEqual &&
|
||||||
|
B->getOverloadedOperator() == OO_EqualEqual &&
|
||||||
|
"unexpected or duplicate implicit virtual function");
|
||||||
|
// We rely on Sema to have declared the operator== members in the
|
||||||
|
// same order as the corresponding operator<=> members.
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
NewVirtualFunctions.append(NewImplicitVirtualFunctions.begin(),
|
||||||
|
NewImplicitVirtualFunctions.end());
|
||||||
|
|
||||||
for (const CXXMethodDecl *MD : NewVirtualFunctions) {
|
for (const CXXMethodDecl *MD : NewVirtualFunctions) {
|
||||||
// Get the final overrider.
|
// Get the final overrider.
|
||||||
|
|
|
@ -0,0 +1,53 @@
|
||||||
|
// RUN: %clang_cc1 -std=c++2a -triple %itanium_abi_triple -emit-llvm %s -o - | FileCheck %s
|
||||||
|
|
||||||
|
#include "Inputs/std-compare.h"
|
||||||
|
|
||||||
|
// CHECK: @_ZTV1A =
|
||||||
|
struct A;
|
||||||
|
struct X {
|
||||||
|
// CHECK-SAME: @_ZN1X1xEv
|
||||||
|
virtual void x();
|
||||||
|
friend auto operator<=>(X, X) = default;
|
||||||
|
};
|
||||||
|
struct Y {
|
||||||
|
virtual ~Y();
|
||||||
|
virtual A &operator=(const A &);
|
||||||
|
friend auto operator<=>(Y, Y) = default;
|
||||||
|
};
|
||||||
|
struct A : X, Y {
|
||||||
|
// CHECK-SAME: @_ZN1A1fEv
|
||||||
|
virtual void f();
|
||||||
|
// CHECK-SAME: @_ZNKR1AssERKS_
|
||||||
|
virtual std::strong_ordering operator<=>(const A &) const & = default;
|
||||||
|
// CHECK-SAME: @_ZN1A1gEv
|
||||||
|
virtual void g();
|
||||||
|
// CHECK-SAME: @_ZNKO1AssERKS_
|
||||||
|
virtual std::strong_ordering operator<=>(const A &) const && = default;
|
||||||
|
// CHECK-SAME: @_ZN1A1hEv
|
||||||
|
virtual void h();
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZN1AaSERKS_
|
||||||
|
// implicit virtual A &operator=(const A&) = default;
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZN1AD1Ev
|
||||||
|
// CHECK-SAME: @_ZN1AD0Ev
|
||||||
|
// implicit virtual ~A();
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZNKR1AeqERKS_
|
||||||
|
// implicit virtual A &operator==(const A&) const & = default;
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZNKO1AeqERKS_
|
||||||
|
// implicit virtual A &operator==(const A&) const && = default;
|
||||||
|
};
|
||||||
|
|
||||||
|
// For Y:
|
||||||
|
// CHECK-SAME: @_ZTI1A
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZThn{{[0-9]*}}_N1AD1Ev
|
||||||
|
// CHECK-SAME: @_ZThn{{[0-9]*}}_N1AD0Ev
|
||||||
|
// virtual ~Y();
|
||||||
|
|
||||||
|
// CHECK-SAME: @_ZThn{{[0-9]*}}_N1AaSERKS_
|
||||||
|
// virtual A &operator=(const A &);
|
||||||
|
|
||||||
|
void A::f() {}
|
Loading…
Reference in New Issue