Hi,
this patch updates inline heuristics so we could analyse also non-SSA vars. I.e.
in:
int aaa(int a)
{
  if (a>4)
   bbb(&a);
}
We are still able to work out that bbb is called only when a>4 despite the fact
that A has address taken.

The patch also makes dataflow to simplify a bit the solution, so we don't have
(a>4 || a<=4) type predicates.

Bootstrapped/regtested x86_64-linux, will commit it later today.

Honza

        * ipa-inline-analysis.c (add_condition): Add conditions parameter;
        simplify obviously true clauses.
        (and_predicates, or_predicates): Add conditions parameter.
        (inline_duplication_hoook): Update.
        (mark_modified): New function.
        (unmodified_parm): New function.
        (eliminated_by_inlining_prob, (set_cond_stmt_execution_predicate,
        set_switch_stmt_execution_predicate, will_be_nonconstant_predicate):
        Use unmodified_parm.
        (estimate_function_body_sizes): Update.
        (remap_predicate): Update.
Index: ipa-inline-analysis.c
===================================================================
--- ipa-inline-analysis.c       (revision 178858)
+++ ipa-inline-analysis.c       (working copy)
@@ -232,11 +232,12 @@ add_condition (struct inline_summary *su
 /* Add clause CLAUSE into the predicate P.  */
 
 static inline void
-add_clause (struct predicate *p, clause_t clause)
+add_clause (conditions conditions, struct predicate *p, clause_t clause)
 {
   int i;
   int i2;
   int insert_here = -1;
+  int c1, c2;
 
   /* True clause.  */
   if (!clause)
@@ -281,6 +282,28 @@ add_clause (struct predicate *p, clause_
       if ((p->clause[i] & clause) != clause)
        i2++;
     }
+
+  /* Look for clauses that are obviously true.  I.e.
+     op0 == 5 || op0 != 5.  */
+  for (c1 = predicate_first_dynamic_condition; c1 < NUM_CONDITIONS; c1++)
+    for (c2 = c1 + 1; c2 <= NUM_CONDITIONS; c2++)
+      if ((clause & (1 << c1))
+         && (clause & (1 << c2)))
+       {
+         condition *cc1 = VEC_index (condition,
+                                     conditions,
+                                     c1 - predicate_first_dynamic_condition);
+         condition *cc2 = VEC_index (condition,
+                                     conditions,
+                                     c2 - predicate_first_dynamic_condition);
+         if (cc1->operand_num == cc2->operand_num
+             && cc1->val == cc2->val
+             && cc1->code == invert_tree_comparison (cc2->code,
+                                                     HONOR_NANS (TYPE_MODE 
(TREE_TYPE (cc1->val)))))
+           return;
+       }
+       
+
   /* We run out of variants.  Be conservative in positive direction.  */
   if (i2 == MAX_CLAUSES)
     return;
@@ -298,7 +321,8 @@ add_clause (struct predicate *p, clause_
 /* Return P & P2.  */
 
 static struct predicate
-and_predicates (struct predicate *p, struct predicate *p2)
+and_predicates (conditions conditions,
+               struct predicate *p, struct predicate *p2)
 {
   struct predicate out = *p;
   int i;
@@ -319,7 +343,7 @@ and_predicates (struct predicate *p, str
   for (; p2->clause[i]; i++)
     {
       gcc_checking_assert (i < MAX_CLAUSES);
-      add_clause (&out, p2->clause[i]);
+      add_clause (conditions, &out, p2->clause[i]);
     }
   return out;
 }
@@ -346,7 +370,7 @@ predicates_equal_p (struct predicate *p,
 /* Return P | P2.  */
 
 static struct predicate
-or_predicates (struct predicate *p, struct predicate *p2)
+or_predicates (conditions conditions, struct predicate *p, struct predicate 
*p2)
 {
   struct predicate out = true_predicate ();
   int i,j;
@@ -364,7 +388,7 @@ or_predicates (struct predicate *p, stru
     for (j = 0; p2->clause[j]; j++)
       {
         gcc_checking_assert (i < MAX_CLAUSES && j < MAX_CLAUSES);
-        add_clause (&out, p->clause[i] | p2->clause[j]);
+        add_clause (conditions, &out, p->clause[i] | p2->clause[j]);
       }
   return out;
 }
@@ -753,7 +777,7 @@ inline_node_duplication_hook (struct cgr
                break;
              }
            else
-             add_clause (&new_predicate,
+             add_clause (info->conds, &new_predicate,
                          possible_truths & e->predicate.clause[j]);
          if (false_predicate_p (&new_predicate))
            {
@@ -781,7 +805,7 @@ inline_node_duplication_hook (struct cgr
                break;
              }
            else
-             add_clause (&new_predicate,
+             add_clause (info->conds, &new_predicate,
                          possible_truths & es->predicate->clause[j]);
          if (false_predicate_p (&new_predicate)
              && !false_predicate_p (es->predicate))
@@ -812,7 +836,7 @@ inline_node_duplication_hook (struct cgr
                break;
              }
            else
-             add_clause (&new_predicate,
+             add_clause (info->conds, &new_predicate,
                          possible_truths & es->predicate->clause[j]);
          if (false_predicate_p (&new_predicate)
              && !false_predicate_p (es->predicate))
@@ -1047,6 +1071,50 @@ initialize_inline_failed (struct cgraph_
     e->inline_failed = CIF_FUNCTION_NOT_CONSIDERED;
 }
 
+/* Callback of walk_aliased_vdefs.  Flags that it has been invoked to the
+   boolean variable pointed to by DATA.  */
+
+static bool
+mark_modified (ao_ref *ao ATTRIBUTE_UNUSED, tree vdef ATTRIBUTE_UNUSED,
+                    void *data)
+{
+  bool *b = (bool *) data;
+  *b = true;
+  return true;
+}
+
+/* If OP reffers to value of function parameter, return 
+   the corresponding parameter.  */
+
+static tree
+unmodified_parm (gimple stmt, tree op)
+{
+  /* SSA_NAME referring to parm default def?  */
+  if (TREE_CODE (op) == SSA_NAME
+      && SSA_NAME_IS_DEFAULT_DEF (op)
+      && TREE_CODE (SSA_NAME_VAR (op)) == PARM_DECL)
+    return SSA_NAME_VAR (op);
+  /* Non-SSA parm reference?  */
+  if (TREE_CODE (op) == PARM_DECL)
+    {
+      bool modified = false;
+
+      ao_ref refd;
+      ao_ref_init (&refd, op);
+      walk_aliased_vdefs (&refd, gimple_vuse (stmt), mark_modified, &modified,
+                         NULL);
+      if (!modified)
+       return op;
+    }
+  /* Assignment from a parameter?  */
+  if (TREE_CODE (op) == SSA_NAME
+      && !SSA_NAME_IS_DEFAULT_DEF (op)
+      && gimple_assign_single_p (SSA_NAME_DEF_STMT (op)))
+    return unmodified_parm (SSA_NAME_DEF_STMT (op),
+                           gimple_assign_rhs1 (SSA_NAME_DEF_STMT (op)));
+  return NULL;
+}
+
 /* See if statement might disappear after inlining.
    0 - means not eliminated
    1 - half of statements goes away
@@ -1058,6 +1126,10 @@ static int
 eliminated_by_inlining_prob (gimple stmt)
 {
   enum gimple_code code = gimple_code (stmt);
+
+  if (!optimize)
+    return 0;
+
   switch (code)
     {
       case GIMPLE_RETURN:
@@ -1090,11 +1162,7 @@ eliminated_by_inlining_prob (gimple stmt
                   || TREE_CODE (inner_rhs) == MEM_REF)
              inner_rhs = TREE_OPERAND (inner_rhs, 0);
 
-
-           if (TREE_CODE (inner_rhs) == PARM_DECL
-               || (TREE_CODE (inner_rhs) == SSA_NAME
-                   && SSA_NAME_IS_DEFAULT_DEF (inner_rhs)
-                   && TREE_CODE (SSA_NAME_VAR (inner_rhs)) == PARM_DECL))
+           if (unmodified_parm (stmt, inner_rhs))
              rhs_free = true;
            if (rhs_free && is_gimple_reg (lhs))
              lhs_free = true;
@@ -1136,6 +1204,7 @@ set_cond_stmt_execution_predicate (struc
   edge_iterator ei;
   gimple set_stmt;
   tree op2;
+  tree parm;
 
   last = last_stmt (bb);
   if (!last
@@ -1147,11 +1216,10 @@ set_cond_stmt_execution_predicate (struc
   /* TODO: handle conditionals like
      var = op0 < 4;
      if (var != 0).  */
-  if (TREE_CODE (op) != SSA_NAME)
-    return;
-  if (SSA_NAME_IS_DEFAULT_DEF (op))
+  parm = unmodified_parm (last, op);
+  if (parm)
     {
-      index = ipa_get_param_decl_index (info, SSA_NAME_VAR (op));
+      index = ipa_get_param_decl_index (info, parm);
       if (index == -1)
        return;
       code = gimple_cond_code (last);
@@ -1170,6 +1238,8 @@ set_cond_stmt_execution_predicate (struc
        }
     }
 
+  if (TREE_CODE (op) != SSA_NAME)
+    return;
   /* Special case
      if (builtin_constant_p (op))
        constant_code
@@ -1185,11 +1255,10 @@ set_cond_stmt_execution_predicate (struc
       || gimple_call_num_args (set_stmt) != 1)
     return;
   op2 = gimple_call_arg (set_stmt, 0);
-  if (TREE_CODE (op2) != SSA_NAME)
-    return;
-  if (!SSA_NAME_IS_DEFAULT_DEF (op2))
+  parm = unmodified_parm (set_stmt, op2);
+  if (!parm)
     return;
-  index = ipa_get_param_decl_index (info, SSA_NAME_VAR (op2));
+  index = ipa_get_param_decl_index (info, parm);
   if (index == -1)
     return;
   if (gimple_cond_code (last) != NE_EXPR
@@ -1223,16 +1292,17 @@ set_switch_stmt_execution_predicate (str
   edge_iterator ei;
   size_t n;
   size_t case_idx;
+  tree parm;
 
   last = last_stmt (bb);
   if (!last
       || gimple_code (last) != GIMPLE_SWITCH)
     return;
   op = gimple_switch_index (last);
-  if (TREE_CODE (op) != SSA_NAME
-      || !SSA_NAME_IS_DEFAULT_DEF (op))
+  parm = unmodified_parm (last, op);
+  if (!parm)
     return;
-  index = ipa_get_param_decl_index (info, SSA_NAME_VAR (op));
+  index = ipa_get_param_decl_index (info, parm);
   if (index == -1)
     return;
 
@@ -1270,10 +1340,10 @@ set_switch_stmt_execution_predicate (str
          p2 = add_condition (summary, index,
                              LE_EXPR,
                              max);
-         p = and_predicates (&p1, &p2);
+         p = and_predicates (summary->conds, &p1, &p2);
        }
       *(struct predicate *)e->aux
-       = or_predicates (&p, (struct predicate *)e->aux);
+       = or_predicates (summary->conds, &p, (struct predicate *)e->aux);
     }
 }
 
@@ -1317,9 +1387,9 @@ compute_bb_predicates (struct cgraph_nod
                {
                  struct predicate this_bb_predicate = *(struct predicate 
*)e->src->aux;
                  if (e->aux)
-                   this_bb_predicate = and_predicates (&this_bb_predicate,
+                   this_bb_predicate = and_predicates (summary->conds, 
&this_bb_predicate,
                                                        (struct predicate 
*)e->aux);
-                 p = or_predicates (&p, &this_bb_predicate);
+                 p = or_predicates (summary->conds, &p, &this_bb_predicate);
                  if (true_predicate_p (&p))
                    break;
                }
@@ -1385,12 +1455,12 @@ will_be_nonconstant_predicate (struct ip
      adding conditionals.  */
   FOR_EACH_SSA_TREE_OPERAND (use, stmt, iter, SSA_OP_USE)
     {
-      if (TREE_CODE (use) != SSA_NAME)
-       return p;
+      tree parm = unmodified_parm (stmt, use);
       /* For arguments we can build a condition.  */
-      if (SSA_NAME_IS_DEFAULT_DEF (use)
-         && ipa_get_param_decl_index (info, SSA_NAME_VAR (use)) >= 0)
+      if (parm && ipa_get_param_decl_index (info, parm) >= 0)
        continue;
+      if (TREE_CODE (use) != SSA_NAME)
+       return p;
       /* If we know when operand is constant,
         we still can say something useful.  */
       if (!true_predicate_p (VEC_index (predicate_t, nonconstant_names,
@@ -1401,15 +1471,15 @@ will_be_nonconstant_predicate (struct ip
   op_non_const = false_predicate ();
   FOR_EACH_SSA_TREE_OPERAND (use, stmt, iter, SSA_OP_USE)
     {
-      if (SSA_NAME_IS_DEFAULT_DEF (use)
-         && ipa_get_param_decl_index (info, SSA_NAME_VAR (use)) >= 0)
+      tree parm = unmodified_parm (stmt, use);
+      if (parm && ipa_get_param_decl_index (info, parm) >= 0)
        p = add_condition (summary,
-                          ipa_get_param_decl_index (info, SSA_NAME_VAR (use)),
+                          ipa_get_param_decl_index (info, parm),
                           IS_NOT_CONSTANT, NULL);
       else
        p = *VEC_index (predicate_t, nonconstant_names,
                        SSA_NAME_VERSION (use));
-      op_non_const = or_predicates (&p, &op_non_const);
+      op_non_const = or_predicates (summary->conds, &p, &op_non_const);
     }
   if (gimple_code (stmt) == GIMPLE_ASSIGN
       && TREE_CODE (gimple_assign_lhs (stmt)) == SSA_NAME)
@@ -1563,7 +1633,7 @@ estimate_function_body_sizes (struct cgr
                fprintf (dump_file, "\t\twill eliminated by inlining\n");
 
              if (parms_info)
-               p = and_predicates (&bb_predicate, &will_be_nonconstant);
+               p = and_predicates (info->conds, &bb_predicate, 
&will_be_nonconstant);
              else
                p = true_predicate ();
 
@@ -1575,7 +1645,7 @@ estimate_function_body_sizes (struct cgr
                  if (prob)
                    {
                      struct predicate ip = not_inlined_predicate ();
-                     ip = and_predicates (&ip, &p);
+                     ip = and_predicates (info->conds, &ip, &p);
                      account_size_time (info, this_size * prob,
                                         this_time * prob, &ip);
                    }
@@ -1901,11 +1971,12 @@ remap_predicate (struct inline_summary *
                cond_predicate.clause[0] = 1 << cond;
                cond_predicate.clause[1] = 0;
              }
-           clause_predicate = or_predicates (&clause_predicate, 
&cond_predicate);
+           clause_predicate = or_predicates (info->conds, &clause_predicate,
+                                             &cond_predicate);
          }
-      out = and_predicates (&out, &clause_predicate);
+      out = and_predicates (info->conds, &out, &clause_predicate);
     }
-  return and_predicates (&out, toplev_predicate);
+  return and_predicates (info->conds, &out, toplev_predicate);
 }
 
 

Reply via email to