diff --git a/clang/include/clang/AST/NestedNameSpecifier.h b/clang/include/clang/AST/NestedNameSpecifier.h --- a/clang/include/clang/AST/NestedNameSpecifier.h +++ b/clang/include/clang/AST/NestedNameSpecifier.h @@ -117,6 +117,15 @@ public: NestedNameSpecifier &operator=(const NestedNameSpecifier &) = delete; + friend bool operator==(const NestedNameSpecifier &a, + const NestedNameSpecifier &b) { + return a.Specifier == b.Specifier && a.Prefix.getInt() == b.Prefix.getInt(); + } + friend bool operator!=(const NestedNameSpecifier &a, + const NestedNameSpecifier &b) { + return !(a == b); + } + /// Builds a specifier combining a prefix and an identifier. /// /// The prefix must be dependent, since nested name specifiers diff --git a/clang/include/clang/Sema/Sema.h b/clang/include/clang/Sema/Sema.h --- a/clang/include/clang/Sema/Sema.h +++ b/clang/include/clang/Sema/Sema.h @@ -7823,8 +7823,7 @@ TemplateArgumentLoc &Arg, SmallVectorImpl &Converted); - bool CheckTemplateArgument(TemplateTypeParmDecl *Param, - TypeSourceInfo *Arg); + bool CheckTemplateArgument(TypeSourceInfo *Arg); ExprResult CheckTemplateArgument(NonTypeTemplateParmDecl *Param, QualType InstantiatedParamType, Expr *Arg, TemplateArgument &Converted, diff --git a/clang/lib/AST/ASTContext.cpp b/clang/lib/AST/ASTContext.cpp --- a/clang/lib/AST/ASTContext.cpp +++ b/clang/lib/AST/ASTContext.cpp @@ -6069,7 +6069,7 @@ if (!NNS) return nullptr; - switch (NNS->getKind()) { + switch (auto Kind = NNS->getKind()) { case NestedNameSpecifier::Identifier: // Canonicalize the prefix but keep the identifier the same. return NestedNameSpecifier::Create(*this, @@ -6091,8 +6091,6 @@ case NestedNameSpecifier::TypeSpec: case NestedNameSpecifier::TypeSpecWithTemplate: { - QualType T = getCanonicalType(QualType(NNS->getAsType(), 0)); - // If we have some kind of dependent-named type (e.g., "typename T::type"), // break it apart into its prefix and identifier, then reconsititute those // as the canonical nested-name-specifier. This is required to canonicalize @@ -6100,15 +6098,12 @@ // types, e.g., // typedef typename T::type T1; // typedef typename T1::type T2; - if (const auto *DNT = T->getAs()) - return NestedNameSpecifier::Create(*this, DNT->getQualifier(), - const_cast(DNT->getIdentifier())); - - // Otherwise, just canonicalize the type, and force it to be a TypeSpec. - // FIXME: Why are TypeSpec and TypeSpecWithTemplate distinct in the - // first place? - return NestedNameSpecifier::Create(*this, nullptr, false, - const_cast(T.getTypePtr())); + // The difference between TypeSpec and TypeSpecWithTemplate is that the + // latter will have the 'template' keyword when printed. + return NestedNameSpecifier::Create( + *this, getCanonicalNestedNameSpecifier(NNS->getPrefix()), + Kind == NestedNameSpecifier::TypeSpecWithTemplate, + const_cast(getCanonicalType(NNS->getAsType()))); } case NestedNameSpecifier::Global: diff --git a/clang/lib/Sema/SemaDeclCXX.cpp b/clang/lib/Sema/SemaDeclCXX.cpp --- a/clang/lib/Sema/SemaDeclCXX.cpp +++ b/clang/lib/Sema/SemaDeclCXX.cpp @@ -12472,6 +12472,8 @@ return false; } + const NestedNameSpecifier *NSS = + Context.getCanonicalNestedNameSpecifier(Qual); for (LookupResult::iterator I = Prev.begin(), E = Prev.end(); I != E; ++I) { NamedDecl *D = *I; @@ -12497,8 +12499,7 @@ // using decls differ if they name different scopes (but note that // template instantiation can cause this check to trigger when it // didn't before instantiation). - if (Context.getCanonicalNestedNameSpecifier(Qual) != - Context.getCanonicalNestedNameSpecifier(DQual)) + if (*NSS != *Context.getCanonicalNestedNameSpecifier(DQual)) continue; Diag(NameLoc, diag::err_using_decl_redeclaration) << SS.getRange(); diff --git a/clang/lib/Sema/SemaTemplate.cpp b/clang/lib/Sema/SemaTemplate.cpp --- a/clang/lib/Sema/SemaTemplate.cpp +++ b/clang/lib/Sema/SemaTemplate.cpp @@ -1079,7 +1079,7 @@ return Param; // Check the template argument itself. - if (CheckTemplateArgument(Param, DefaultTInfo)) { + if (CheckTemplateArgument(DefaultTInfo)) { Param->setInvalidDecl(); return Param; } @@ -5042,7 +5042,7 @@ } } - if (CheckTemplateArgument(Param, TSI)) + if (CheckTemplateArgument(TSI)) return true; // Add the converted template type argument. @@ -5661,7 +5661,7 @@ TemplateArgumentListInfo NewArgs = TemplateArgs; // Make sure we get the template parameter list from the most - // recentdeclaration, since that is the only one that has is guaranteed to + // recent declaration, since that is the only one that is guaranteed to // have all the default template argument information. TemplateParameterList *Params = cast(Template->getMostRecentDecl()) @@ -6208,8 +6208,7 @@ /// /// This routine implements the semantics of C++ [temp.arg.type]. It /// returns true if an error occurred, and false otherwise. -bool Sema::CheckTemplateArgument(TemplateTypeParmDecl *Param, - TypeSourceInfo *ArgInfo) { +bool Sema::CheckTemplateArgument(TypeSourceInfo *ArgInfo) { assert(ArgInfo && "invalid TypeSourceInfo"); QualType Arg = ArgInfo->getType(); SourceRange SR = ArgInfo->getTypeLoc().getSourceRange(); diff --git a/clang/test/CXX/expr/expr.prim/expr.prim.req/equivalence.cpp b/clang/test/CXX/expr/expr.prim/expr.prim.req/equivalence.cpp --- a/clang/test/CXX/expr/expr.prim/expr.prim.req/equivalence.cpp +++ b/clang/test/CXX/expr/expr.prim/expr.prim.req/equivalence.cpp @@ -18,9 +18,9 @@ template requires requires { typename identity::type; } struct r2; -template requires requires { typename identity::type; } +template requires requires { typename identity::type; } // expected-note 3{{previous template declaration is here}} struct r2; -template requires requires { typename ::identity::type; } // expected-note 2{{previous template declaration is here}} +template requires requires { typename ::identity::type; } // expected-error{{requires clause differs in template redeclaration}} struct r2; template requires requires { typename identity::typr; } // expected-error{{requires clause differs in template redeclaration}} struct r2; diff --git a/clang/test/CXX/temp/temp.constr/temp.constr.normal/p1.cpp b/clang/test/CXX/temp/temp.constr/temp.constr.normal/p1.cpp --- a/clang/test/CXX/temp/temp.constr/temp.constr.normal/p1.cpp +++ b/clang/test/CXX/temp/temp.constr/temp.constr.normal/p1.cpp @@ -81,4 +81,23 @@ requires true struct S4; // expected-note {{template is declared here}} template requires true struct S4; // expected-error {{class template partial specialization is not more specialized than the primary template}} + +struct X { + template struct Y { + using type = int; + }; +}; + +template concept C1 = sizeof(T) != 0; +template concept C2 = C1::type>; + +template requires C1 void t1() = delete; // expected-note {{candidate function}} +template requires C1 && C2 void t1() = delete; // expected-note {{candidate function}} +template void t1(); +void t1() { t1(); } // expected-error {{call to deleted function 't1'}} + +template requires C1 void t2() {}; // expected-note 2 {{candidate function}} +template requires C2 void t2() {}; // expected-note 2 {{candidate function}} +template void t2(); // expected-error {{partial ordering for explicit instantiation of 't2' is ambiguous}} +void t2() { t2(); } // expected-error {{call to 't2' is ambiguous}} } // namespace PR47174