https://gcc.gnu.org/g:96e0370f4daef29b918aafcff68c7f5e4ef397fd

commit r15-3933-g96e0370f4daef29b918aafcff68c7f5e4ef397fd
Author: Marek Polacek <pola...@redhat.com>
Date:   Thu Sep 5 16:45:32 2024 -0400

    c++: ICE with structured bindings and m-d array [PR102594]
    
    We ICE in decay_conversion with this test:
    
      struct S {
        S() {}
      };
      S arr[1][1];
      auto [m](arr3);
    
    But not when the last line is:
    
      auto [n] = arr3;
    
    Therefore the difference is between copy- and direct-init.  In
    particular, in build_vec_init we have:
    
      if (direct_init)
        from = build_tree_list (NULL_TREE, from);
    
    and then we call build_vec_init again with init==from.  Then
    decay_conversion gets the TREE_LIST and it crashes.
    
    build_aggr_init has:
    
                  /* Wrap the initializer in a CONSTRUCTOR so that 
build_vec_init
                     recognizes it as direct-initialization.  */
                  init = build_constructor_single (init_list_type_node,
                                                   NULL_TREE, init);
                  CONSTRUCTOR_IS_DIRECT_INIT (init) = true;
    
    so I propose to do the same in build_vec_init.
    
            PR c++/102594
    
    gcc/cp/ChangeLog:
    
            * init.cc (build_vec_init): Build up a CONSTRUCTOR to signal
            direct-initialization rather than a TREE_LIST.
    
    gcc/testsuite/ChangeLog:
    
            * g++.dg/cpp1z/decomp61.C: New test.
    
    Reviewed-by: Jason Merrill <ja...@redhat.com>

Diff:
---
 gcc/cp/init.cc                        |  8 +++++++-
 gcc/testsuite/g++.dg/cpp1z/decomp61.C | 28 ++++++++++++++++++++++++++++
 2 files changed, 35 insertions(+), 1 deletion(-)

diff --git a/gcc/cp/init.cc b/gcc/cp/init.cc
index be7fdb40dd6c..f785015e4774 100644
--- a/gcc/cp/init.cc
+++ b/gcc/cp/init.cc
@@ -4958,7 +4958,13 @@ build_vec_init (tree base, tree maxindex, tree init,
              if (xvalue)
                from = move (from);
              if (direct_init)
-               from = build_tree_list (NULL_TREE, from);
+               {
+                 /* Wrap the initializer in a CONSTRUCTOR so that
+                    build_vec_init recognizes it as direct-initialization.  */
+                 from = build_constructor_single (init_list_type_node,
+                                                  NULL_TREE, from);
+                 CONSTRUCTOR_IS_DIRECT_INIT (from) = true;
+               }
            }
          else
            from = NULL_TREE;
diff --git a/gcc/testsuite/g++.dg/cpp1z/decomp61.C 
b/gcc/testsuite/g++.dg/cpp1z/decomp61.C
new file mode 100644
index 000000000000..ad0a20c1addd
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp1z/decomp61.C
@@ -0,0 +1,28 @@
+// PR c++/102594
+// { dg-do compile { target c++17 } }
+
+struct S {
+  S() {}
+};
+S arr1[2];
+S arr2[2][1];
+S arr3[1][1];
+auto [m](arr3);
+auto [n] = arr3;
+
+struct X {
+  int i;
+};
+
+void
+g (X x)
+{
+  auto [a, b](arr2);
+  auto [c, d] = arr2;
+  auto [e, f] = (arr2);
+  auto [i, j](arr1);
+  auto [k, l] = arr1;
+  auto [m, n] = (arr1);
+  auto [z] = x;
+  auto [y](x);
+}

Reply via email to