OSDN Git Service

2010-04-22 Laurynas Biveinis <laurynas.biveinis@gmail.com>
[pf3gnuchains/gcc-fork.git] / gcc / gimplify.c
index 8d2bc58..287d621 100644 (file)
@@ -27,6 +27,7 @@ along with GCC; see the file COPYING3.  If not see
 #include "tm.h"
 #include "tree.h"
 #include "rtl.h"
+#include "varray.h"
 #include "gimple.h"
 #include "tree-iterator.h"
 #include "tree-inline.h"
@@ -1228,22 +1229,9 @@ gimplify_return_expr (tree stmt, gimple_seq *pre_p)
      hard_function_value generates a PARALLEL, we'll die during normal
      expansion of structure assignments; there's special code in expand_return
      to handle this case that does not exist in expand_expr.  */
-  if (!result_decl)
-    result = NULL_TREE;
-  else if (aggregate_value_p (result_decl, TREE_TYPE (current_function_decl)))
-    {
-      if (TREE_CODE (DECL_SIZE (result_decl)) != INTEGER_CST)
-       {
-         if (!TYPE_SIZES_GIMPLIFIED (TREE_TYPE (result_decl)))
-           gimplify_type_sizes (TREE_TYPE (result_decl), pre_p);
-         /* Note that we don't use gimplify_vla_decl because the RESULT_DECL
-            should be effectively allocated by the caller, i.e. all calls to
-            this function must be subject to the Return Slot Optimization.  */
-         gimplify_one_sizepos (&DECL_SIZE (result_decl), pre_p);
-         gimplify_one_sizepos (&DECL_SIZE_UNIT (result_decl), pre_p);
-       }
-      result = result_decl;
-    }
+  if (!result_decl
+      || aggregate_value_p (result_decl, TREE_TYPE (current_function_decl)))
+    result = result_decl;
   else if (gimplify_ctxp->return_temp)
     result = gimplify_ctxp->return_temp;
   else
