From 40bd10b770813bd1471d46f514545437516aa4ba Mon Sep 17 00:00:00 2001 From: Richard Smith Date: Fri, 15 Feb 2019 00:29:04 +0000 Subject: Fix implementation of [temp.local]p4. When a template-name is looked up, we need to give injected-class-name declarations of class templates special treatment, as they denote a template rather than a type. Previously we achieved this by applying a filter to the lookup results after completing name lookup, but that is incorrect in various ways, not least of which is that it lost all information about access and how members were named, and the filtering caused us to generally lose all ambiguity errors between templates and non-templates. We now preserve the lookup results exactly, and the few places that need to map from a declaration found by name lookup into a declaration of a template do so explicitly. Deduplication of repeated lookup results of the same injected-class-name declaration is done by name lookup instead of after the fact. llvm-svn: 354091 --- clang/test/SemaTemplate/temp.cpp | 39 +++++++++++++++++++++++++++++++++++---- 1 file changed, 35 insertions(+), 4 deletions(-) (limited to 'clang/test/SemaTemplate/temp.cpp') diff --git a/clang/test/SemaTemplate/temp.cpp b/clang/test/SemaTemplate/temp.cpp index e037f0f0713..a8a2daeac31 100644 --- a/clang/test/SemaTemplate/temp.cpp +++ b/clang/test/SemaTemplate/temp.cpp @@ -8,12 +8,43 @@ namespace test0 { // PR7252 namespace test1 { - namespace A { template struct Base { typedef T t; }; } // expected-note {{member found}} + namespace A { template struct Base { typedef T t; }; } // expected-note 3{{member}} namespace B { template struct Base { typedef T t; }; } // expected-note {{member found}} template struct Derived : A::Base, B::Base { - // FIXME: the syntax error here is unfortunate - typename Derived::Base::t x; // expected-error {{found in multiple base classes of different types}} \ - // expected-error {{expected member name or ';'}} + typename Derived::Base::t x; // expected-error {{found in multiple base classes of different types}} }; + + class X : A::Base {}; // expected-note 2{{private}} + class Y : A::Base {}; + struct Z : A::Base {}; + struct Use1 : X, Y { + Base b1; // expected-error {{private}} + Use1::Base b2; // expected-error {{private}} + }; + struct Use2 : Z, Y { + Base b1; + Use2::Base b2; + }; + struct Use3 : X, Z { + Base b1; + Use3::Base b2; + }; +} + +namespace test2 { + struct A { static int x; }; // expected-note 4{{member}} + struct B { template static T x(); }; // expected-note 4{{member}} + struct C { template struct x {}; }; // expected-note 3{{member}} + struct D { template static T x(); }; // expected-note {{member}} + + template struct X : T... {}; + + void f() { + X::x(); // expected-error {{found in multiple base classes of different types}} + X::x(); // expected-error {{found in multiple base classes of different types}} + X::x(); // expected-error {{found in multiple base classes of different types}} + X::x(); // expected-error {{found in multiple base classes of different types}} + X::x(); // expected-error {{found in multiple base classes of different types}} + } } -- cgit v1.2.3