Blame SOURCES/gcc8-rh1960701.patch

5f3da3
--- gcc/cp/call.c
5f3da3
+++ gcc/cp/call.c
5f3da3
@@ -6904,7 +6904,7 @@ convert_like_real (conversion *convs, tree expr, tree fn, int argnum,
5f3da3
 	elttype = cp_build_qualified_type
5f3da3
 	  (elttype, cp_type_quals (elttype) | TYPE_QUAL_CONST);
5f3da3
 	array = build_array_of_n_type (elttype, len);
5f3da3
-	array = finish_compound_literal (array, new_ctor, complain, fcl_c99);
5f3da3
+	array = finish_compound_literal (array, new_ctor, complain);
5f3da3
 	/* Take the address explicitly rather than via decay_conversion
5f3da3
 	   to avoid the error about taking the address of a temporary.  */
5f3da3
 	array = cp_build_addr_expr (array, complain);
5f3da3
@@ -10984,13 +10984,11 @@ set_up_extended_ref_temp (tree decl, tree expr, vec<tree, va_gc> **cleanups,
5f3da3
 	     lvalue-rvalue conversion applied to "a glvalue of literal type
5f3da3
 	     that refers to a non-volatile temporary object initialized
5f3da3
 	     with a constant expression".  Rather than try to communicate
5f3da3
-	     that this VAR_DECL is a temporary, just mark it constexpr.
5f3da3
-
5f3da3
-	     Currently this is only useful for initializer_list temporaries,
5f3da3
-	     since reference vars can't appear in constant expressions.  */
5f3da3
+	     that this VAR_DECL is a temporary, just mark it constexpr.  */
5f3da3
 	  DECL_DECLARED_CONSTEXPR_P (var) = true;
5f3da3
 	  DECL_INITIALIZED_BY_CONSTANT_EXPRESSION_P (var) = true;
5f3da3
 	  TREE_CONSTANT (var) = true;
5f3da3
+	  TREE_READONLY (var) = true;
5f3da3
 	}
5f3da3
       DECL_INITIAL (var) = init;
5f3da3
       init = NULL_TREE;
5f3da3
--- gcc/cp/tree.c
5f3da3
+++ gcc/cp/tree.c
5f3da3
@@ -442,6 +442,14 @@ build_target_expr (tree decl, tree value, tsubst_flags_t complain)
5f3da3
 		       || useless_type_conversion_p (TREE_TYPE (decl),
5f3da3
 						     TREE_TYPE (value)));
5f3da3
 
5f3da3
+  /* Set TREE_READONLY for optimization, such as gimplify_init_constructor
5f3da3
+     moving a constant aggregate into .rodata.  */
5f3da3
+  if (CP_TYPE_CONST_NON_VOLATILE_P (type)
5f3da3
+      && !TYPE_HAS_NONTRIVIAL_DESTRUCTOR (type)
5f3da3
+      && !VOID_TYPE_P (TREE_TYPE (value))
5f3da3
+      && reduced_constant_expression_p (value))
5f3da3
+    TREE_READONLY (decl) = true;
5f3da3
+
5f3da3
   if (complain & tf_no_cleanup)
5f3da3
     /* The caller is building a new-expr and does not need a cleanup.  */
5f3da3
     t = NULL_TREE;
5f3da3
--- /dev/null
5f3da3
+++ gcc/testsuite/g++.dg/cpp1y/pr95226.C
5f3da3
@@ -0,0 +1,17 @@
5f3da3
+// PR c++/95226
5f3da3
+// { dg-do run { target c++14 } }
5f3da3
+
5f3da3
+#include <vector>
5f3da3
+
5f3da3
+struct T {
5f3da3
+  unsigned a;
5f3da3
+  float b {8.};
5f3da3
+};
5f3da3
+
5f3da3
+int main()
5f3da3
+{
5f3da3
+  T t = {1};
5f3da3
+  std::vector<T> tt = {{1}, {2}};
5f3da3
+  if (t.a != 1 || t.b != 8.0f || tt[0].a != 1 || tt[0].b != 8.0f || tt[1].a != 2 || tt[1].b != 8.0f)
5f3da3
+    __builtin_abort ();
5f3da3
+}