re PR middle-end/62263 (Good codegen for bitwise rotate requires code that is technically undefined behavior)

PR middle-end/62263
	PR middle-end/82498
	* tree-ssa-phiopt.c (value_replacement): Comment fix.  Handle
	up to 2 preparation statements for ASSIGN in MIDDLE_BB.

	* c-c++-common/rotate-8.c: Expect no PHIs in optimized dump.

From-SVN: r253761
This commit is contained in:
Jakub Jelinek 2017-10-14 20:48:38 +02:00 committed by Jakub Jelinek
parent cc453086d2
commit 14745bcac0
4 changed files with 118 additions and 7 deletions

View File

@ -1,5 +1,10 @@
2017-10-14 Jakub Jelinek <jakub@redhat.com> 2017-10-14 Jakub Jelinek <jakub@redhat.com>
PR middle-end/62263
PR middle-end/82498
* tree-ssa-phiopt.c (value_replacement): Comment fix. Handle
up to 2 preparation statements for ASSIGN in MIDDLE_BB.
PR middle-end/62263 PR middle-end/62263
PR middle-end/82498 PR middle-end/82498
* tree-ssa-forwprop.c (simplify_rotate): Allow def_arg1[N] * tree-ssa-forwprop.c (simplify_rotate): Allow def_arg1[N]

View File

@ -1,5 +1,9 @@
2017-10-14 Jakub Jelinek <jakub@redhat.com> 2017-10-14 Jakub Jelinek <jakub@redhat.com>
PR middle-end/62263
PR middle-end/82498
* c-c++-common/rotate-8.c: Expect no PHIs in optimized dump.
PR middle-end/62263 PR middle-end/62263
PR middle-end/82498 PR middle-end/82498
* c-c++-common/rotate-5.c (f2): New function. Move old * c-c++-common/rotate-5.c (f2): New function. Move old

View File

@ -3,6 +3,7 @@
/* { dg-do compile } */ /* { dg-do compile } */
/* { dg-options "-O2 -fno-ipa-icf -fdump-tree-optimized" } */ /* { dg-options "-O2 -fno-ipa-icf -fdump-tree-optimized" } */
/* { dg-final { scan-tree-dump-times "r\[<>]\[<>]" 23 "optimized" } } */ /* { dg-final { scan-tree-dump-times "r\[<>]\[<>]" 23 "optimized" } } */
/* { dg-final { scan-tree-dump-not "PHI <" "optimized" } } */
unsigned int unsigned int
f1 (unsigned int x, unsigned char y) f1 (unsigned int x, unsigned char y)

View File

