From: John McCall Date: Tue, 24 Nov 2009 20:33:45 +0000 (+0000) Subject: Fix some major problems dealing with dependently-qualified names in implicit X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=26416068d3eb883a280fdceeffa74fffc9131031;p=clang Fix some major problems dealing with dependently-qualified names in implicit member-reference contexts. Fixes some clang-on-clang asserts. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@89796 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/SemaTemplate.cpp b/lib/Sema/SemaTemplate.cpp index 0e680f6438..8ab8d93186 100644 --- a/lib/Sema/SemaTemplate.cpp +++ b/lib/Sema/SemaTemplate.cpp @@ -327,7 +327,11 @@ static bool HasDependentTypeAsBase(ASTContext &Context, // }; CanQual RT = BaseT->getAs(); - assert(RT && "base is not a record type"); + + // Base might be a dependent member type, in which case we + // obviously can't look into it. + if (!RT) continue; + CXXRecordDecl *BaseRecord = cast(RT->getDecl()); if (BaseRecord->isDefinition() && HasDependentTypeAsBase(Context, BaseRecord, T)) @@ -364,14 +368,17 @@ static bool IsImplicitDependentMemberReference(Sema &SemaRef, QualType QT = GetTypeForQualifier(Context, Qualifier); CanQualType T = Context.getCanonicalType(QT); - + // And now, just walk the non-dependent type hierarchy, trying to // find the given type as a literal base class. CXXRecordDecl *Record = cast(MD->getParent()); - if (Context.getCanonicalType(Context.getTypeDeclType(Record)) == T) + if (Context.getCanonicalType(Context.getTypeDeclType(Record)) == T || + HasDependentTypeAsBase(Context, Record, T)) { + ThisType = MD->getThisType(Context); return true; + } - return HasDependentTypeAsBase(Context, Record, T); + return false; } /// ActOnDependentIdExpression - Handle a dependent declaration name diff --git a/test/SemaTemplate/dependent-names.cpp b/test/SemaTemplate/dependent-names.cpp index 95ee2d2b9d..aef43cb29d 100644 --- a/test/SemaTemplate/dependent-names.cpp +++ b/test/SemaTemplate/dependent-names.cpp @@ -14,3 +14,73 @@ int a0[sizeof(X::a) == sizeof(double) ? 1 : -1]; // PR4365. template class Q; template class R : Q {T current;}; + + +namespace test0 { + template class Base { + void instance_foo(); + static void static_foo(); + class Inner { + void instance_foo(); + static void static_foo(); + }; + }; + + template class Derived1 : Base { + void test0() { + Base::static_foo(); + Base::instance_foo(); + } + + void test1() { + Base::Inner::static_foo(); + Base::Inner::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + + static void test2() { + Base::static_foo(); + Base::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + + static void test3() { + Base::Inner::static_foo(); + Base::Inner::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + }; + + template class Derived2 : Base::Inner { + void test0() { + Base::static_foo(); + Base::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + + void test1() { + Base::Inner::static_foo(); + Base::Inner::instance_foo(); + } + + static void test2() { + Base::static_foo(); + Base::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + + static void test3() { + Base::Inner::static_foo(); + Base::Inner::instance_foo(); // expected-error {{call to non-static member function without an object argument}} + } + }; + + void test0() { + Derived1 d1; + d1.test0(); + d1.test1(); // expected-note {{in instantiation of member function}} + d1.test2(); // expected-note {{in instantiation of member function}} + d1.test3(); // expected-note {{in instantiation of member function}} + + Derived2 d2; + d2.test0(); // expected-note {{in instantiation of member function}} + d2.test1(); + d2.test2(); // expected-note {{in instantiation of member function}} + d2.test3(); // expected-note {{in instantiation of member function}} + } +}