===================================================================
@@ -3447,6 +3447,19 @@ estimate_operator_cost (enum tree_code c
if (TREE_CODE (op2) != INTEGER_CST)
return weights->div_mod_cost;
return 1;
+ case ADDR_EXPR:
+ {
+ tree addr_base;
+ HOST_WIDE_INT addr_offset;
+
+ addr_base = get_addr_base_and_unit_offset (TREE_OPERAND (op1, 0),
+ &addr_offset);
+ /* If the offset is variable or with non-zero offset, return 2. */
+ if (!addr_base || addr_offset != 0 || TREE_CODE (addr_base) != MEM_REF
+ || !integer_zerop (TREE_OPERAND (addr_base, 1)))
+ return 1;
+ }
+ return 0;
default:
/* We expect a copy assignment with no operator. */
@@ -3512,14 +3525,24 @@ estimate_num_insns (gimple stmt, eni_wei
lhs = gimple_assign_lhs (stmt);
rhs = gimple_assign_rhs1 (stmt);
- if (is_gimple_reg (lhs))
- cost = 0;
- else
+ /* Store. */
+ if (gimple_vdef (stmt))
cost = estimate_move_cost (TREE_TYPE (lhs));
+ else
+ cost = 0;
- if (!is_gimple_reg (rhs) && !is_gimple_min_invariant (rhs))
+ /* Load. */
+ if (gimple_vuse (stmt))
cost += estimate_move_cost (TREE_TYPE (rhs));
+ /* Stores, loads and address expressions may have variable array
+ references in them. Account these. */
+ if (gimple_vuse (stmt))
+ cost += num_ssa_operands (stmt, SSA_OP_USE);
+ else if (gimple_vdef (stmt)
+ || gimple_assign_rhs_code (stmt) == ADDR_EXPR)
+ cost += MAX (0, (int)num_ssa_operands (stmt, SSA_OP_USE) - 1);
+
cost += estimate_operator_cost (gimple_assign_rhs_code (stmt), weights,
gimple_assign_rhs1 (stmt),
get_gimple_rhs_class (gimple_assign_rhs_code (stmt))
@@ -3596,6 +3619,7 @@ estimate_num_insns (gimple stmt, eni_wei
case GIMPLE_RETURN:
return weights->return_cost;
+ + MAX (0, (int)num_ssa_operands (stmt, SSA_OP_USE) - 1);
case GIMPLE_GOTO:
case GIMPLE_LABEL:
@@ -3606,7 +3630,10 @@ estimate_num_insns (gimple stmt, eni_wei
return 0;
case GIMPLE_ASM:
- return asm_str_count (gimple_asm_string (stmt));
+ return (asm_str_count (gimple_asm_string (stmt))
+ /* Account also possible non-trivial addr_exprs
+ in the arguments. */
+ + num_ssa_operands (stmt, SSA_OP_USE));
case GIMPLE_RESX:
/* This is either going to be an external function call with one