// RUN: clang-cc -fsyntax-only -verify %s namespace N { namespace M { template struct Promote; template<> struct Promote { typedef int type; }; template<> struct Promote { typedef int type; }; template<> struct Promote { typedef double type; }; Promote::type *ret_intptr(int* ip) { return ip; } Promote::type *ret_intptr2(int* ip) { return ip; } } M::Promote::type *ret_intptr3(int* ip) { return ip; } M::template Promote::type *ret_intptr4(int* ip) { return ip; } } N::M::Promote::type *ret_intptr5(int* ip) { return ip; } ::N::M::Promote::type *ret_intptr6(int* ip) { return ip; } N::M::template; // expected-error{{expected template name after 'template' keyword in nested name specifier}} \ // expected-error{{expected unqualified-id}} N::M::template Promote; // expected-error{{expected '<' after 'template Promote' in nested name specifier}} \ // expected-error{{C++ requires a type specifier for all declarations}} namespace N { template struct A; template<> struct A { struct X; }; struct B; } struct ::N::A::X { int foo; }; #if 0 // FIXME: the following crashes the parser, because Sema has no way to // communicate that the "dependent" template-name N::template B doesn't // actually refer to a template. template struct TestA { typedef typename N::template B::type type; // xpected-error{{'B' following the 'template' keyword does not refer to a template}} // FIXME: should show what B *does* refer to. }; #endif