From: Eli Friedman Date: Sat, 10 Mar 2012 01:39:01 +0000 (+0000) Subject: Make sure the accessors for overridden methods don't return inherited constructors... X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=540659e102670e08773986862b191ed2c46a0e86;p=clang Make sure the accessors for overridden methods don't return inherited constructors. Fixes PR12219. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@152470 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/AST/DeclCXX.cpp b/lib/AST/DeclCXX.cpp index 89a6661cbf..33d3130848 100644 --- a/lib/AST/DeclCXX.cpp +++ b/lib/AST/DeclCXX.cpp @@ -1421,19 +1421,23 @@ void CXXMethodDecl::addOverriddenMethod(const CXXMethodDecl *MD) { assert(MD->isCanonicalDecl() && "Method is not canonical!"); assert(!MD->getParent()->isDependentContext() && "Can't add an overridden method to a class template!"); + assert(MD->isVirtual() && "Method is not virtual!"); getASTContext().addOverriddenMethod(this, MD); } CXXMethodDecl::method_iterator CXXMethodDecl::begin_overridden_methods() const { + if (isa(this)) return 0; return getASTContext().overridden_methods_begin(this); } CXXMethodDecl::method_iterator CXXMethodDecl::end_overridden_methods() const { + if (isa(this)) return 0; return getASTContext().overridden_methods_end(this); } unsigned CXXMethodDecl::size_overridden_methods() const { + if (isa(this)) return 0; return getASTContext().overridden_methods_size(this); } @@ -1712,8 +1716,8 @@ bool CXXConstructorDecl::isSpecializationCopyingObject() const { const CXXConstructorDecl *CXXConstructorDecl::getInheritedConstructor() const { // Hack: we store the inherited constructor in the overridden method table - method_iterator It = begin_overridden_methods(); - if (It == end_overridden_methods()) + method_iterator It = getASTContext().overridden_methods_begin(this); + if (It == getASTContext().overridden_methods_end(this)) return 0; return cast(*It); @@ -1722,8 +1726,9 @@ const CXXConstructorDecl *CXXConstructorDecl::getInheritedConstructor() const { void CXXConstructorDecl::setInheritedConstructor(const CXXConstructorDecl *BaseCtor){ // Hack: we store the inherited constructor in the overridden method table - assert(size_overridden_methods() == 0 && "Base ctor already set."); - addOverriddenMethod(BaseCtor); + assert(getASTContext().overridden_methods_size(this) == 0 && + "Base ctor already set."); + getASTContext().addOverriddenMethod(this, BaseCtor); } void CXXDestructorDecl::anchor() { } diff --git a/test/CodeGenCXX/inheriting-constructor.cpp b/test/CodeGenCXX/inheriting-constructor.cpp new file mode 100644 index 0000000000..b921a6d2bb --- /dev/null +++ b/test/CodeGenCXX/inheriting-constructor.cpp @@ -0,0 +1,9 @@ +// RUN: %clang_cc1 -std=c++11 -triple x86_64-apple-darwin10 -emit-llvm -o - %s | FileCheck %s + +// PR12219 +struct A { A(int); virtual ~A(); }; +struct B : A { using A::A; ~B(); }; +B::~B() {} +// CHECK: define void @_ZN1BD0Ev +// CHECK: define void @_ZN1BD1Ev +// CHECK: define void @_ZN1BD2Ev