llvmbot wrote:

<!--LLVM PR SUMMARY COMMENT-->

@llvm/pr-subscribers-clang

Author: None (offsetof)

<details>
<summary>Changes</summary>

* Handle pack expansion types in `ASTContext::getAdjustedParameterType`, 
adjusting their pattern if necessary.
* Perform the usual constant template parameter type adjustments in 
`Sema::RebuildTemplateParamsInCurrentInstantiation`.
* Augment `Sema::CheckNonTypeTemplateParameterType` with a `Diagnose` parameter 
to perform the aforementioned adjustments without emitting redundant 
diagnostics.

---
Full diff: https://github.com/llvm/llvm-project/pull/132189.diff


10 Files Affected:

- (modified) clang/docs/ReleaseNotes.rst (+3) 
- (modified) clang/include/clang/AST/ASTContext.h (+3) 
- (modified) clang/include/clang/Sema/Sema.h (+9-6) 
- (modified) clang/lib/AST/ASTContext.cpp (+4) 
- (modified) clang/lib/Sema/SemaTemplate.cpp (+36-24) 
- (modified) clang/test/AST/ast-dump-templates.cpp (+2-2) 
- (modified) clang/test/CXX/temp/temp.decls/temp.mem/p1.cpp (+67) 
- (modified) clang/test/CXX/temp/temp.param/p7.cpp (+1-1) 
- (added) clang/test/SemaTemplate/dependent-type-decay.cpp (+123) 
- (modified) clang/test/SemaTemplate/pack-deduction.cpp (+2-3) 


``````````diff
diff --git a/clang/docs/ReleaseNotes.rst b/clang/docs/ReleaseNotes.rst
index 4258d0d72c950..28b05a6dd62e5 100644
--- a/clang/docs/ReleaseNotes.rst
+++ b/clang/docs/ReleaseNotes.rst
@@ -352,6 +352,9 @@ Bug Fixes to C++ Support
   The issue has been addressed by propagating qualifiers during 
