re PR c++/83958 (ICE: Segmentation fault (in find_decomp_class_base))
authorJakub Jelinek <jakub@redhat.com>
Tue, 23 Jan 2018 13:52:23 +0000 (14:52 +0100)
committerJakub Jelinek <jakub@gcc.gnu.org>
Tue, 23 Jan 2018 13:52:23 +0000 (14:52 +0100)
PR c++/83958
* decl.c (cp_finish_decomp): Diagnose if reference structure binding
refers to incomplete type.

* g++.dg/cpp1z/decomp35.C: New test.

From-SVN: r256984

gcc/cp/ChangeLog
gcc/cp/decl.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp1z/decomp35.C [new file with mode: 0644]

index bd15f4c045a97dd8538da777ccfa8f79022c0f7a..5adf93075d5221ac7a0a9dbfe16da5ff8d86aa51 100644 (file)
@@ -1,3 +1,9 @@
+2018-01-23  Jakub Jelinek  <jakub@redhat.com>
+
+       PR c++/83958
+       * decl.c (cp_finish_decomp): Diagnose if reference structure binding
+       refers to incomplete type.
+
 2018-01-23  Nathan Sidwell  <nathan@acm.org>
 
        Deprecate ARM-era for scope handling
index f6fab422d17a6eb6d944b5d86f14970437cf956a..5f197ef68d8c4a8383ef433f76d06b0f2017fbc2 100644 (file)
@@ -7435,6 +7435,12 @@ cp_finish_decomp (tree decl, tree first, unsigned int count)
       type = complete_type (TREE_TYPE (type));
       if (type == error_mark_node)
        goto error_out;
+      if (!COMPLETE_TYPE_P (type))
+       {
+         error_at (loc, "structured binding refers to incomplete type %qT",
+                   type);
+         goto error_out;
+       }
     }
 
   tree eltype = NULL_TREE;
index bcbaa4c60034958cf55cef21477e765831ec6d7f..8a997aa8614c14ce836892735f45eaf6fab5043d 100644 (file)
@@ -1,3 +1,8 @@
+2018-01-23  Jakub Jelinek  <jakub@redhat.com>
+
+       PR c++/83958
+       * g++.dg/cpp1z/decomp35.C: New test.
+
 2018-01-23  Nathan Sidwell  <nathan@acm.org>
 
        * g++.dg/cpp0x/range-for10.C: Adjust.
diff --git a/gcc/testsuite/g++.dg/cpp1z/decomp35.C b/gcc/testsuite/g++.dg/cpp1z/decomp35.C
new file mode 100644 (file)
index 0000000..844ad43
--- /dev/null
@@ -0,0 +1,35 @@
+// PR c++/83958
+// { dg-do compile { target c++11 } }
+// { dg-options "" }
+
+template <typename = void> struct A;
+class B;
+template <typename, typename, typename = A<>> class C;
+template <typename, typename> struct D;
+template <typename T, typename U, typename V, typename, typename, typename W>
+struct E {
+  using X = W;
+  X operator* ();
+  T operator++ ();
+  template <typename P, typename R, typename S, typename Q>
+  bool operator!= (E<P, U, V, R, S, Q>);
+};
+template <typename T, typename U, typename>
+struct F {
+  class G;
+  using H = D<T, U>;
+  using I = E<G, T, U, G, H, H &>;
+  class G : public I {};
+  G begin ();
+  G end ();
+};
+template <typename T, typename U, typename V> struct C : F<T, U, V> {
+  using J = F<T, U, V>;
+  using J::begin;
+  using J::end;
+};
+using K = class L;
+struct M {
+  void foo () { for (auto & [ a ] : m) {} }    // { dg-error "incomplete type" }
+  C<K, B> m;                                   // { dg-warning "only available with" "" { target c++14_down } .-1 }
+};