1// RUN: %clang_cc1 -fsyntax-only -verify %s 2namespace N { 3 struct A { 4 typedef int type; 5 }; 6 7 struct B { 8 }; 9 10 struct C { 11 struct type { }; 12 int type; // expected-note 2{{referenced member 'type' is declared here}} 13 }; 14} 15 16int i; 17 18typename N::A::type *ip1 = &i; // expected-warning{{'typename' occurs outside of a template}} 19typename N::B::type *ip2 = &i; // expected-error{{no type named 'type' in 'N::B'}} \ 20// expected-warning{{'typename' occurs outside of a template}} 21typename N::C::type *ip3 = &i; // expected-error{{typename specifier refers to non-type member 'type'}} \ 22// expected-warning{{'typename' occurs outside of a template}} 23 24void test(double d) { 25 typename N::A::type f(typename N::A::type(a)); // expected-warning{{parentheses were disambiguated as a function declarator}} \ 26 // expected-warning 2{{'typename' occurs outside of a template}} 27 int five = f(5); 28 29 using namespace N; 30 for (typename A::type i = 0; i < 10; ++i) // expected-warning{{'typename' occurs outside of a template}} 31 five += 1; 32 33 const typename N::A::type f2(d); // expected-warning{{'typename' occurs outside of a template}} 34} 35 36namespace N { 37 template<typename T> 38 struct X { 39 typedef typename T::type type; // expected-error {{no type named 'type' in 'N::B'}} \ 40 // expected-error {{no type named 'type' in 'B'}} \ 41 // FIXME: location info for error above isn't very good \ 42 // expected-error 2{{typename specifier refers to non-type member 'type'}} \ 43 // expected-error{{type 'int' cannot be used prior to '::' because it has no members}} 44 }; 45} 46 47N::X<N::A>::type *ip4 = &i; 48N::X<N::B>::type *ip5 = &i; // expected-note{{in instantiation of template class 'N::X<N::B>' requested here}} 49N::X<N::C>::type *ip6 = &i; // expected-note{{in instantiation of template class 'N::X<N::C>' requested here}} 50 51N::X<int>::type fail1; // expected-note{{in instantiation of template class 'N::X<int>' requested here}} 52 53template<typename T> 54struct Y { 55 typedef typename N::X<T>::type *type; // expected-note{{in instantiation of template class 'N::X<B>' requested here}} \ 56 // expected-note{{in instantiation of template class 'N::X<C>' requested here}} 57}; 58 59struct A { 60 typedef int type; 61}; 62 63struct B { 64}; 65 66struct C { 67 struct type { }; 68 int type; // expected-note{{referenced member 'type' is declared here}} 69}; 70 71::Y<A>::type ip7 = &i; 72::Y<B>::type ip8 = &i; // expected-note{{in instantiation of template class 'Y<B>' requested here}} 73::Y<C>::type ip9 = &i; // expected-note{{in instantiation of template class 'Y<C>' requested here}} 74 75template<typename T> struct D { 76 typedef typename T::foo foo; // expected-error {{type 'long' cannot be used prior to '::' because it has no members}} 77 typedef typename foo::bar bar; 78}; 79 80D<long> struct_D; // expected-note {{in instantiation of template class 'D<long>' requested here}} 81 82template<typename T> struct E { 83 typedef typename T::foo foo; 84 typedef typename foo::bar bar; // expected-error {{type 'foo' (aka 'double') cannot be used prior to '::' because it has no members}} 85}; 86 87struct F { 88 typedef double foo; 89}; 90 91E<F> struct_E; // expected-note {{in instantiation of template class 'E<F>' requested here}} 92 93template<typename T> struct G { 94 typedef typename T::foo foo; 95 typedef typename foo::bar bar; 96}; 97 98struct H { 99 struct foo { 100 typedef double bar; 101 }; 102}; 103 104G<H> struct_G; 105 106namespace PR10925 { 107 template< int mydim, typename Traits > 108 class BasicGeometry 109 { 110 typedef int some_type_t; 111 }; 112 113 template<class ctype, int mydim, int coorddim> 114 class MockGeometry : BasicGeometry<mydim, int>{ 115 using typename BasicGeometry<mydim, int>::operator[]; // expected-error {{typename is allowed for identifiers only}} 116 }; 117} 118