@@ -2879,67 +2867,71 @@ static enum gimplify_status
 gimplify_cond_expr (tree *expr_p, gimple_seq *pre_p, fallback_t fallback)
 {
   tree expr = *expr_p;
-  tree type = TREE_TYPE (expr);
-  location_t loc = EXPR_LOCATION (expr);
-  tree tmp, arm1, arm2;
+  tree tmp, type, arm1, arm2;
   enum gimplify_status ret;
   tree label_true, label_false, label_cont;
   bool have_then_clause_p, have_else_clause_p;
   gimple gimple_cond;
   enum tree_code pred_code;
   gimple_seq seq = NULL;
+  location_t loc = EXPR_LOCATION (*expr_p);
+
+  type = TREE_TYPE (expr);
 
   /* If this COND_EXPR has a value, copy the values into a temporary within
      the arms.  */
-  if (!VOID_TYPE_P (type))
+  if (! VOID_TYPE_P (type))
     {
-      tree then_ = TREE_OPERAND (expr, 1), else_ = TREE_OPERAND (expr, 2);
       tree result;
 
-      /* If either an rvalue is ok or we do not require an lvalue, create the
-        temporary.  But we cannot do that if the type is addressable.  */
-      if (((fallback & fb_rvalue) || !(fallback & fb_lvalue))
+      /* If an rvalue is ok or we do not require an lvalue, avoid creating
+        an addressable temporary.  */
+      if (((fallback & fb_rvalue)
+          || !(fallback & fb_lvalue))
          && !TREE_ADDRESSABLE (type))
        {
          if (gimplify_ctxp->allow_rhs_cond_expr
              /* If either branch has side effects or could trap, it can't be
                 evaluated unconditionally.  */
-             && !TREE_SIDE_EFFECTS (then_)
-             && !generic_expr_could_trap_p (then_)
-             && !TREE_SIDE_EFFECTS (else_)
-             && !generic_expr_could_trap_p (else_))
+             && !TREE_SIDE_EFFECTS (TREE_OPERAND (*expr_p, 1))
+             && !generic_expr_could_trap_p (TREE_OPERAND (*expr_p, 1))
+             && !TREE_SIDE_EFFECTS (TREE_OPERAND (*expr_p, 2))
+             && !generic_expr_could_trap_p (TREE_OPERAND (*expr_p, 2)))
            return gimplify_pure_cond_expr (expr_p, pre_p);
 
-         tmp = create_tmp_var (type, "iftmp");
-         result = tmp;
+         result = tmp = create_tmp_var (TREE_TYPE (expr), "iftmp");
+         ret = GS_ALL_DONE;
        }
-
-      /* Otherwise, only create and copy references to the values.  */
       else
        {
-         type = build_pointer_type (type);
+         tree type = build_pointer_type (TREE_TYPE (expr));
 
-         if (!VOID_TYPE_P (TREE_TYPE (then_)))
-           then_ = build_fold_addr_expr_loc (loc, then_);
+         if (TREE_TYPE (TREE_OPERAND (expr, 1)) != void_type_node)
+           TREE_OPERAND (expr, 1) =
+             build_fold_addr_expr_loc (loc, TREE_OPERAND (expr, 1));
 
-         if (!VOID_TYPE_P (TREE_TYPE (else_)))
-           else_ = build_fold_addr_expr_loc (loc, else_);
-         expr
-           = build3 (COND_EXPR, type, TREE_OPERAND (expr, 0), then_, else_);
+         if (TREE_TYPE (TREE_OPERAND (expr, 2)) != void_type_node)
+           TREE_OPERAND (expr, 2) =
+             build_fold_addr_expr_loc (loc, TREE_OPERAND (expr, 2));
 
          tmp = create_tmp_var (type, "iftmp");
+
+         expr = build3 (COND_EXPR, void_type_node, TREE_OPERAND (expr, 0),
+                        TREE_OPERAND (expr, 1), TREE_OPERAND (expr, 2));
+
          result = build_fold_indirect_ref_loc (loc, tmp);
        }
 
-      /* Build the new then clause, `tmp = then_;'.  But don't build the
-        assignment if the value is void; in C++ it can be if it's a throw.  */
-      if (!VOID_TYPE_P (TREE_TYPE (then_)))
-       TREE_OPERAND (expr, 1) = build2 (MODIFY_EXPR, type, tmp, then_);
+      /* Build the then clause, 't1 = a;'.  But don't build an assignment
+        if this branch is void; in C++ it can be, if it's a throw.  */
+      if (TREE_TYPE (TREE_OPERAND (expr, 1)) != void_type_node)
+       TREE_OPERAND (expr, 1)
+         = build2 (MODIFY_EXPR, TREE_TYPE (tmp), tmp, TREE_OPERAND (expr, 1));
 
-      /* Similarly, build the new else clause, `tmp = else_;'.  */
-      if (!VOID_TYPE_P (TREE_TYPE (else_)))
-       TREE_OPERAND (expr, 2) = build2 (MODIFY_EXPR, type, tmp, else_);
+      /* Build the else clause, 't1 = b;'.  */
+      if (TREE_TYPE (TREE_OPERAND (expr, 2)) != void_type_node)
+       TREE_OPERAND (expr, 2)
+         = build2 (MODIFY_EXPR, TREE_TYPE (tmp), tmp, TREE_OPERAND (expr, 2));
 
       TREE_TYPE (expr) = void_type_node;
       recalculate_side_effects (expr);
@@ -3770,11 +3762,25 @@ gimplify_init_constructor (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p,
                && num_nonzero_elements > 1
                && !can_move_by_pieces (size, align))
              {
+               tree new_tree;
+
                if (notify_temp_creation)
                  return GS_ERROR;
 
-               walk_tree (&ctor, force_labels_r, NULL, NULL);
-               TREE_OPERAND (*expr_p, 1) = tree_output_constant_def (ctor);
+               new_tree = create_tmp_var_raw (type, "C");
+
+               gimple_add_tmp_var (new_tree);
+               TREE_STATIC (new_tree) = 1;
+               TREE_READONLY (new_tree) = 1;
+               DECL_INITIAL (new_tree) = ctor;
+               if (align > DECL_ALIGN (new_tree))
+                 {
+                   DECL_ALIGN (new_tree) = align;
+                   DECL_USER_ALIGN (new_tree) = 1;
+                 }
+               walk_tree (&DECL_INITIAL (new_tree), force_labels_r, NULL, NULL);
+
+               TREE_OPERAND (*expr_p, 1) = new_tree;
 
                /* This is no longer an assignment of a CONSTRUCTOR, but
                   we still may have processing to do on the LHS.  So
@@ -4102,252 +4108,253 @@ gimplify_modify_expr_rhs (tree *expr_p, tree *from_p, tree *to_p,
                          gimple_seq *pre_p, gimple_seq *post_p,
                          bool want_value)
 {
-  enum gimplify_status ret = GS_UNHANDLED;
-  bool changed;
+  enum gimplify_status ret = GS_OK;
 
-  do
-    {
-      changed = false;
-      switch (TREE_CODE (*from_p))
-       {
-       case VAR_DECL:
-         /* If we're assigning from a read-only variable initialized with
-            a constructor, do the direct assignment from the constructor,
-            but only if neither source nor target are volatile since this
-            latter assignment might end up being done on a per-field basis.  */
-         if (DECL_INITIAL (*from_p)
-             && TREE_READONLY (*from_p)
-             && !TREE_THIS_VOLATILE (*from_p)
-             && !TREE_THIS_VOLATILE (*to_p)
-             && TREE_CODE (DECL_INITIAL (*from_p)) == CONSTRUCTOR)
-           {
-             tree old_from = *from_p;
-             enum gimplify_status subret;
-
-             /* Move the constructor into the RHS.  */
-             *from_p = unshare_expr (DECL_INITIAL (*from_p));
-
-             /* Let's see if gimplify_init_constructor will need to put
-                it in memory.  */
-             subret = gimplify_init_constructor (expr_p, NULL, NULL,
-                                                 false, true);
-             if (subret == GS_ERROR)
-               {
-                 /* If so, revert the change.  */
-                 *from_p = old_from;
-               }
-             else
-               {
-                 ret = GS_OK;
-                 changed = true;
-               }
-           }
-         break;
-       case INDIRECT_REF:
+  while (ret != GS_UNHANDLED)
+    switch (TREE_CODE (*from_p))
+      {
+      case VAR_DECL:
+       /* If we're assigning from a read-only variable initialized with
+          a constructor, do the direct assignment from the constructor,
+          but only if neither source nor target are volatile since this
+          latter assignment might end up being done on a per-field basis.  */
+       if (DECL_INITIAL (*from_p)
+           && TREE_READONLY (*from_p)
+           && !TREE_THIS_VOLATILE (*from_p)
+           && !TREE_THIS_VOLATILE (*to_p)
+           && TREE_CODE (DECL_INITIAL (*from_p)) == CONSTRUCTOR)
          {
-           /* If we have code like
+           tree old_from = *from_p;
 
-            *(const A*)(A*)&x
+           /* Move the constructor into the RHS.  */
+           *from_p = unshare_expr (DECL_INITIAL (*from_p));
 
-            where the type of "x" is a (possibly cv-qualified variant
-            of "A"), treat the entire expression as identical to "x".
-            This kind of code arises in C++ when an object is bound
-            to a const reference, and if "x" is a TARGET_EXPR we want
-            to take advantage of the optimization below.  */
-           tree t = gimple_fold_indirect_ref_rhs (TREE_OPERAND (*from_p, 0));
-           if (t)
+           /* Let's see if gimplify_init_constructor will need to put
+              it in memory.  If so, revert the change.  */
+           ret = gimplify_init_constructor (expr_p, NULL, NULL, false, true);
+           if (ret == GS_ERROR)
              {
-               *from_p = t;
-               ret = GS_OK;
-               changed = true;
+               *from_p = old_from;
+               /* Fall through.  */
              }
-           break;
-         }
-
-       case TARGET_EXPR:
-         {
-           /* If we are initializing something from a TARGET_EXPR, strip the
-              TARGET_EXPR and initialize it directly, if possible.  This can't
-              be done if the initializer is void, since that implies that the
-              temporary is set in some non-trivial way.
-
-              ??? What about code that pulls out the temp and uses it
-              elsewhere? I think that such code never uses the TARGET_EXPR as
-              an initializer.  If I'm wrong, we'll die because the temp won't
-              have any RTL.  In that case, I guess we'll need to replace
-              references somehow.  */
-           tree init = TARGET_EXPR_INITIAL (*from_p);
-
-           if (init
-               && !VOID_TYPE_P (TREE_TYPE (init)))
+           else
              {
-               *from_p = init;
                ret = GS_OK;
-               changed = true;
+               break;
              }
          }
-         break;
-
-       case COMPOUND_EXPR:
-         /* Remove any COMPOUND_EXPR in the RHS so the following cases will be
-            caught.  */
-         gimplify_compound_expr (from_p, pre_p, true);
-         ret = GS_OK;
-         changed = true;
-         break;
-
-       case CONSTRUCTOR:
-         /* If we're initializing from a CONSTRUCTOR, break this into
-            individual MODIFY_EXPRs.  */
-         return gimplify_init_constructor (expr_p, pre_p, post_p, want_value,
-                                           false);
-
-       case COND_EXPR:
-         /* If we're assigning to a non-register type, push the assignment
-            down into the branches.  This is mandatory for ADDRESSABLE types,
-            since we cannot generate temporaries for such, but it saves a
-            copy in other cases as well.  */
-         if (!is_gimple_reg_type (TREE_TYPE (*from_p)))
-           {
-             /* This code should mirror the code in gimplify_cond_expr. */
-             enum tree_code code = TREE_CODE (*expr_p);
-             tree cond = *from_p;
-             tree result = *to_p;
-
-             ret = gimplify_expr (&result, pre_p, post_p,
-                                  is_gimple_lvalue, fb_lvalue);
-             if (ret != GS_ERROR)
-               ret = GS_OK;
-
-             if (TREE_TYPE (TREE_OPERAND (cond, 1)) != void_type_node)
-               TREE_OPERAND (cond, 1)
-                 = build2 (code, void_type_node, result,
-                           TREE_OPERAND (cond, 1));
-             if (TREE_TYPE (TREE_OPERAND (cond, 2)) != void_type_node)
-               TREE_OPERAND (cond, 2)
-                 = build2 (code, void_type_node, unshare_expr (result),
-                           TREE_OPERAND (cond, 2));
-
-             TREE_TYPE (cond) = void_type_node;
-             recalculate_side_effects (cond);
+       ret = GS_UNHANDLED;
+       break;
+      case INDIRECT_REF:
+       {
+         /* If we have code like
 
-             if (want_value)
-               {
-                 gimplify_and_add (cond, pre_p);
-                 *expr_p = unshare_expr (result);
-               }
-             else
-               *expr_p = cond;
-             return ret;
-           }
-         break;
+               *(const A*)(A*)&x
 
-       case CALL_EXPR:
-         /* For calls that return in memory, give *to_p as the CALL_EXPR's
-            return slot so that we don't generate a temporary.  */
-         if (!CALL_EXPR_RETURN_SLOT_OPT (*from_p)
-             && aggregate_value_p (*from_p, *from_p))
+            where the type of "x" is a (possibly cv-qualified variant
+            of "A"), treat the entire expression as identical to "x".
+            This kind of code arises in C++ when an object is bound
+            to a const reference, and if "x" is a TARGET_EXPR we want
+            to take advantage of the optimization below.  */
+         tree t = gimple_fold_indirect_ref_rhs (TREE_OPERAND (*from_p, 0));
+         if (t)
            {
-             bool use_target;
-
-             if (!(rhs_predicate_for (*to_p))(*from_p))
-               /* If we need a temporary, *to_p isn't accurate.  */
-               use_target = false;
-             else if (TREE_CODE (*to_p) == RESULT_DECL
-                      && DECL_NAME (*to_p) == NULL_TREE
-                      && needs_to_live_in_memory (*to_p))
-               /* It's OK to use the return slot directly unless it's an NRV. */
-               use_target = true;
-             else if (is_gimple_reg_type (TREE_TYPE (*to_p))
-                      || (DECL_P (*to_p) && DECL_REGISTER (*to_p)))
-               /* Don't force regs into memory.  */
-               use_target = false;
-             else if (TREE_CODE (*expr_p) == INIT_EXPR)
-               /* It's OK to use the target directly if it's being
-                  initialized. */
-               use_target = true;
-             else if (!is_gimple_non_addressable (*to_p))
-               /* Don't use the original target if it's already addressable;
-                  if its address escapes, and the called function uses the
-                  NRV optimization, a conforming program could see *to_p
-                  change before the called function returns; see c++/19317.
-                  When optimizing, the return_slot pass marks more functions
-                  as safe after we have escape info.  */
-               use_target = false;
-             else
-               use_target = true;
-
-             if (use_target)
-               {
-                 CALL_EXPR_RETURN_SLOT_OPT (*from_p) = 1;
-                 mark_addressable (*to_p);
-               }
+             *from_p = t;
+             ret = GS_OK;
            }
+         else
+           ret = GS_UNHANDLED;
          break;
+       }
 
-       case WITH_SIZE_EXPR:
-         /* Likewise for calls that return an aggregate of non-constant size,
-            since we would not be able to generate a temporary at all.  */
-         if (TREE_CODE (TREE_OPERAND (*from_p, 0)) == CALL_EXPR)
+      case TARGET_EXPR:
+       {
+         /* If we are initializing something from a TARGET_EXPR, strip the
+            TARGET_EXPR and initialize it directly, if possible.  This can't
+            be done if the initializer is void, since that implies that the
+            temporary is set in some non-trivial way.
+
+            ??? What about code that pulls out the temp and uses it
+            elsewhere? I think that such code never uses the TARGET_EXPR as
+            an initializer.  If I'm wrong, we'll die because the temp won't
+            have any RTL.  In that case, I guess we'll need to replace
+            references somehow.  */
+         tree init = TARGET_EXPR_INITIAL (*from_p);
+
+         if (init
+             && !VOID_TYPE_P (TREE_TYPE (init)))
            {
-             *from_p = TREE_OPERAND (*from_p, 0);
+             *from_p = init;
              ret = GS_OK;
-             changed = true;
            }
-         break;
+         else
+           ret = GS_UNHANDLED;
+       }
+       break;
 
-         /* If we're initializing from a container, push the initialization
-            inside it.  */
-       case CLEANUP_POINT_EXPR:
-       case BIND_EXPR:
-       case STATEMENT_LIST:
+      case COMPOUND_EXPR:
+       /* Remove any COMPOUND_EXPR in the RHS so the following cases will be
+          caught.  */
+       gimplify_compound_expr (from_p, pre_p, true);
+       ret = GS_OK;
+       break;
+
+      case CONSTRUCTOR:
+       /* If we're initializing from a CONSTRUCTOR, break this into
+          individual MODIFY_EXPRs.  */
+       return gimplify_init_constructor (expr_p, pre_p, post_p, want_value,
+                                         false);
+
+      case COND_EXPR:
+       /* If we're assigning to a non-register type, push the assignment
+          down into the branches.  This is mandatory for ADDRESSABLE types,
+          since we cannot generate temporaries for such, but it saves a
+          copy in other cases as well.  */
+       if (!is_gimple_reg_type (TREE_TYPE (*from_p)))
          {
-           tree wrap = *from_p;
-           tree t;
+           /* This code should mirror the code in gimplify_cond_expr. */
+           enum tree_code code = TREE_CODE (*expr_p);
+           tree cond = *from_p;
+           tree result = *to_p;
 
-           ret = gimplify_expr (to_p, pre_p, post_p, is_gimple_min_lval,
-                                fb_lvalue);
+           ret = gimplify_expr (&result, pre_p, post_p,
+                                is_gimple_lvalue, fb_lvalue);
            if (ret != GS_ERROR)
              ret = GS_OK;
 
-           t = voidify_wrapper_expr (wrap, *expr_p);
-           gcc_assert (t == *expr_p);
+           if (TREE_TYPE (TREE_OPERAND (cond, 1)) != void_type_node)
+             TREE_OPERAND (cond, 1)
+               = build2 (code, void_type_node, result,
+                         TREE_OPERAND (cond, 1));
+           if (TREE_TYPE (TREE_OPERAND (cond, 2)) != void_type_node)
+             TREE_OPERAND (cond, 2)
+               = build2 (code, void_type_node, unshare_expr (result),
+                         TREE_OPERAND (cond, 2));
+
+           TREE_TYPE (cond) = void_type_node;
+           recalculate_side_effects (cond);
 
            if (want_value)
              {
-               gimplify_and_add (wrap, pre_p);
-               *expr_p = unshare_expr (*to_p);
+               gimplify_and_add (cond, pre_p);
+               *expr_p = unshare_expr (result);
              }
            else
-             *expr_p = wrap;
-           return GS_OK;
+             *expr_p = cond;
+           return ret;
          }
+       else
+         ret = GS_UNHANDLED;
+       break;
 
-       case COMPOUND_LITERAL_EXPR:
+      case CALL_EXPR:
+       /* For calls that return in memory, give *to_p as the CALL_EXPR's
+          return slot so that we don't generate a temporary.  */
+       if (!CALL_EXPR_RETURN_SLOT_OPT (*from_p)
+           && aggregate_value_p (*from_p, *from_p))
          {
-           tree complit = TREE_OPERAND (*expr_p, 1);
-           tree decl_s = COMPOUND_LITERAL_EXPR_DECL_EXPR (complit);
-           tree decl = DECL_EXPR_DECL (decl_s);
-           tree init = DECL_INITIAL (decl);
-
-           /* struct T x = (struct T) { 0, 1, 2 } can be optimized
-              into struct T x = { 0, 1, 2 } if the address of the
-              compound literal has never been taken.  */
-           if (!TREE_ADDRESSABLE (complit)
-               && !TREE_ADDRESSABLE (decl)
-               && init)
+           bool use_target;
+
+           if (!(rhs_predicate_for (*to_p))(*from_p))
+             /* If we need a temporary, *to_p isn't accurate.  */
+             use_target = false;
+           else if (TREE_CODE (*to_p) == RESULT_DECL
+                    && DECL_NAME (*to_p) == NULL_TREE
+                    && needs_to_live_in_memory (*to_p))
+             /* It's OK to use the return slot directly unless it's an NRV. */
+             use_target = true;
+           else if (is_gimple_reg_type (TREE_TYPE (*to_p))
+                    || (DECL_P (*to_p) && DECL_REGISTER (*to_p)))
+             /* Don't force regs into memory.  */
+             use_target = false;
+           else if (TREE_CODE (*expr_p) == INIT_EXPR)
+             /* It's OK to use the target directly if it's being
+                initialized. */
+             use_target = true;
+           else if (!is_gimple_non_addressable (*to_p))
+             /* Don't use the original target if it's already addressable;
+                if its address escapes, and the called function uses the
+                NRV optimization, a conforming program could see *to_p
+                change before the called function returns; see c++/19317.
+                When optimizing, the return_slot pass marks more functions
+                as safe after we have escape info.  */
+             use_target = false;
+           else
+             use_target = true;
+
+           if (use_target)
              {
-               *expr_p = copy_node (*expr_p);
-               TREE_OPERAND (*expr_p, 1) = init;
-               return GS_OK;
+               CALL_EXPR_RETURN_SLOT_OPT (*from_p) = 1;
+               mark_addressable (*to_p);
              }
          }
 
-       default:
-         break;
+       ret = GS_UNHANDLED;
+       break;
+
+      case WITH_SIZE_EXPR:
+       /* Likewise for calls that return an aggregate of non-constant size,
+          since we would not be able to generate a temporary at all.  */
+       if (TREE_CODE (TREE_OPERAND (*from_p, 0)) == CALL_EXPR)
+         {
+           *from_p = TREE_OPERAND (*from_p, 0);
+           ret = GS_OK;
+         }
+       else
+         ret = GS_UNHANDLED;
+       break;
+
+       /* If we're initializing from a container, push the initialization
+          inside it.  */
+      case CLEANUP_POINT_EXPR:
+      case BIND_EXPR:
+      case STATEMENT_LIST:
+       {
+         tree wrap = *from_p;
+         tree t;
+
+         ret = gimplify_expr (to_p, pre_p, post_p, is_gimple_min_lval,
+                              fb_lvalue);
+         if (ret != GS_ERROR)
+           ret = GS_OK;
+
+         t = voidify_wrapper_expr (wrap, *expr_p);
+         gcc_assert (t == *expr_p);
+
+         if (want_value)
+           {
+             gimplify_and_add (wrap, pre_p);
+             *expr_p = unshare_expr (*to_p);
+           }
+         else
+           *expr_p = wrap;
+         return GS_OK;
        }
-    }
-  while (changed);
+
+      case COMPOUND_LITERAL_EXPR:
+       {
+         tree complit = TREE_OPERAND (*expr_p, 1);
+         tree decl_s = COMPOUND_LITERAL_EXPR_DECL_EXPR (complit);
+         tree decl = DECL_EXPR_DECL (decl_s);
+         tree init = DECL_INITIAL (decl);
+
+         /* struct T x = (struct T) { 0, 1, 2 } can be optimized
+            into struct T x = { 0, 1, 2 } if the address of the
+            compound literal has never been taken.  */
+         if (!TREE_ADDRESSABLE (complit)
+             && !TREE_ADDRESSABLE (decl)
+             && init)
+           {
+             *expr_p = copy_node (*expr_p);
+             TREE_OPERAND (*expr_p, 1) = init;
+             return GS_OK;
+           }
+       }
+
+      default:
+       ret = GS_UNHANDLED;
+       break;
+      }
 
   return ret;
 }
@@ -6628,16 +6635,8 @@ gimplify_expr (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p,
 
        case MODIFY_EXPR:
        case INIT_EXPR:
-         {
-           tree from = TREE_OPERAND (*expr_p, 1);
-           ret = gimplify_modify_expr (expr_p, pre_p, post_p,
-                                       fallback != fb_none);
-           /* Don't let the end of loop logic change GS_OK into GS_ALL_DONE
-              if the RHS has changed.  */
-           if (ret == GS_OK && *expr_p == save_expr
-               && TREE_OPERAND (*expr_p, 1) != from)
-             continue;
-         }
+         ret = gimplify_modify_expr (expr_p, pre_p, post_p,
+                                     fallback != fb_none);
          break;
 
        case TRUTH_ANDIF_EXPR: