From: Richard Smith Date: Sat, 21 Apr 2012 01:27:54 +0000 (+0000) Subject: When declaring a template, check that the context doesn't already contain a X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=71c598fc0515f7a95f2e0051ca056b6114970fe5;p=clang When declaring a template, check that the context doesn't already contain a declaration of the same name. r155187 caused us to miss this if the prior declaration did not declare a type. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@155269 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Sema/SemaTemplate.cpp b/lib/Sema/SemaTemplate.cpp index e16e7d67c6..f1581e0c34 100644 --- a/lib/Sema/SemaTemplate.cpp +++ b/lib/Sema/SemaTemplate.cpp @@ -865,9 +865,13 @@ Sema::CheckClassTemplate(Scope *S, unsigned TagSpec, TagUseKind TUK, return true; } - // Find any previous declaration with this name. + // Find any previous declaration with this name. For a friend with no + // scope explicitly specified, we only look for tag declarations (per + // C++11 [basic.lookup.elab]p2). DeclContext *SemanticContext; - LookupResult Previous(*this, Name, NameLoc, LookupTagName, + LookupResult Previous(*this, Name, NameLoc, + (SS.isEmpty() && TUK == TUK_Friend) + ? LookupTagName : LookupOrdinaryName, ForRedeclaration); if (SS.isNotEmpty() && !SS.isInvalid()) { SemanticContext = computeDeclContext(SS, true); @@ -893,7 +897,7 @@ Sema::CheckClassTemplate(Scope *S, unsigned TagSpec, TagUseKind TUK, Invalid = true; } else if (TUK != TUK_Friend && TUK != TUK_Reference) diagnoseQualifiedDeclaration(SS, SemanticContext, Name, NameLoc); - + LookupQualifiedName(Previous, SemanticContext); } else { SemanticContext = CurContext; @@ -948,6 +952,21 @@ Sema::CheckClassTemplate(Scope *S, unsigned TagSpec, TagUseKind TUK, // declaration. PrevDecl = PrevClassTemplate = 0; SemanticContext = OutermostContext; + + // Check that the chosen semantic context doesn't already contain a + // declaration of this name as a non-tag type. + LookupResult Previous(*this, Name, NameLoc, LookupOrdinaryName, + ForRedeclaration); + DeclContext *LookupContext = SemanticContext; + while (LookupContext->isTransparentContext()) + LookupContext = LookupContext->getLookupParent(); + LookupQualifiedName(Previous, LookupContext); + + if (Previous.isAmbiguous()) + return true; + + if (Previous.begin() != Previous.end()) + PrevDecl = (*Previous.begin())->getUnderlyingDecl(); } } diff --git a/test/SemaTemplate/class-template-decl.cpp b/test/SemaTemplate/class-template-decl.cpp index bd2accee3c..23385a701e 100644 --- a/test/SemaTemplate/class-template-decl.cpp +++ b/test/SemaTemplate/class-template-decl.cpp @@ -94,3 +94,46 @@ namespace rdar9676205 { }; }; } + +namespace redecl { + int A; // expected-note {{here}} + template struct A; // expected-error {{different kind of symbol}} + + int B; + template struct B { // expected-error {{different kind of symbol}} + }; + + template struct F; + template struct K; + + int G, H; + + struct S { + int C; // expected-note {{here}} + template struct C; // expected-error {{different kind of symbol}} + + int D; + template struct D { // expected-error {{different kind of symbol}} + }; + + int E; + template friend struct E { // expected-error {{cannot define a type in a friend}} + }; + + int F; + template friend struct F; // ok, redecl::F + + template struct G; // ok + + template friend struct H; // expected-error {{different kind of symbol}} + + int I, J, K; + + struct U { + template struct I; // ok + template struct J { // ok + }; + template friend struct K; // ok, redecl::K + }; + }; +}