]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
c++: maybe_dependent_member_ref and stripped alias [PR118626]
authorPatrick Palka <ppalka@redhat.com>
Sun, 6 Apr 2025 02:39:12 +0000 (22:39 -0400)
committerPatrick Palka <ppalka@redhat.com>
Sun, 6 Apr 2025 02:39:12 +0000 (22:39 -0400)
Here during maybe_dependent_member_ref (as part of CTAD rewriting
for the variant constructor) for __accepted_type<_Up> we strip this
alias all the way to type _Nth_type<__accepted_index<_Up>>, for which
we return NULL since _Nth_type is at namespace scope and so no
longer needs rewriting.

Note that however the template argument __accepted_index<_Up> of this
stripped type _does_ need rewriting (since it specializes a variable
template from the current instantiation).  We end up not rewriting this
variable template reference at any point however because upon returning
NULL, the caller (tsubst) proceeds to substitute the original form of
the type __accepted_type<_Up>, which doesn't directly refer to
__accepted_index.  This later leads to an ICE during subsequent alias
CTAD rewriting of this guide that contains a non-rewritten reference
to __accepted_index.

So when maybe_dependent_member_ref decides to not rewrite a class-scope
alias that's been stripped, the caller needs to commit to substituting
the stripped type rather than the original type.  This patch essentially
implements that by making maybe_dependent_member_ref call tsubst itself
in that case.

PR c++/118626

gcc/cp/ChangeLog:

* pt.cc (maybe_dependent_member_ref): Substitute and return the
stripped type if we decided to not rewrite it directly.

gcc/testsuite/ChangeLog:

* g++.dg/cpp2a/class-deduction-alias25.C: New test.

Reviewed-by: Jason Merrill <jason@redhat.com>
gcc/cp/pt.cc
gcc/testsuite/g++.dg/cpp2a/class-deduction-alias25.C [new file with mode: 0644]

index f7c56a107945aa099b6531f7f79dd95ab794b648..23e41a0fdcb189a1f73f990a1ae272bba95f7746 100644 (file)
@@ -17741,13 +17741,37 @@ maybe_dependent_member_ref (tree t, tree args, tsubst_flags_t complain,
 
   if (TYPE_P (t))
     {
+      bool stripped = false;
       if (typedef_variant_p (t))
-       t = strip_typedefs (t);
-      tree decl = TYPE_NAME (t);
+       {
+         /* Since this transformation may undesirably turn a deduced context
+            into a non-deduced one, we'd rather strip typedefs than perform
+            the transformation.  */
+         tree u = strip_typedefs (t);
+         if (u != t)
+           {
+             stripped = true;
+             t = u;
+           }
+       }
+      decl = TYPE_NAME (t);
       if (decl)
        decl = maybe_dependent_member_ref (decl, args, complain, in_decl);
       if (!decl)
-       return NULL_TREE;
+       {
+         if (stripped)
+           /* The original type was an alias from the current instantiation
+              which we stripped to something outside it.  At this point we
+              need to commit to using the stripped type rather than deferring
+              to the caller (which would use the original type), to ensure
+              eligible bits of the stripped type get transformed.  */
+           return tsubst (t, args, complain, in_decl);
+         else
+           /* The original type wasn't a typedef, and we decided it doesn't
+              need rewriting, so just let the caller (tsubst) substitute it
+              normally.  */
+           return NULL_TREE;
+       }
       return cp_build_qualified_type (TREE_TYPE (decl), cp_type_quals (t),
                                      complain);
     }
diff --git a/gcc/testsuite/g++.dg/cpp2a/class-deduction-alias25.C b/gcc/testsuite/g++.dg/cpp2a/class-deduction-alias25.C
new file mode 100644 (file)
index 0000000..37ab932
--- /dev/null
@@ -0,0 +1,19 @@
+// PR c++/118626
+// { dg-do compile { target c++20 } }
+
+template<long> struct _Nth_type;
+
+template<class _Up>
+struct variant {
+  template<class _Tp> static constexpr long __accepted_index = 0;
+  template<long _Np> using __to_type = _Nth_type<_Np>;
+  template<class _Tp> using __accepted_type = __to_type<__accepted_index<_Tp>>;
+  template<class = __accepted_type<_Up>> variant(_Up);
+};
+
+template<class _Tp>
+struct Node { Node(_Tp); };
+
+template<class R> using Tree = variant<Node<R>>;
+using type = decltype(Tree{Node{42}});
+using type = Tree<int>;