OSDN Git Service

* ifcvt.c (noce_emit_cmove): If both of the values are SUBREGs, try
[pf3gnuchains/gcc-fork.git] / gcc / ipa-split.c
index 28f96b2..23882b2 100644 (file)
@@ -120,27 +120,46 @@ struct split_point
 
   /* Basic blocks we are splitting away.  */
   bitmap split_bbs;
+
+  /* True when return value is computed on split part and thus it needs
+     to be returned.  */
+  bool split_part_set_retval;
 };
 
 /* Best split point found.  */
 
 struct split_point best_split_point;
 
-/* Callback for walk_stmt_load_store_addr_ops.  If T is non-ssa automatic
+static tree find_retval (basic_block return_bb);
+
+/* Callback for walk_stmt_load_store_addr_ops.  If T is non-SSA automatic
    variable, check it if it is present in bitmap passed via DATA.  */
 
 static bool
-test_nonssa_use (gimple stmt ATTRIBUTE_UNUSED, tree t,
-                void *data ATTRIBUTE_UNUSED)
+test_nonssa_use (gimple stmt ATTRIBUTE_UNUSED, tree t, void *data)
 {
   t = get_base_address (t);
 
-  if (t && !is_gimple_reg (t)
-      && ((TREE_CODE (t) == VAR_DECL
+  if (!t || is_gimple_reg (t))
+    return false;
+
+  if (TREE_CODE (t) == PARM_DECL
+      || (TREE_CODE (t) == VAR_DECL
          && auto_var_in_fn_p (t, current_function_decl))
-         || (TREE_CODE (t) == RESULT_DECL)
-         || (TREE_CODE (t) == PARM_DECL)))
+      || TREE_CODE (t) == RESULT_DECL
+      || TREE_CODE (t) == LABEL_DECL)
     return bitmap_bit_p ((bitmap)data, DECL_UID (t));
+
+  /* For DECL_BY_REFERENCE, the return value is actually a pointer.  We want
+     to pretend that the value pointed to is actual result decl.  */
+  if ((TREE_CODE (t) == MEM_REF || INDIRECT_REF_P (t))
+      && TREE_CODE (TREE_OPERAND (t, 0)) == SSA_NAME
+      && TREE_CODE (SSA_NAME_VAR (TREE_OPERAND (t, 0))) == RESULT_DECL
+      && DECL_BY_REFERENCE (DECL_RESULT (current_function_decl)))
+    return
+      bitmap_bit_p ((bitmap)data,
+                   DECL_UID (DECL_RESULT (current_function_decl)));
+
   return false;
 }
 
@@ -159,8 +178,8 @@ dump_split_point (FILE * file, struct split_point *current)
   dump_bitmap (file, current->ssa_names_to_pass);
 }
 
-/* Look for all BBs in header that might lead to split part and verify that
-   they are not defining any of SSA vars used by split part. 
+/* Look for all BBs in header that might lead to the split part and verify
+   that they are not defining any non-SSA var used by the split part.
    Parameters are the same as for consider_split.  */
 
 static bool
@@ -172,7 +191,7 @@ verify_non_ssa_vars (struct split_point *current, bitmap non_ssa_vars,
   edge e;
   edge_iterator ei;
   bool ok = true;
-  
+
   FOR_EACH_EDGE (e, ei, current->entry_bb->preds)
     if (e->src != ENTRY_BLOCK_PTR
        && !bitmap_bit_p (current->split_bbs, e->src->index))
@@ -180,7 +199,7 @@ verify_non_ssa_vars (struct split_point *current, bitmap non_ssa_vars,
         VEC_safe_push (basic_block, heap, worklist, e->src);
        bitmap_set_bit (seen, e->src->index);
       }
-  
+
   while (!VEC_empty (basic_block, worklist))
     {
       gimple_stmt_iterator bsi;
@@ -188,30 +207,37 @@ verify_non_ssa_vars (struct split_point *current, bitmap non_ssa_vars,
 
       FOR_EACH_EDGE (e, ei, bb->preds)
        if (e->src != ENTRY_BLOCK_PTR
-           && !bitmap_bit_p (seen, e->src->index))
+           && bitmap_set_bit (seen, e->src->index))
          {
            gcc_checking_assert (!bitmap_bit_p (current->split_bbs,
                                                e->src->index));
            VEC_safe_push (basic_block, heap, worklist, e->src);
-           bitmap_set_bit (seen, e->src->index);
          }
       for (bsi = gsi_start_bb (bb); !gsi_end_p (bsi); gsi_next (&bsi))
        {
-         if (is_gimple_debug (gsi_stmt (bsi)))
+         gimple stmt = gsi_stmt (bsi);
+         if (is_gimple_debug (stmt))
            continue;
          if (walk_stmt_load_store_addr_ops
-             (gsi_stmt (bsi), non_ssa_vars, test_nonssa_use,
-              test_nonssa_use, test_nonssa_use))
+             (stmt, non_ssa_vars, test_nonssa_use, test_nonssa_use,
+              test_nonssa_use))
            {
              ok = false;
              goto done;
            }
+         if (gimple_code (stmt) == GIMPLE_LABEL
+             && test_nonssa_use (stmt, gimple_label_label (stmt),
+                                 non_ssa_vars))
+         {
+           ok = false;
+           goto done;
+         }
        }
       for (bsi = gsi_start_phis (bb); !gsi_end_p (bsi); gsi_next (&bsi))
        {
          if (walk_stmt_load_store_addr_ops
-             (gsi_stmt (bsi), non_ssa_vars, test_nonssa_use,
-              test_nonssa_use, test_nonssa_use))
+             (gsi_stmt (bsi), non_ssa_vars, test_nonssa_use, test_nonssa_use,
+              test_nonssa_use))
            {
              ok = false;
              goto done;
@@ -260,6 +286,7 @@ consider_split (struct split_point *current, bitmap non_ssa_vars,
   gimple_stmt_iterator bsi;
   unsigned int i;
   int incomming_freq = 0;
+  tree retval;
 
   if (dump_file && (dump_flags & TDF_DETAILS))
     dump_split_point (dump_file, current);
@@ -319,7 +346,7 @@ consider_split (struct split_point *current, bitmap non_ssa_vars,
      call overhead.  */
   call_overhead = eni_size_weights.call_cost;
   for (parm = DECL_ARGUMENTS (current_function_decl); parm;
-       parm = TREE_CHAIN (parm))
+       parm = DECL_CHAIN (parm))
     {
       if (!is_gimple_reg (parm))
        {
@@ -388,6 +415,43 @@ consider_split (struct split_point *current, bitmap non_ssa_vars,
   if (dump_file && (dump_flags & TDF_DETAILS))
     fprintf (dump_file, "  Accepted!\n");
 
+  /* See if retval used by return bb is computed by header or split part.
+     When it is computed by split part, we need to produce return statement
+     in the split part and add code to header to pass it around.
+
+     This is bit tricky to test:
+       1) When there is no return_bb or no return value, we always pass
+          value around.
+       2) Invariants are always computed by caller.
+       3) For SSA we need to look if defining statement is in header or split part
+       4) For non-SSA we need to look where the var is computed. */
+  retval = find_retval (return_bb);
+  if (!retval)
+    current->split_part_set_retval = true;
+  else if (is_gimple_min_invariant (retval))
+    current->split_part_set_retval = false;
+  /* Special case is value returned by reference we record as if it was non-ssa
+     set to result_decl.  */
+  else if (TREE_CODE (retval) == SSA_NAME
+          && TREE_CODE (SSA_NAME_VAR (retval)) == RESULT_DECL
+          && DECL_BY_REFERENCE (DECL_RESULT (current_function_decl)))
+    current->split_part_set_retval
+       = bitmap_bit_p (non_ssa_vars, DECL_UID (SSA_NAME_VAR (retval)));
+  else if (TREE_CODE (retval) == SSA_NAME)
+    current->split_part_set_retval
+      = (!SSA_NAME_IS_DEFAULT_DEF (retval)
+        && (bitmap_bit_p (current->split_bbs,
+                         gimple_bb (SSA_NAME_DEF_STMT (retval))->index)
+            || gimple_bb (SSA_NAME_DEF_STMT (retval)) == return_bb));
+  else if (TREE_CODE (retval) == PARM_DECL)
+    current->split_part_set_retval = false;
+  else if (TREE_CODE (retval) == VAR_DECL
+          || TREE_CODE (retval) == RESULT_DECL)
+    current->split_part_set_retval
+      = bitmap_bit_p (non_ssa_vars, DECL_UID (retval));
+  else
+    current->split_part_set_retval = true;
+
   /* At the moment chose split point with lowest frequency and that leaves
      out smallest size of header.
      In future we might re-consider this heuristics.  */
@@ -491,13 +555,12 @@ find_retval (basic_block return_bb)
   return NULL;
 }
 
-/* Callback for walk_stmt_load_store_addr_ops.  If T is non-ssa automatic
-   variable, mark it as used in bitmap passed via DATA. 
+/* Callback for walk_stmt_load_store_addr_ops.  If T is non-SSA automatic
+   variable, mark it as used in bitmap passed via DATA.
    Return true when access to T prevents splitting the function.  */
 
 static bool
-mark_nonssa_use (gimple stmt ATTRIBUTE_UNUSED, tree t,
-                void *data ATTRIBUTE_UNUSED)
+mark_nonssa_use (gimple stmt ATTRIBUTE_UNUSED, tree t, void *data)
 {
   t = get_base_address (t);
 
@@ -509,13 +572,27 @@ mark_nonssa_use (gimple stmt ATTRIBUTE_UNUSED, tree t,
   if (TREE_CODE (t) == PARM_DECL)
     {
       if (dump_file && (dump_flags & TDF_DETAILS))
-       fprintf (dump_file, "Can not split use of non-ssa function parameter.\n");
+       fprintf (dump_file,
+                "Cannot split: use of non-ssa function parameter.\n");
       return true;
     }
 
-  if ((TREE_CODE (t) == VAR_DECL && auto_var_in_fn_p (t, current_function_decl))
-      || (TREE_CODE (t) == RESULT_DECL))
+  if ((TREE_CODE (t) == VAR_DECL
+       && auto_var_in_fn_p (t, current_function_decl))
+      || TREE_CODE (t) == RESULT_DECL
+      || TREE_CODE (t) == LABEL_DECL)
     bitmap_set_bit ((bitmap)data, DECL_UID (t));
+
+  /* For DECL_BY_REFERENCE, the return value is actually a pointer.  We want
+     to pretend that the value pointed to is actual result decl.  */
+  if ((TREE_CODE (t) == MEM_REF || INDIRECT_REF_P (t))
+      && TREE_CODE (TREE_OPERAND (t, 0)) == SSA_NAME
+      && TREE_CODE (SSA_NAME_VAR (TREE_OPERAND (t, 0))) == RESULT_DECL
+      && DECL_BY_REFERENCE (DECL_RESULT (current_function_decl)))
+    return
+      bitmap_bit_p ((bitmap)data,
+                   DECL_UID (DECL_RESULT (current_function_decl)));
+
   return false;
 }
 
@@ -558,13 +635,13 @@ visit_bb (basic_block bb, basic_block return_bb,
          && stmt_can_throw_external (stmt))
        {
          if (dump_file && (dump_flags & TDF_DETAILS))
-           fprintf (dump_file, "Can not split external resx.\n");
+           fprintf (dump_file, "Cannot split: external resx.\n");
          can_split = false;
        }
       if (gimple_code (stmt) == GIMPLE_EH_DISPATCH)
        {
          if (dump_file && (dump_flags & TDF_DETAILS))
-           fprintf (dump_file, "Can not split eh dispatch.\n");
+           fprintf (dump_file, "Cannot split: eh dispatch.\n");
          can_split = false;
        }
 
@@ -583,12 +660,13 @@ visit_bb (basic_block bb, basic_block return_bb,
          case BUILT_IN_APPLY:
          case BUILT_IN_VA_START:
            if (dump_file && (dump_flags & TDF_DETAILS))
-             fprintf (dump_file, "Can not split builtin_apply and va_start.\n");
+             fprintf (dump_file,
+                      "Cannot split: builtin_apply and va_start.\n");
            can_split = false;
            break;
          case BUILT_IN_EH_POINTER:
            if (dump_file && (dump_flags & TDF_DETAILS))
-             fprintf (dump_file, "Can not split builtin_eh_pointer.\n");
+             fprintf (dump_file, "Cannot split: builtin_eh_pointer.\n");
            can_split = false;
            break;
          default:
@@ -630,7 +708,6 @@ visit_bb (basic_block bb, basic_block return_bb,
   FOR_EACH_EDGE (e, ei, bb->succs)
     if (e->dest == return_bb)
       {
-       bool found_phi = false;
        for (bsi = gsi_start_phis (return_bb); !gsi_end_p (bsi); gsi_next (&bsi))
          {
            gimple stmt = gsi_stmt (bsi);
@@ -640,25 +717,11 @@ visit_bb (basic_block bb, basic_block return_bb,
              continue;
            if (!is_gimple_reg (gimple_phi_result (stmt)))
              continue;
-           found_phi = true;
            if (TREE_CODE (op) == SSA_NAME)
              bitmap_set_bit (used_ssa_names, SSA_NAME_VERSION (op));
            else
              can_split &= !mark_nonssa_use (stmt, op, non_ssa_vars);
          }
-       if (!gsi_end_p (gsi_last_bb (return_bb)))
-         {
-           ssa_op_iter iter;
-           gimple stmt = gsi_stmt (gsi_last_bb (return_bb));
-           tree op;
-           if (!found_phi)
-             FOR_EACH_SSA_TREE_OPERAND (op, stmt, iter, SSA_OP_USE)
-               bitmap_set_bit (used_ssa_names, SSA_NAME_VERSION (op));
-           can_split &= !walk_stmt_load_store_addr_ops (stmt, non_ssa_vars,
-                                                        mark_nonssa_use,
-                                                        mark_nonssa_use,
-                                                        mark_nonssa_use);
-         }
       }
   return can_split;
 }
@@ -858,6 +921,7 @@ find_split_points (int overall_time, int overall_size)
   ENTRY_BLOCK_PTR->aux = NULL;
   FOR_EACH_BB (bb)
     bb->aux = NULL;
+  VEC_free (stack_entry, heap, stack);
   BITMAP_FREE (current.ssa_names_to_pass);
 }
 
@@ -889,7 +953,7 @@ split_function (struct split_point *split_point)
 
   /* Collect the parameters of new function and args_to_skip bitmap.  */
   for (parm = DECL_ARGUMENTS (current_function_decl);
-       parm; parm = TREE_CHAIN (parm), num++)
+       parm; parm = DECL_CHAIN (parm), num++)
     if (!is_gimple_reg (parm)
        || !gimple_default_def (cfun, parm)
        || !bitmap_bit_p (split_point->ssa_names_to_pass,
@@ -905,8 +969,55 @@ split_function (struct split_point *split_point)
   if (e)
     split_part_return_p = true;
 
-  /* If we return, we will need the return block.  */
-  if (return_bb != EXIT_BLOCK_PTR && split_part_return_p)
+  /* Add return block to what will become the split function.
+     We do not return; no return block is needed.  */
+  if (!split_part_return_p)
+    ;
+  /* We have no return block, so nothing is needed.  */
+  else if (return_bb == EXIT_BLOCK_PTR)
+    ;
+  /* When we do not want to return value, we need to construct
+     new return block with empty return statement.
+     FIXME: Once we are able to change return type, we should change function
+     to return void instead of just outputting function with undefined return
+     value.  For structures this affects quality of codegen.  */
+  else if (!split_point->split_part_set_retval
+           && find_retval (return_bb))
+    {
+      bool redirected = true;
+      basic_block new_return_bb = create_basic_block (NULL, 0, return_bb);
+      gimple_stmt_iterator gsi = gsi_start_bb (new_return_bb);
+      gsi_insert_after (&gsi, gimple_build_return (NULL), GSI_NEW_STMT);
+      while (redirected)
+       {
+         redirected = false;
+         FOR_EACH_EDGE (e, ei, return_bb->preds)
+           if (bitmap_bit_p (split_point->split_bbs, e->src->index))
+             {
+               new_return_bb->count += e->count;
+               new_return_bb->frequency += EDGE_FREQUENCY (e);
+               redirect_edge_and_branch (e, new_return_bb);
+               redirected = true;
+               break;
+             }
+       }
+      e = make_edge (new_return_bb, EXIT_BLOCK_PTR, 0);
+      e->probability = REG_BR_PROB_BASE;
+      e->count = new_return_bb->count;
+      bitmap_set_bit (split_point->split_bbs, new_return_bb->index);
+      /* We change CFG in a way tree-inline is not able to compensate on while
+        updating PHIs.  There are only virtuals in return_bb, so recompute
+        them.  */
+      for (gsi = gsi_start_phis (return_bb); !gsi_end_p (gsi);)
+       {
+         gimple stmt = gsi_stmt (gsi);
+         gcc_assert (!is_gimple_reg (gimple_phi_result (stmt)));
+         mark_virtual_phi_result_for_renaming (stmt);
+         remove_phi_node (&gsi, true);
+       }
+    }
+  /* When we pass aorund the value, use existing return block.  */
+  else
     bitmap_set_bit (split_point->split_bbs, return_bb->index);
 
   /* Now create the actual clone.  */
@@ -974,15 +1085,7 @@ split_function (struct split_point *split_point)
        {
          real_retval = retval = find_retval (return_bb);
 
-         /* See if return value is computed by split part;
-            function might just return its argument, invariant or undefined
-            value.  In this case we don't need to do any updating.  */
-         if (real_retval
-             && !is_gimple_min_invariant (retval)
-             && (TREE_CODE (retval) != SSA_NAME
-                 || (!SSA_NAME_IS_DEFAULT_DEF (retval)
-                     || DECL_BY_REFERENCE
-                          (DECL_RESULT (current_function_decl)))))
+         if (real_retval && split_point->split_part_set_retval)
            {
              gimple_stmt_iterator psi;
 
@@ -1038,7 +1141,8 @@ split_function (struct split_point *split_point)
       else
        {
          gimple ret;
-         if (!VOID_TYPE_P (TREE_TYPE (TREE_TYPE (current_function_decl))))
+         if (split_point->split_part_set_retval
+             && !VOID_TYPE_P (TREE_TYPE (TREE_TYPE (current_function_decl))))
            {
              retval = DECL_RESULT (current_function_decl);