re PR c++/89481 (constexpr function allows writing one active union member and readin...
authorJakub Jelinek <jakub@redhat.com>
Tue, 26 Feb 2019 14:37:39 +0000 (15:37 +0100)
committerJakub Jelinek <jakub@gcc.gnu.org>
Tue, 26 Feb 2019 14:37:39 +0000 (15:37 +0100)
PR c++/89481
* constexpr.c (cxx_eval_store_expression): When changing active union
member, set no_zero_init.

* g++.dg/cpp1y/constexpr-89481.C: New test.

From-SVN: r269213

gcc/cp/ChangeLog
gcc/cp/constexpr.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp1y/constexpr-89481.C [new file with mode: 0644]

index 2da2e3d60dc3009149844ff04679895cf8cda68c..6344e4e5febcfe1edec700ac81646735c8a7f82a 100644 (file)
@@ -1,3 +1,9 @@
+2019-02-26  Jakub Jelinek  <jakub@redhat.com>
+
+       PR c++/89481
+       * constexpr.c (cxx_eval_store_expression): When changing active union
+       member, set no_zero_init.
+
 2019-02-23  Marek Polacek  <polacek@redhat.com>
 
        PR c++/88294 - ICE with non-constant noexcept-specifier.
index 40080271c11e8109f8cc6ea8fc6164576180cb15..5bc5efe4b2cc23958aef3ae12fe099779064b812 100644 (file)
@@ -3860,6 +3860,7 @@ cxx_eval_store_expression (const constexpr_ctx *ctx, tree t,
                }
              /* Changing active member.  */
              vec_safe_truncate (CONSTRUCTOR_ELTS (*valp), 0);
+             no_zero_init = true;
            }
 
          for (idx = 0;
index 3f76c4527643f6dd3804d0f09d8e18f8d077db4e..4317ad895ef3a06c27f801ec3fb210d28c66c955 100644 (file)
@@ -1,3 +1,8 @@
+2019-02-26  Jakub Jelinek  <jakub@redhat.com>
+
+       PR c++/89481
+       * g++.dg/cpp1y/constexpr-89481.C: New test.
+
 2019-02-26  Richard Biener  <rguenther@suse.de>
 
        PR tree-optimization/89505
diff --git a/gcc/testsuite/g++.dg/cpp1y/constexpr-89481.C b/gcc/testsuite/g++.dg/cpp1y/constexpr-89481.C
new file mode 100644 (file)
index 0000000..8ac4ef0
--- /dev/null
@@ -0,0 +1,24 @@
+// PR c++/89481
+// { dg-do compile { target c++14 } }
+
+constexpr int
+foo ()
+{
+  union U { long long a; int b[2]; } u { 5LL };
+  u.b[1] = 4;          // { dg-error "change of the active member of a union from" "" { target c++17_down } }
+  return u.b[0];
+}
+
+constexpr int
+bar ()
+{
+  union U { long long a; int b[2]; } u { 5LL };
+  u.b[1] = 4;          // { dg-error "change of the active member of a union from" "" { target c++17_down } }
+  return u.b[1];
+}
+
+static_assert (foo () == 0, "");       // { dg-error "non-constant condition for static assertion" }
+                                       // { dg-message "in 'constexpr' expansion of" "" { target *-*-* } .-1 }
+                                       // { dg-error "accessing uninitialized array element" "" { target c++2a } .-2 }
+static_assert (bar () == 4, "");       // { dg-error "non-constant condition for static assertion" "" { target c++17_down } }
+                                       // { dg-message "in 'constexpr' expansion of" "" { target c++17_down } .-1 }