Index: include/clang/Basic/DiagnosticSemaKinds.td =================================================================== --- include/clang/Basic/DiagnosticSemaKinds.td +++ include/clang/Basic/DiagnosticSemaKinds.td @@ -3308,6 +3308,9 @@ "previous declaration of variable template partial specialization is here">; def err_var_spec_no_template : Error< "no variable template matches%select{| partial}0 specialization">; +def err_var_spec_no_template_but_method : Error< + "no variable template matches specialization; " + "did you mean to use %0 as function template instead?">; // C++ Function template specializations def err_function_template_spec_no_match : Error< Index: lib/Sema/SemaTemplate.cpp =================================================================== --- lib/Sema/SemaTemplate.cpp +++ lib/Sema/SemaTemplate.cpp @@ -2419,10 +2419,27 @@ // The template-id must name a variable template. VarTemplateDecl *VarTemplate = - dyn_cast(Name.getAsTemplateDecl()); - if (!VarTemplate) + dyn_cast_or_null(Name.getAsTemplateDecl()); + if (!VarTemplate) { + FunctionTemplateDecl *FnTemplate = + dyn_cast_or_null(Name.getAsTemplateDecl()); + if (FnTemplate) { + return Diag(D.getIdentifierLoc(), diag::err_var_spec_no_template_but_method) + << FnTemplate->getDeclName(); + } + else { + if (OverloadedTemplateStorage *OST = Name.getAsOverloadedTemplate()) { + OverloadedTemplateStorage::iterator I = OST->begin(); + OverloadedTemplateStorage::iterator IEnd = OST->end(); + if (I != IEnd) { + return Diag(D.getIdentifierLoc(), diag::err_var_spec_no_template_but_method) + << (*I)->getDeclName(); + } + } + } return Diag(D.getIdentifierLoc(), diag::err_var_spec_no_template) << IsPartialSpecialization; + } // Check for unexpanded parameter packs in any of the template arguments. for (unsigned I = 0, N = TemplateArgs.size(); I != N; ++I) Index: test/SemaCXX/cxx1y-variable-templates_top_level.cpp =================================================================== --- test/SemaCXX/cxx1y-variable-templates_top_level.cpp +++ test/SemaCXX/cxx1y-variable-templates_top_level.cpp @@ -448,3 +448,12 @@ static_assert(x == 1, ""); #endif } + +namespace PR19169 { + template int* f(); + template void f(); + template<> int f; // expected-error {{no variable template matches specialization; did you mean to use 'f' as function template instead?}} + + template void g(); + template<> int g; // expected-error {{no variable template matches specialization; did you mean to use 'g' as function template instead?}} +}