@ -995,11 +995,13 @@ value_replacement (basic_block cond_bb, basic_block middle_bb,
} }
/* Now optimize (x != 0) ? x + y : y to just y. /* Now optimize (x != 0) ? x + y : y to just x + y. */
The following condition is too restrictive, there can easily be another gsi = gsi_last_nondebug_bb (middle_bb);
stmt in middle_bb, for instance a CONVERT_EXPR for the second argument. */ if (gsi_end_p (gsi))
gimple *assign = last_and_only_stmt (middle_bb); return 0;
if (!assign || gimple_code (assign) != GIMPLE_ASSIGN
gimple *assign = gsi_stmt (gsi);
if (!is_gimple_assign (assign)
|| gimple_assign_rhs_class (assign) != GIMPLE_BINARY_RHS || gimple_assign_rhs_class (assign) != GIMPLE_BINARY_RHS
|| (!INTEGRAL_TYPE_P (TREE_TYPE (arg0)) || (!INTEGRAL_TYPE_P (TREE_TYPE (arg0))
&& !POINTER_TYPE_P (TREE_TYPE (arg0)))) && !POINTER_TYPE_P (TREE_TYPE (arg0))))
@ -1009,6 +1011,71 @@ value_replacement (basic_block cond_bb, basic_block middle_bb,
if (!gimple_seq_empty_p (phi_nodes (middle_bb))) if (!gimple_seq_empty_p (phi_nodes (middle_bb)))
return 0; return 0;
/* Allow up to 2 cheap preparation statements that prepare argument
for assign, e.g.:
if (y_4 != 0)
goto <bb 3>;
else
goto <bb 4>;
<bb 3>:
_1 = (int) y_4;
iftmp.0_6 = x_5(D) r<< _1;
<bb 4>:
# iftmp.0_2 = PHI <iftmp.0_6(3), x_5(D)(2)>
or:
if (y_3(D) == 0)
goto <bb 4>;
else
goto <bb 3>;
<bb 3>:
y_4 = y_3(D) & 31;
_1 = (int) y_4;
_6 = x_5(D) r<< _1;
<bb 4>:
# _2 = PHI <x_5(D)(2), _6(3)> */
gimple *prep_stmt[2] = { NULL, NULL };
int prep_cnt;
for (prep_cnt = 0; ; prep_cnt++)
{
gsi_prev_nondebug (&gsi);
if (gsi_end_p (gsi))
break;
gimple *g = gsi_stmt (gsi);
if (gimple_code (g) == GIMPLE_LABEL)
break;
if (prep_cnt == 2 || !is_gimple_assign (g))
return 0;
tree lhs = gimple_assign_lhs (g);
tree rhs1 = gimple_assign_rhs1 (g);
use_operand_p use_p;
gimple *use_stmt;
if (TREE_CODE (lhs) != SSA_NAME
|| TREE_CODE (rhs1) != SSA_NAME
|| !INTEGRAL_TYPE_P (TREE_TYPE (lhs))
|| !INTEGRAL_TYPE_P (TREE_TYPE (rhs1))
|| !single_imm_use (lhs, &use_p, &use_stmt)
|| use_stmt != (prep_cnt ? prep_stmt[prep_cnt - 1] : assign))
return 0;
switch (gimple_assign_rhs_code (g))
{
CASE_CONVERT:
break;
case PLUS_EXPR:
case BIT_AND_EXPR:
case BIT_IOR_EXPR:
case BIT_XOR_EXPR:
if (TREE_CODE (gimple_assign_rhs2 (g)) != INTEGER_CST)
return 0;
break;
default:
return 0;
}
prep_stmt[prep_cnt] = g;
}
/* Only transform if it removes the condition. */ /* Only transform if it removes the condition. */
if (!single_non_singleton_phi_for_edges (phi_nodes (gimple_bb (phi)), e0, e1)) if (!single_non_singleton_phi_for_edges (phi_nodes (gimple_bb (phi)), e0, e1))
return 0; return 0;
@ -1019,7 +1086,7 @@ value_replacement (basic_block cond_bb, basic_block middle_bb,
&& profile_status_for_fn (cfun) != PROFILE_ABSENT && profile_status_for_fn (cfun) != PROFILE_ABSENT
&& EDGE_PRED (middle_bb, 0)->probability < profile_probability::even () && EDGE_PRED (middle_bb, 0)->probability < profile_probability::even ()
/* If assign is cheap, there is no point avoiding it. */ /* If assign is cheap, there is no point avoiding it. */
&& estimate_num_insns (assign, &eni_time_weights) && estimate_num_insns (bb_seq (middle_bb), &eni_time_weights)
>= 3 * estimate_num_insns (cond, &eni_time_weights)) >= 3 * estimate_num_insns (cond, &eni_time_weights))
return 0; return 0;
@ -1030,6 +1097,32 @@ value_replacement (basic_block cond_bb, basic_block middle_bb,
tree cond_lhs = gimple_cond_lhs (cond); tree cond_lhs = gimple_cond_lhs (cond);
tree cond_rhs = gimple_cond_rhs (cond); tree cond_rhs = gimple_cond_rhs (cond);
/* Propagate the cond_rhs constant through preparation stmts,
make sure UB isn't invoked while doing that. */
for (int i = prep_cnt - 1; i >= 0; --i)
{
gimple *g = prep_stmt[i];
tree grhs1 = gimple_assign_rhs1 (g);
if (!operand_equal_for_phi_arg_p (cond_lhs, grhs1))
return 0;
cond_lhs = gimple_assign_lhs (g);
cond_rhs = fold_convert (TREE_TYPE (grhs1), cond_rhs);
if (TREE_CODE (cond_rhs) != INTEGER_CST
|| TREE_OVERFLOW (cond_rhs))
return 0;
if (gimple_assign_rhs_class (g) == GIMPLE_BINARY_RHS)
{
cond_rhs = int_const_binop (gimple_assign_rhs_code (g), cond_rhs,
gimple_assign_rhs2 (g));
if (TREE_OVERFLOW (cond_rhs))
return 0;
}
cond_rhs = fold_convert (TREE_TYPE (cond_lhs), cond_rhs);
if (TREE_CODE (cond_rhs) != INTEGER_CST
|| TREE_OVERFLOW (cond_rhs))
return 0;
}
if (((code == NE_EXPR && e1 == false_edge) if (((code == NE_EXPR && e1 == false_edge)
|| (code == EQ_EXPR && e1 == true_edge)) || (code == EQ_EXPR && e1 == true_edge))
&& arg0 == lhs && arg0 == lhs
@ -1071,7 +1164,15 @@ value_replacement (basic_block cond_bb, basic_block middle_bb,
duplicate_ssa_name_range_info (lhs, SSA_NAME_RANGE_TYPE (phires), duplicate_ssa_name_range_info (lhs, SSA_NAME_RANGE_TYPE (phires),
phires_range_info); phires_range_info);
} }
gimple_stmt_iterator gsi_from = gsi_for_stmt (assign); gimple_stmt_iterator gsi_from;
for (int i = prep_cnt - 1; i >= 0; --i)
{
tree plhs = gimple_assign_lhs (prep_stmt[i]);
SSA_NAME_RANGE_INFO (plhs) = NULL;
gsi_from = gsi_for_stmt (prep_stmt[i]);
gsi_move_before (&gsi_from, &gsi);
}
gsi_from = gsi_for_stmt (assign);
gsi_move_before (&gsi_from, &gsi); gsi_move_before (&gsi_from, &gsi);
replace_phi_edge_with_variable (cond_bb, e1, phi, lhs); replace_phi_edge_with_variable (cond_bb, e1, phi, lhs);
return 2; return 2;