derived-to-base conversions in the AST. (#GH127824)
 - Clang now emits the ``-Wunused-variable`` warning when some structured 
bindings are unused
   and the ``[[maybe_unused]]`` attribute is not applied. (#GH125810)
+- Function parameter packs and constant (non-type) template parameter packs
+  declared with an array or function type are now correctly adjusted to be of
+  the corresponding decayed pointer type.
 
 Bug Fixes to AST Handling
 ^^^^^^^^^^^^^^^^^^^^^^^^^
diff --git a/clang/include/clang/AST/ASTContext.h 
b/clang/include/clang/AST/ASTContext.h
index f9a12260a6590..35835408e8575 100644
--- a/clang/include/clang/AST/ASTContext.h
+++ b/clang/include/clang/AST/ASTContext.h
@@ -2977,6 +2977,9 @@ class ASTContext : public RefCountedBase<ASTContext> {
   /// This routine adjusts the given parameter type @p T to the actual
   /// parameter type used by semantic analysis (C99 6.7.5.3p[7,8],
   /// C++ [dcl.fct]p3). The adjusted parameter type is returned.
+  ///
+  /// If @p T is a pack expansion type, the adjustment is performed
+  /// on its pattern.
   QualType getAdjustedParameterType(QualType T) const;
 
   /// Retrieve the parameter type as adjusted for use in the signature
diff --git a/clang/include/clang/Sema/Sema.h b/clang/include/clang/Sema/Sema.h
index 9724f0def743a..da7dd670120c9 100644
--- a/clang/include/clang/Sema/Sema.h
+++ b/clang/include/clang/Sema/Sema.h
@@ -11352,19 +11352,22 @@ class Sema final : public SemaBase {
                             NonTypeTemplateParmDecl *OrigConstrainedParm,
                             SourceLocation EllipsisLoc);
 
-  /// Require the given type to be a structural type, and diagnose if it is 
not.
+  /// Require the given type to be a structural type,
+  /// and optionally diagnose if it is not.
   ///
-  /// \return \c true if an error was produced.
-  bool RequireStructuralType(QualType T, SourceLocation Loc);
+  /// \return \c true if the type is *not* a structural type.
+  bool RequireStructuralType(QualType T, SourceLocation Loc, bool Diagnose);
 
   /// Check that the type of a non-type template parameter is
   /// well-formed.
   ///
   /// \returns the (possibly-promoted) parameter type if valid;
-  /// otherwise, produces a diagnostic and returns a NULL type.
+  /// otherwise, returns a NULL type and optionally produces a diagnostic.
   QualType CheckNonTypeTemplateParameterType(TypeSourceInfo *&TSI,
-                                             SourceLocation Loc);
-  QualType CheckNonTypeTemplateParameterType(QualType T, SourceLocation Loc);
+                                             SourceLocation Loc,
+                                             bool Diagnose = true);
+  QualType CheckNonTypeTemplateParameterType(QualType T, SourceLocation Loc,
+                                             bool Diagnose = true);
 
   NamedDecl *ActOnNonTypeTemplateParameter(Scope *S, Declarator &D,
                                            unsigned Depth, unsigned Position,
diff --git a/clang/lib/AST/ASTContext.cpp b/clang/lib/AST/ASTContext.cpp
index 68a02f3bbe1ec..2403eadc2043c 100644
--- a/clang/lib/AST/ASTContext.cpp
+++ b/clang/lib/AST/ASTContext.cpp
@@ -7672,6 +7672,10 @@ const ArrayType *ASTContext::getAsArrayType(QualType T) 
const {
 }
 
 QualType ASTContext::getAdjustedParameterType(QualType T) const {
+  if (auto *PET = T->getAs<PackExpansionType>())
+    return getPackExpansionType(getAdjustedParameterType(PET->getPattern()),
+                                PET->getNumExpansions(),
+                                /*ExpectPackInType=*/false);
   if (getLangOpts().HLSL && T->isConstantArrayType())
     return getArrayParameterType(T);
   if (T->isArrayType() || T->isFunctionType())
diff --git a/clang/lib/Sema/SemaTemplate.cpp b/clang/lib/Sema/SemaTemplate.cpp
index c3c993d51b79d..91f675006508b 100644
--- a/clang/lib/Sema/SemaTemplate.cpp
+++ b/clang/lib/Sema/SemaTemplate.cpp
@@ -1267,7 +1267,8 @@ bool Sema::AttachTypeConstraint(AutoTypeLoc TL,
 }
 
 QualType Sema::CheckNonTypeTemplateParameterType(TypeSourceInfo *&TSI,
-                                                 SourceLocation Loc) {
+                                                 SourceLocation Loc,
+                                                 bool Diagnose) {
   if (TSI->getType()->isUndeducedType()) {
     // C++17 [temp.dep.expr]p3:
     //   An id-expression is type-dependent if it contains
@@ -1277,19 +1278,27 @@ QualType 
Sema::CheckNonTypeTemplateParameterType(TypeSourceInfo *&TSI,
     TSI = SubstAutoTypeSourceInfoDependent(TSI);
   }
 
-  return CheckNonTypeTemplateParameterType(TSI->getType(), Loc);
+  return CheckNonTypeTemplateParameterType(TSI->getType(), Loc, Diagnose);
 }
 
-bool Sema::RequireStructuralType(QualType T, SourceLocation Loc) {
+bool Sema::RequireStructuralType(QualType T, SourceLocation Loc,
+                                 bool Diagnose) {
   if (T->isDependentType())
     return false;
 
-  if (RequireCompleteType(Loc, T, diag::err_template_nontype_parm_incomplete))
+  if (Diagnose ? RequireCompleteType(Loc, T,
+                                     
diag::err_template_nontype_parm_incomplete)
+               : !isCompleteType(Loc, T))
     return true;
 
   if (T->isStructuralType())
     return false;
 
+  // If we're not emitting diagnostics, there's no need to figure out
+  // why exactly T is not a structural type.
+  if (!Diagnose)
+    return true;
+
   // Structural types are required to be object types or lvalue references.
   if (T->isRValueReferenceType()) {
     Diag(Loc, diag::err_template_nontype_parm_rvalue_ref) << T;
@@ -1379,16 +1388,21 @@ bool Sema::RequireStructuralType(QualType T, 
SourceLocation Loc) {
   return true;
 }
 
-QualType Sema::CheckNonTypeTemplateParameterType(QualType T,
-                                                 SourceLocation Loc) {
+QualType Sema::CheckNonTypeTemplateParameterType(QualType T, SourceLocation 
Loc,
+                                                 bool Diagnose) {
   // We don't allow variably-modified types as the type of non-type template
   // parameters.
   if (T->isVariablyModifiedType()) {
-    Diag(Loc, diag::err_variably_modified_nontype_template_param)
-      << T;
+    if (Diagnose)
+      Diag(Loc, diag::err_variably_modified_nontype_template_param) << T;
     return QualType();
   }
 
+  // C++2c [temp.param] p10:
+  //   A constant template parameter of type "array of T" or of
+  //   function type T is adjusted to be of type "pointer to T".
+  T = Context.getAdjustedParameterType(T);
+
   // C++ [temp.param]p4:
   //
   // A non-type template-parameter shall have one of the following
@@ -1411,14 +1425,6 @@ QualType 
Sema::CheckNonTypeTemplateParameterType(QualType T,
     return T.getUnqualifiedType();
   }
 
-  // C++ [temp.param]p8:
-  //
-  //   A non-type template-parameter of type "array of T" or
-  //   "function returning T" is adjusted to be of type "pointer to
-  //   T" or "pointer to function returning T", respectively.
-  if (T->isArrayType() || T->isFunctionType())
-    return Context.getDecayedType(T);
-
   // If T is a dependent type, we can't do the check now, so we
   // assume that it is well-formed. Note that stripping off the
   // qualifiers here is not really correct if T turns out to be
@@ -1430,18 +1436,20 @@ QualType 
Sema::CheckNonTypeTemplateParameterType(QualType T,
 
   // C++20 [temp.param]p6:
   //   -- a structural type
-  if (RequireStructuralType(T, Loc))
+  if (RequireStructuralType(T, Loc, Diagnose))
     return QualType();
 
   if (!getLangOpts().CPlusPlus20) {
     // FIXME: Consider allowing structural types as an extension in C++17. (In
     // earlier language modes, the template argument evaluation rules are too
     // inflexible.)
-    Diag(Loc, diag::err_template_nontype_parm_bad_structural_type) << T;
+    if (Diagnose)
+      Diag(Loc, diag::err_template_nontype_parm_bad_structural_type) << T;
     return QualType();
   }
 
-  Diag(Loc, diag::warn_cxx17_compat_template_nontype_parm_type) << T;
+  if (Diagnose)
+    Diag(Loc, diag::warn_cxx17_compat_template_nontype_parm_type) << T;
   return T.getUnqualifiedType();
 }
 
@@ -1518,7 +1526,7 @@ NamedDecl *Sema::ActOnNonTypeTemplateParameter(Scope *S, 
Declarator &D,
 
   QualType T = CheckNonTypeTemplateParameterType(TInfo, D.getIdentifierLoc());
   if (T.isNull()) {
-    T = Context.IntTy; // Recover with an 'int' type.
+    T = TInfo->getType();
     Invalid = true;
   }
 
@@ -11080,9 +11088,7 @@ bool 
Sema::RebuildNestedNameSpecifierInCurrentInstantiation(CXXScopeSpec &SS) {
 
 bool Sema::RebuildTemplateParamsInCurrentInstantiation(
                                                TemplateParameterList *Params) {
-  for (unsigned I = 0, N = Params->size(); I != N; ++I) {
-    Decl *Param = Params->getParam(I);
-
+  for (Decl *Param : *Params) {
     // There is nothing to rebuild in a type parameter.
     if (isa<TemplateTypeParmDecl>(Param))
       continue;
@@ -11116,8 +11122,14 @@ bool Sema::RebuildTemplateParamsInCurrentInstantiation(
     }
 
     if (NewTSI != NTTP->getTypeSourceInfo()) {
+      QualType NewT = CheckNonTypeTemplateParameterType(
+          NewTSI->getType(), NTTP->getLocation(), /*Diagnose=*/false);
+      if (NewT.isNull()) {
+        NewT = NewTSI->getType();
+        NTTP->setInvalidDecl();
+      }
       NTTP->setTypeSourceInfo(NewTSI);
-      NTTP->setType(NewTSI->getType());
+      NTTP->setType(NewT);
     }
   }
 
diff --git a/clang/test/AST/ast-dump-templates.cpp 
b/clang/test/AST/ast-dump-templates.cpp
index 2728dc151c3c5..807f88b1ae388 100644
--- a/clang/test/AST/ast-dump-templates.cpp
+++ b/clang/test/AST/ast-dump-templates.cpp
@@ -48,7 +48,7 @@ void baz() {
 // CHECK2: template<> int bar<5, int>()
 
 // CHECK1-LABEL: template <typename ...T> struct A {
-// CHECK1-NEXT:    template <T ...x[3]> struct B {
+// CHECK1-NEXT:      template <T *...x> struct B {
 template <typename ...T> struct A {
   template <T ...x[3]> struct B {};
 };
@@ -3174,7 +3174,7 @@ struct pr126341<{1, 2}>;
 // JSON-NEXT:          },
 // JSON-NEXT:          "name": "x",
 // JSON-NEXT:          "type": {
-// JSON-NEXT:           "qualType": "T[3]..."
+// JSON-NEXT:           "qualType": "T *..."
 // JSON-NEXT:          },
 // JSON-NEXT:          "depth": 1,
 // JSON-NEXT:          "index": 0,
diff --git a/clang/test/CXX/temp/temp.decls/temp.mem/p1.cpp 
b/clang/test/CXX/temp/temp.decls/temp.mem/p1.cpp
index 4ec41521f9a3b..5adfb6be573b2 100644
--- a/clang/test/CXX/temp/temp.decls/temp.mem/p1.cpp
+++ b/clang/test/CXX/temp/temp.decls/temp.mem/p1.cpp
@@ -141,4 +141,71 @@ namespace OutOfLine {
   template<>
   template<typename U, A<int>::B V>
   struct A<int>::C<U&, V> { }; // expected-error {{redefinition of 'C<U &, 
V>'}}
+
+
+  template<class T>
+  class X { // expected-note 2 {{X defined here}}
+    using A = int;
+    template<A> void a1();
+    template<A> static int a2;
+    template<A> class a3;
+    template<OutOfLine::X<T>::A> void a4();
+    template<OutOfLine::X<T>::A> static int a5;
+    template<OutOfLine::X<T>::A> class a6;
+
+    using B = int[];
+    template<B> void b1();
+    template<B> static int b2;
+    template<B> class b3;
+    template<OutOfLine::X<T>::B> void b4();
+    template<OutOfLine::X<T>::B> static int b5;
+    template<OutOfLine::X<T>::B> class b6;
+
+    using Bad = int&&;
+    template<Bad> void bad1();                      // expected-error 
{{non-type template parameter has rvalue reference type}}
+    template<Bad> static int bad2;                  // expected-error 
{{non-type template parameter has rvalue reference type}}
+    template<Bad> class bad3;                       // expected-error 
{{non-type template parameter has rvalue reference type}}
+    template<OutOfLine::X<T>::Bad> void bad4();     // expected-error 
{{non-type template parameter has rvalue reference type}}
+    template<OutOfLine::X<T>::Bad> static int bad5; // expected-error 
{{non-type template parameter has rvalue reference type}}
+    template<OutOfLine::X<T>::Bad> class bad6;      // expected-error 
{{non-type template parameter has rvalue reference type}}
+
+    template<const T> class Q;
+    template<const T...> class Qp;
+
+    template<int> void good();
+  };
+
+  template<class T> template<X<T>::A> void X<T>::a1() {}
+  template<class T> template<X<T>::A> int X<T>::a2 = 2;
+  template<class T> template<X<T>::A> class X<T>::a3 {};
+  template<class T> template<X<T>::A> void X<T>::a4() {}
+  template<class T> template<X<T>::A> int X<T>::a5 = 5;
+  template<class T> template<X<T>::A> class X<T>::a6 {};
+
+  template<class T> template<X<T>::B> void X<T>::b1() {}
+  template<class T> template<X<T>::B> int X<T>::b2 = 2;
+  template<class T> template<X<T>::B> class X<T>::b3 {};
+  template<class T> template<X<T>::B> void X<T>::b4() {}
+  template<class T> template<X<T>::B> int X<T>::b5 = 5;
+  template<class T> template<X<T>::B> class X<T>::b6 {};
+
+  template<class T> template<X<T>::Bad> void X<T>::bad1() {}
+  template<class T> template<X<T>::Bad> int X<T>::bad2 = 2;
+  template<class T> template<X<T>::Bad> class X<T>::bad3 {};
+  template<class T> template<X<T>::Bad> void X<T>::bad4() {}
+  template<class T> template<X<T>::Bad> int X<T>::bad5 = 5;
+  template<class T> template<X<T>::Bad> class X<T>::bad6 {};
+
+  template<class T> template<const T> class X<T>::Q {};
+  template<class T> template<const T...> class X<T>::Qp {};
+
+  template<class T>
+  template<X<T>::Bad>
+  void X<T>::good() {} // expected-error {{out-of-line definition of 'good' 
does not match any declaration in 'X<T>'}}
+
+  template<class> using RRef = int&&;
+
+  template<class T>
+  template<RRef<T>> // expected-error {{non-type template parameter has rvalue 
reference type 'RRef<T>' (aka 'int &&')}}
+  void X<T>::good() {} // expected-error {{out-of-line definition of 'good' 
does not match any declaration in 'X<T>'}}
 }
diff --git a/clang/test/CXX/temp/temp.param/p7.cpp 
b/clang/test/CXX/temp/temp.param/p7.cpp
index f804306d7c577..dbd310fe6a1f4 100644
--- a/clang/test/CXX/temp/temp.param/p7.cpp
+++ b/clang/test/CXX/temp/temp.param/p7.cpp
@@ -123,4 +123,4 @@ template<MutableField> struct WithMutableField {}; // 
cxx17-error {{cannot have
 
 template<typename T> struct BadExtType { T t; }; // cxx20-note 2{{has a 
non-static data member of non-structural type}}
 template<BadExtType<_Atomic float> > struct AtomicFloatField; // cxx17-error 
{{cannot have type}} cxx20-error {{is not a structural type}}
-template<BadExtType<_Atomic int> > struct AtomicInt; // cxx17-error {{cannot 
have type}} cxx20-error {{is not a structural type}}
+template<BadExtType<_Atomic int> > struct AtomicIntField; // cxx17-error 
{{cannot have type}} cxx20-error {{is not a structural type}}
diff --git a/clang/test/SemaTemplate/dependent-type-decay.cpp 
b/clang/test/SemaTemplate/dependent-type-decay.cpp
new file mode 100644
index 0000000000000..dd1b04d982b2c
--- /dev/null
+++ b/clang/test/SemaTemplate/dependent-type-decay.cpp
@@ -0,0 +1,123 @@
+// RUN: %clang_cc1 %s -fsyntax-only -std=c++20 -verify
+
+void f(auto*) {} // expected-note {{previous definition is here}}
+void f(auto[]) {} // expected-error {{redefinition of 'f'}}
+
+void g(auto()) {} // expected-note {{previous definition is here}}
+void g(auto (*)()) {} // expected-error {{redefinition of 'g'}}
+
+void fp(auto*...) {} // expected-note {{previous definition is here}}
+void fp(auto... _[]) {} // expected-error {{redefinition of 'fp'}}
+
+void gp(auto...()) {} // expected-note {{previous definition is here}}
+void gp(auto (*..._)()) {} // expected-error {{redefinition of 'gp'}}
+
+
+template<int*> class A;
+template<int _[]> class A;
+
+template<int()> class B;
+template<int (*)()> class B;
+
+template<int*...> class Ap;
+template<int... _[]> class Ap;
+
+template<int...()> class Bp;
+template<int (*..._)()> class Bp;
+
+template<class T, class... Ts>
+class C {
+    template<T*> void f(); // expected-note {{previous declaration is here}}
+    template<T[]> void f(); // expected-error {{class member cannot be 
redeclared}}
+
+    template<T()> void g(); // expected-note {{previous declaration is here}}
+    template<T(*)()> void g(); // expected-error {{class member cannot be 
redeclared}}
+
+    template<Ts*...> void fp(); // expected-note {{previous declaration is 
here}}
+    template<Ts... _[]> void fp(); // expected-error {{class member cannot be 
redeclared}}
+
+    template<Ts...()> void gp(); // expected-note {{previous declaration is 
here}}
+    template<Ts(* ..._)()> void gp(); // expected-error {{class member cannot 
be redeclared}}
+
+    template<T[]> class X;
+    template<T()> class Y;
+
+    template<Ts... _[]> class Xp;
+    template<Ts...()> class Yp;
+};
+
+template<class T, class... Ts>
+template<T*>
+class C<T, Ts...>::X {};
+
+template<class T, class... Ts>
+template<T (*)()>
+class C<T, Ts...>::Y {};
+
+template<class T, class... Ts>
+template<Ts*...>
+class C<T, Ts...>::Xp {};
+
+template<class T, class... Ts>
+template<Ts (*..._)()>
+class C<T, Ts...>::Yp {};
+
+
+template<class...> class R;
+
+template<class T>
+R<T>* d0(T[]) { return 0; }
+R<int>* r0 = d0((int*)0);
+
+template<class T>
+R<T>* d1(T(T[])) { return 0; }
+R<int>* r1 = d1((int (*)(int*))0);
+
+template<class T>
+R<T>* d2(T(T(T))) { return 0; }
+R<int>* r2 = d2((int (*)(int (*)(int)))0);
+
+template<class... Ts>
+R<Ts...>* d0p(Ts... _[]) { return 0; }
+R<int, char>* r0p = d0p((int*)0, (char*)0);
+
+template<class... Ts>
+R<Ts...>* d1p(Ts...(Ts... _[])) { return 0; }
+R<int, char>* r1p = d1p((int (*)(int*, char*))0, (char (*)(int*, char*))0);
+
+template<class... Ts>
+R<Ts...>* d2p(Ts...(Ts...(Ts...))) { return 0; }
+R<int, char>* r2p = d2p(
+    (int (*)(int(int, char), char(int, char)))0,
+    (char (*)(int(int, char), char(int, char)))0);
+
+
+template<class T> concept Y = sizeof(T*) != 0;
+
+template<class T, class... Ts>
+struct S {
+    static int f() requires Y<int(T*)>;
+    static constexpr int f() requires Y<int(T[1])> && true {
+        return 1;
+    }
+
+    static int g() requires Y<int(T (*)())>;
+    static constexpr int g() requires Y<int(T())> && true {
+        return 2;
+    }
+
+    static int fp() requires Y<int(Ts*...)>;
+    static constexpr int fp() requires Y<int(Ts... _[1])> && true {
+        return 3;
+    }
+
+    static int gp() requires Y<int(Ts (*..._)(Ts...))>;
+    static constexpr int gp() requires Y<int(Ts...(Ts...))> && true {
+        return 4;
+    }
+};
+
+static_assert(S<char, short, int>::f() == 1);
+static_assert(S<char, short, int>::g() == 2);
+static_assert(S<char, short, int>::fp() == 3);
+static_assert(S<char, short, int>::gp() == 4);
diff --git a/clang/test/SemaTemplate/pack-deduction.cpp 
b/clang/test/SemaTemplate/pack-deduction.cpp
index b3104609994a4..af188def8991b 100644
--- a/clang/test/SemaTemplate/pack-deduction.cpp
+++ b/clang/test/SemaTemplate/pack-deduction.cpp
@@ -158,12 +158,11 @@ namespace partial_full_mix {
 namespace substitution_vs_function_deduction {
   template <typename... T> struct A {
     template <typename... U> void f(void(*...)(T, U)); // expected-warning 
{{ISO C++11 requires a parenthesized pack declaration to have a name}}
-    template <typename... U> void g(void...(T, U)); // expected-note {{could 
not match 'void (T, U)' against 'void (*)(int, int)'}}
+    template <typename... U> void g(void...(T, U));
   };
   void f(int, int) {
     A<int>().f(f);
-    // FIXME: We fail to decay the parameter to a pointer type.
-    A<int>().g(f); // expected-error {{no match}}
+    A<int>().g(f);
   }
 }
 

``````````

</details>


https://github.com/llvm/llvm-project/pull/132189
_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to