Commit 1addb9e6 by Patrick Palka

re PR tree-optimization/72810 (error: type mismatch for case label in switch statement)

Fix PR tree-optimization/72810

gcc/ChangeLog:

	PR tree-optimization/72810
	* tree-vrp.c (simplify_switch_using_ranges): Avoid changing
	the type of the case labels when truncating.

gcc/testsuite/ChangeLog:

	PR tree-optimization/72810
	* gcc.dg/tree-ssa/vrp110.c: New test.

From-SVN: r239174
parent 1b62ed4f
2016-08-05 Patrick Palka <ppalka@gcc.gnu.org>
PR tree-optimization/72810
* tree-vrp.c (simplify_switch_using_ranges): Avoid changing
the type of the case labels when truncating.
2016-08-05 James Greenhalgh <james.greenhalgh@arm.com> 2016-08-05 James Greenhalgh <james.greenhalgh@arm.com>
PR Target/72819 PR Target/72819
......
2016-08-05 Patrick Palka <ppalka@gcc.gnu.org>
PR tree-optimization/72810
* gcc.dg/tree-ssa/vrp110.c: New test.
2016-08-05 James Greenhalgh <james.greenhalgh@arm.com> 2016-08-05 James Greenhalgh <james.greenhalgh@arm.com>
PR Target/72819 PR Target/72819
......
/* { dg-options "-O2" } */
extern void foo (void);
extern void bar (void);
void
test (int i)
{
if (i == 1)
return;
typedef int t;
t j = i;
switch (j)
{
case 1:
case 2:
foo ();
break;
case 7:
bar ();
break;
}
}
...@@ -9644,61 +9644,66 @@ simplify_switch_using_ranges (gswitch *stmt) ...@@ -9644,61 +9644,66 @@ simplify_switch_using_ranges (gswitch *stmt)
tree min_label = gimple_switch_label (stmt, min_idx); tree min_label = gimple_switch_label (stmt, min_idx);
tree max_label = gimple_switch_label (stmt, max_idx); tree max_label = gimple_switch_label (stmt, max_idx);
/* Avoid changing the type of the case labels when truncating. */
tree case_label_type = TREE_TYPE (CASE_LOW (min_label));
tree vr_min = fold_convert (case_label_type, vr->min);
tree vr_max = fold_convert (case_label_type, vr->max);
if (vr->type == VR_RANGE) if (vr->type == VR_RANGE)
{ {
/* If OP's value range is [2,8] and the low label range is /* If OP's value range is [2,8] and the low label range is
0 ... 3, truncate the label's range to 2 .. 3. */ 0 ... 3, truncate the label's range to 2 .. 3. */
if (tree_int_cst_compare (CASE_LOW (min_label), vr->min) < 0 if (tree_int_cst_compare (CASE_LOW (min_label), vr_min) < 0
&& CASE_HIGH (min_label) != NULL_TREE && CASE_HIGH (min_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (min_label), vr->min) >= 0) && tree_int_cst_compare (CASE_HIGH (min_label), vr_min) >= 0)
CASE_LOW (min_label) = vr->min; CASE_LOW (min_label) = vr_min;
/* If OP's value range is [2,8] and the high label range is /* If OP's value range is [2,8] and the high label range is
7 ... 10, truncate the label's range to 7 .. 8. */ 7 ... 10, truncate the label's range to 7 .. 8. */
if (tree_int_cst_compare (CASE_LOW (max_label), vr->max) <= 0 if (tree_int_cst_compare (CASE_LOW (max_label), vr_max) <= 0
&& CASE_HIGH (max_label) != NULL_TREE && CASE_HIGH (max_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (max_label), vr->max) > 0) && tree_int_cst_compare (CASE_HIGH (max_label), vr_max) > 0)
CASE_HIGH (max_label) = vr->max; CASE_HIGH (max_label) = vr_max;
} }
else if (vr->type == VR_ANTI_RANGE) else if (vr->type == VR_ANTI_RANGE)
{ {
tree one_cst = build_one_cst (TREE_TYPE (op)); tree one_cst = build_one_cst (case_label_type);
if (min_label == max_label) if (min_label == max_label)
{ {
/* If OP's value range is ~[7,8] and the label's range is /* If OP's value range is ~[7,8] and the label's range is
7 ... 10, truncate the label's range to 9 ... 10. */ 7 ... 10, truncate the label's range to 9 ... 10. */
if (tree_int_cst_compare (CASE_LOW (min_label), vr->min) == 0 if (tree_int_cst_compare (CASE_LOW (min_label), vr_min) == 0
&& CASE_HIGH (min_label) != NULL_TREE && CASE_HIGH (min_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (min_label), vr->max) > 0) && tree_int_cst_compare (CASE_HIGH (min_label), vr_max) > 0)
CASE_LOW (min_label) CASE_LOW (min_label)
= int_const_binop (PLUS_EXPR, vr->max, one_cst); = int_const_binop (PLUS_EXPR, vr_max, one_cst);
/* If OP's value range is ~[7,8] and the label's range is /* If OP's value range is ~[7,8] and the label's range is
5 ... 8, truncate the label's range to 5 ... 6. */ 5 ... 8, truncate the label's range to 5 ... 6. */
if (tree_int_cst_compare (CASE_LOW (min_label), vr->min) < 0 if (tree_int_cst_compare (CASE_LOW (min_label), vr_min) < 0
&& CASE_HIGH (min_label) != NULL_TREE && CASE_HIGH (min_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (min_label), vr->max) == 0) && tree_int_cst_compare (CASE_HIGH (min_label), vr_max) == 0)
CASE_HIGH (min_label) CASE_HIGH (min_label)
= int_const_binop (MINUS_EXPR, vr->min, one_cst); = int_const_binop (MINUS_EXPR, vr_min, one_cst);
} }
else else
{ {
/* If OP's value range is ~[2,8] and the low label range is /* If OP's value range is ~[2,8] and the low label range is
0 ... 3, truncate the label's range to 0 ... 1. */ 0 ... 3, truncate the label's range to 0 ... 1. */
if (tree_int_cst_compare (CASE_LOW (min_label), vr->min) < 0 if (tree_int_cst_compare (CASE_LOW (min_label), vr_min) < 0
&& CASE_HIGH (min_label) != NULL_TREE && CASE_HIGH (min_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (min_label), vr->min) >= 0) && tree_int_cst_compare (CASE_HIGH (min_label), vr_min) >= 0)
CASE_HIGH (min_label) CASE_HIGH (min_label)
= int_const_binop (MINUS_EXPR, vr->min, one_cst); = int_const_binop (MINUS_EXPR, vr_min, one_cst);
/* If OP's value range is ~[2,8] and the high label range is /* If OP's value range is ~[2,8] and the high label range is
7 ... 10, truncate the label's range to 9 ... 10. */ 7 ... 10, truncate the label's range to 9 ... 10. */
if (tree_int_cst_compare (CASE_LOW (max_label), vr->max) <= 0 if (tree_int_cst_compare (CASE_LOW (max_label), vr_max) <= 0
&& CASE_HIGH (max_label) != NULL_TREE && CASE_HIGH (max_label) != NULL_TREE
&& tree_int_cst_compare (CASE_HIGH (max_label), vr->max) > 0) && tree_int_cst_compare (CASE_HIGH (max_label), vr_max) > 0)
CASE_LOW (max_label) CASE_LOW (max_label)
= int_const_binop (PLUS_EXPR, vr->max, one_cst); = int_const_binop (PLUS_EXPR, vr_max, one_cst);
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment