Bootstrapped/regtested on x86_64-pc-linux-gnu, ok for trunk?

-- >8 --
The recent r15-6369 unfortunately caused a bad wrong-code issue.
Here we have

  TARGET_EXPR <D.2996, (void) (D.2996 = {.status=0, .data={._vptr.Foo=&_ZTV3Foo 
+ 16}})>

and call cp_fold_r -> maybe_constant_init with object=D.2996.  In
cxx_eval_outermost_constant_expr we now take the type of the object
if present.  An object can't have type 'void' and so we continue to
evaluate the initializer.  That evaluates into a VOID_CST, meaning
we disregard the whole initializer, and terrible things ensue.

I think the new prvalue optimization (r15-6052) should only be enabled
for simple TARGET_EXPRs -- ones that don't initialize the object via
a sequence of statement as the one above.

I also think we want an assert so that this doesn't happen again.

        PR c++/118396
        PR c++/118523

gcc/cp/ChangeLog:

        * constexpr.cc (cxx_eval_outermost_constant_expr): Assert
        that we don't throw away the initializer by evaluating it away.
        * cp-gimplify.cc (cp_fold_r): Only attempt to evaluate the initializer
        if the TARGET_EXPR is simple.

gcc/testsuite/ChangeLog:

        * g++.dg/tree-ssa/pr78687.C: Revert r15-6052.
        * g++.dg/tree-ssa/pr90883.C: Likewise.
        * g++.dg/cpp0x/constexpr-prvalue4.C: New test.
        * g++.dg/cpp1y/constexpr-prvalue3.C: New test.
---
 gcc/cp/constexpr.cc                           | 14 ++++--
 gcc/cp/cp-gimplify.cc                         |  4 +-
 .../g++.dg/cpp0x/constexpr-prvalue4.C         | 33 ++++++++++++++
 .../g++.dg/cpp1y/constexpr-prvalue3.C         | 45 +++++++++++++++++++
 gcc/testsuite/g++.dg/tree-ssa/pr78687.C       |  3 +-
 gcc/testsuite/g++.dg/tree-ssa/pr90883.C       |  3 +-
 6 files changed, 94 insertions(+), 8 deletions(-)
 create mode 100644 gcc/testsuite/g++.dg/cpp0x/constexpr-prvalue4.C
 create mode 100644 gcc/testsuite/g++.dg/cpp1y/constexpr-prvalue3.C

diff --git a/gcc/cp/constexpr.cc b/gcc/cp/constexpr.cc
index c898e3bfa6e..61b25277419 100644
--- a/gcc/cp/constexpr.cc
+++ b/gcc/cp/constexpr.cc
@@ -8876,9 +8876,17 @@ cxx_eval_outermost_constant_expr (tree t, bool 
allow_non_constant,
   /* Turn off -frounding-math for manifestly constant evaluation.  */
   warning_sentinel rm (flag_rounding_math,
                       ctx.manifestly_const_eval == mce_true);
-  tree type = (object
-              ? cv_unqualified (TREE_TYPE (object))
-              : initialized_type (t));
+  tree type;
+  if (object)
+    {
+      type = cv_unqualified (TREE_TYPE (object));
+      /* If there is an object to initialize, make sure we don't throw
+        away the initializer.  */
+      gcc_assert (!VOID_TYPE_P (initialized_type (t)) || constexpr_dtor);
+    }
+  else
+    type = initialized_type (t);
+
   tree r = t;
   bool is_consteval = false;
   if (VOID_TYPE_P (type))
diff --git a/gcc/cp/cp-gimplify.cc b/gcc/cp/cp-gimplify.cc
index c7074b00cef..283c0fa3e26 100644
--- a/gcc/cp/cp-gimplify.cc
+++ b/gcc/cp/cp-gimplify.cc
@@ -1475,7 +1475,9 @@ cp_fold_r (tree *stmt_p, int *walk_subtrees, void *data_)
          cp_walk_tree (&init, cp_fold_r, data, NULL);
          cp_walk_tree (&TARGET_EXPR_CLEANUP (stmt), cp_fold_r, data, NULL);
          *walk_subtrees = 0;
-         if (!flag_no_inline)
+         /* Only attempt to evaluate the initializer if we're inlining and
+            the TARGET_EXPR is simple.  */
+         if (!flag_no_inline && !VOID_TYPE_P (TREE_TYPE (init)))
            {
              tree folded = maybe_constant_init (init, TARGET_EXPR_SLOT (stmt));
              if (folded != init && TREE_CONSTANT (folded))
diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-prvalue4.C 
b/gcc/testsuite/g++.dg/cpp0x/constexpr-prvalue4.C
new file mode 100644
index 00000000000..afcee65f880
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-prvalue4.C
@@ -0,0 +1,33 @@
+// PR c++/118396
+// { dg-do run { target c++11 } }
+// { dg-options "-O" }
+
+void *operator new(__SIZE_TYPE__, void *__p) { return __p; }
+
+struct Foo {
+  virtual ~Foo() = default;
+};
+struct Data {
+  int status;
+  Foo data{};
+};
+
+Data *P, *Q;
+
+struct vector {
+  vector (const Data &__value) {
+    P = static_cast<Data *>(__builtin_operator_new(0));
+    new (P) Data (__value);
+    Q = P + 1;
+  }
+  Data *begin() { return P; }
+  Data *end() { return Q; }
+};
+
+int
+main ()
+{
+  vector items_(Data{});
+  for (auto item : items_)
+    item.status == 0 ? void() : __builtin_abort ();
+}
diff --git a/gcc/testsuite/g++.dg/cpp1y/constexpr-prvalue3.C 
b/gcc/testsuite/g++.dg/cpp1y/constexpr-prvalue3.C
new file mode 100644
index 00000000000..8ea86c60be5
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp1y/constexpr-prvalue3.C
@@ -0,0 +1,45 @@
+// PR c++/118523
+// { dg-do compile { target c++14 } }
+// { dg-options "-O2 -Wall" }
+
+struct __new_allocator {
+  constexpr __new_allocator() {}
+  __new_allocator(__new_allocator &) {}
+};
+template <typename> using __allocator_base = __new_allocator;
+template <typename> struct allocator_traits;
+template <typename> struct allocator : __allocator_base<int> {};
+template <typename _Tp> struct allocator_traits<allocator<_Tp>> {
+  using pointer = _Tp *;
+  template <typename _Up> using rebind_alloc = allocator<_Up>;
+  static void deallocate(allocator<_Tp>, pointer, long);
+};
+struct __alloc_traits : allocator_traits<allocator<int>> {};
+struct _Vector_impl_data {
+  __alloc_traits::pointer _M_start;
+  __alloc_traits::pointer _M_end_of_storage;
+  constexpr _Vector_impl_data() : _M_start(), _M_end_of_storage() {}
+};
+struct _Vector_impl : __alloc_traits::rebind_alloc<int>, _Vector_impl_data {};
+struct _Vector_base {
+  ~_Vector_base() {
+    _M_deallocate(_M_impl._M_start,
+                  _M_impl._M_end_of_storage - _M_impl._M_start);
+  }
+  _Vector_impl _M_impl;
+  void _M_deallocate(__alloc_traits::pointer __p, long __n) {
+    if (__p)
+      __alloc_traits::deallocate(_M_impl, __p, __n);
+  }
+};
+struct vector : protected _Vector_base {};
+struct S {
+  vector a{};
+};
+struct B2 {
+  B2(S);
+};
+struct E : B2 {
+  E(S opts = {}) : B2{opts} {}
+};
+void fun() { E{}; }
diff --git a/gcc/testsuite/g++.dg/tree-ssa/pr78687.C 
b/gcc/testsuite/g++.dg/tree-ssa/pr78687.C
index 6e8a5b7efdf..698458f0e9a 100644
--- a/gcc/testsuite/g++.dg/tree-ssa/pr78687.C
+++ b/gcc/testsuite/g++.dg/tree-ssa/pr78687.C
@@ -480,5 +480,4 @@ int main(int argc, char* argv[])
         f();
 }
 
-// XFAILed due to PR116416
-/* { dg-final { scan-tree-dump "Removing load:.*ptr;" "sra" { xfail { *-*-* } 
} } } */
+/* { dg-final { scan-tree-dump "Removing load:.*ptr;" "sra" } } */
diff --git a/gcc/testsuite/g++.dg/tree-ssa/pr90883.C 
b/gcc/testsuite/g++.dg/tree-ssa/pr90883.C
index ad9231eaff2..f540e78fa00 100644
--- a/gcc/testsuite/g++.dg/tree-ssa/pr90883.C
+++ b/gcc/testsuite/g++.dg/tree-ssa/pr90883.C
@@ -16,5 +16,4 @@
 // We want to match enough here to capture that we deleted an empty
 // constructor store
 // mips will expand to loop to clear because CLEAR_RATIO.
-// { dg-final { scan-tree-dump-not ".*\.a = {}" "dse1" { xfail { mips*-*-* } } 
} }
-// { dg-final { scan-tree-dump-not ".*\.b = 0" "dse1" { xfail { mips*-*-* } } 
} }
+// { dg-final { scan-tree-dump "Deleted redundant store: .*\.a = {}" "dse1" { 
xfail { mips*-*-* } } } }

base-commit: 29da6a642402ac64002f5edeab268606b4637103
-- 
2.48.0

Reply via email to