]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
c++: Refine check for CTAD placeholder [PR99586]
authorPatrick Palka <ppalka@redhat.com>
Fri, 2 Apr 2021 23:47:09 +0000 (19:47 -0400)
committerPatrick Palka <ppalka@redhat.com>
Fri, 2 Apr 2021 23:47:09 +0000 (19:47 -0400)
In the below testcase, during finish_compound_literal for A<B{V}>{},
type_uses_auto finds and returns the CTAD placeholder for B{V}, which
tricks us into attempting CTAD on A<B{V}>{} and leads to bogus errors.

AFAICT 'type' will always be a bare 'auto' in the CTAD case so we don't
need to look deeply to find it; checking template_placeholder_p instead
should suffice here.

gcc/cp/ChangeLog:

PR c++/99586
* semantics.c (finish_compound_literal): Check
template_placeholder_p instead of type_uses_auto.

gcc/testsuite/ChangeLog:

PR c++/99586
* g++.dg/cpp2a/nontype-class42.C: New test.

gcc/cp/semantics.c
gcc/testsuite/g++.dg/cpp2a/nontype-class42.C [new file with mode: 0644]

index b02596f73bd1bbc22d9f5088b30006a5c2895e30..8eaaaefe2d6116d156fcdf224ad31701cedfe5db 100644 (file)
@@ -3036,14 +3036,13 @@ finish_compound_literal (tree type, tree compound_literal,
       return error_mark_node;
     }
 
-  if (tree anode = type_uses_auto (type))
-    if (CLASS_PLACEHOLDER_TEMPLATE (anode))
-      {
-       type = do_auto_deduction (type, compound_literal, anode, complain,
-                                 adc_variable_type);
-       if (type == error_mark_node)
-         return error_mark_node;
-      }
+  if (template_placeholder_p (type))
+    {
+      type = do_auto_deduction (type, compound_literal, type, complain,
+                               adc_variable_type);
+      if (type == error_mark_node)
+       return error_mark_node;
+    }
 
   /* Used to hold a copy of the compound literal in a template.  */
   tree orig_cl = NULL_TREE;
diff --git a/gcc/testsuite/g++.dg/cpp2a/nontype-class42.C b/gcc/testsuite/g++.dg/cpp2a/nontype-class42.C
new file mode 100644 (file)
index 0000000..c2e04b2
--- /dev/null
@@ -0,0 +1,8 @@
+// PR c++/99586
+// { dg-do compile { target c++20 } }
+
+template <class T>
+struct B { constexpr B(T); };
+
+template <auto> struct A{};
+template <auto V> auto a = A<B{V}>{};