diff options
author | spop <spop@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-08-13 17:28:43 +0000 |
---|---|---|
committer | spop <spop@138bc75d-0d04-0410-961f-82ee72b054a4> | 2005-08-13 17:28:43 +0000 |
commit | 903dae48bdeed6191d043cd56f2345d332916ee4 (patch) | |
tree | e16e6f08588cfad8cd3b6712bf30a607ac5f488d /gcc | |
parent | 77beec480bdbed142d45a84de7bf61f265bdf3a2 (diff) | |
download | gcc-903dae48bdeed6191d043cd56f2345d332916ee4.tar.gz |
PR tree-optimization/22236
* tree-cfg.c (print_pred_bbs, print_succ_bbs): Correctly print
successors and predecessors.
* tree-chrec.c (chrec_convert): Before converting, check that
sequences don't wrap.
* tree-data-ref.c (compute_estimated_nb_iterations): Moved ...
(analyze_array): Extern.
(find_data_references_in_loop): Remove call to
compute_estimated_nb_iterations.
* tree-data-ref.h (analyze_array): Declared.
* tree-flow-inline.h (single_ssa_tree_operand, single_ssa_use_operand,
single_ssa_def_operand, zero_ssa_operands): Fix documentation.
* tree-flow.h (scev_probably_wraps_p): Declare with an extra parameter.
* tree-scalar-evolution.c (instantiate_parameters_1): Factor entry
condition.
* tree-ssa-loop-ivcanon.c: Fix documentation.
* tree-ssa-loop-ivopts.c (idx_find_step): Add a fixme note.
* tree-ssa-loop-niter.c (compute_estimated_nb_iterations): ... here.
(infer_loop_bounds_from_undefined): New.
(estimate_numbers_of_iterations_loop): Use
infer_loop_bounds_from_undefined.
(used_in_pointer_arithmetic_p): New.
(scev_probably_wraps_p): Pass an extra parameter. Call
used_in_pointer_arithmetic_p. Check that AT_STMT is not null.
(convert_step): Fix documentation.
* tree-vrp.c (adjust_range_with_scev): Call instantiate_parameters.
Use initial_condition_in_loop_num and evolution_part_in_loop_num
instead of CHREC_LEFT and CHREC_RIGHT. Adjust the call to
scev_probably_wraps_p.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@103055 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc')
-rw-r--r-- | gcc/ChangeLog | 32 | ||||
-rw-r--r-- | gcc/testsuite/gcc.dg/tree-ssa/pr22236.c | 33 | ||||
-rw-r--r-- | gcc/tree-cfg.c | 8 | ||||
-rw-r--r-- | gcc/tree-chrec.c | 21 | ||||
-rw-r--r-- | gcc/tree-data-ref.c | 22 | ||||
-rw-r--r-- | gcc/tree-data-ref.h | 1 | ||||
-rw-r--r-- | gcc/tree-flow-inline.h | 8 | ||||
-rw-r--r-- | gcc/tree-flow.h | 3 | ||||
-rw-r--r-- | gcc/tree-scalar-evolution.c | 7 | ||||
-rw-r--r-- | gcc/tree-ssa-loop-ivcanon.c | 2 | ||||
-rw-r--r-- | gcc/tree-ssa-loop-ivopts.c | 2 | ||||
-rw-r--r-- | gcc/tree-ssa-loop-niter.c | 228 | ||||
-rw-r--r-- | gcc/tree-vect-analyze.c | 2 | ||||
-rw-r--r-- | gcc/tree-vrp.c | 14 |
14 files changed, 323 insertions, 60 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 3776860daf6..456967757d4 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,35 @@ +2005-08-13 Sebastian Pop <pop@cri.ensmp.fr> + + PR tree-optimization/22236 + * tree-cfg.c (print_pred_bbs, print_succ_bbs): Correctly print + successors and predecessors. + * tree-chrec.c (chrec_convert): Before converting, check that + sequences don't wrap. + * tree-data-ref.c (compute_estimated_nb_iterations): Moved ... + (analyze_array): Extern. + (find_data_references_in_loop): Remove call to + compute_estimated_nb_iterations. + * tree-data-ref.h (analyze_array): Declared. + * tree-flow-inline.h (single_ssa_tree_operand, single_ssa_use_operand, + single_ssa_def_operand, zero_ssa_operands): Fix documentation. + * tree-flow.h (scev_probably_wraps_p): Declare with an extra parameter. + * tree-scalar-evolution.c (instantiate_parameters_1): Factor entry + condition. + * tree-ssa-loop-ivcanon.c: Fix documentation. + * tree-ssa-loop-ivopts.c (idx_find_step): Add a fixme note. + * tree-ssa-loop-niter.c (compute_estimated_nb_iterations): ... here. + (infer_loop_bounds_from_undefined): New. + (estimate_numbers_of_iterations_loop): Use + infer_loop_bounds_from_undefined. + (used_in_pointer_arithmetic_p): New. + (scev_probably_wraps_p): Pass an extra parameter. Call + used_in_pointer_arithmetic_p. Check that AT_STMT is not null. + (convert_step): Fix documentation. + * tree-vrp.c (adjust_range_with_scev): Call instantiate_parameters. + Use initial_condition_in_loop_num and evolution_part_in_loop_num + instead of CHREC_LEFT and CHREC_RIGHT. Adjust the call to + scev_probably_wraps_p. + 2005-08-13 Ulrich Weigand <uweigand@de.ibm.com> * config/s390/s390.c (s390_split_branches): Revert 2005-08-12 change. diff --git a/gcc/testsuite/gcc.dg/tree-ssa/pr22236.c b/gcc/testsuite/gcc.dg/tree-ssa/pr22236.c new file mode 100644 index 00000000000..1986c2f0cfe --- /dev/null +++ b/gcc/testsuite/gcc.dg/tree-ssa/pr22236.c @@ -0,0 +1,33 @@ +/* { dg-do run } */ +/* { dg-options "-O1 -fno-tree-vrp -fwrapv" } */ + +/* PR tree-optimization/22236 + + Avoid conversion of (signed char) {(uchar)1, +, (uchar)1}_x when + it is not possible to prove that the scev does not wrap. + + In this PR, a sequence 1, 2, ..., 255 has to be converted to + signed char, but this would wrap: 1, 2, ..., 127, -128, ... The + result should not be a linear scev {(schar)1, +, (schar)1}_x. + The conversion should be kept: (schar) {(uchar)1, +, (uchar)1}_x. + */ + +void abort(void); + +static inline void +foo (signed char a) +{ + int b = a - 0x7F; + if (b > 1) + abort(); +} + +int main() +{ + unsigned char b; + for(b = 0; b < 0xFF; b++) + foo (b); + + return 0; +} + diff --git a/gcc/tree-cfg.c b/gcc/tree-cfg.c index 70dca21bc83..ad399fbce64 100644 --- a/gcc/tree-cfg.c +++ b/gcc/tree-cfg.c @@ -4502,7 +4502,7 @@ static void print_pred_bbs (FILE *, basic_block bb); static void print_succ_bbs (FILE *, basic_block bb); -/* Print the predecessors indexes of edge E on FILE. */ +/* Print on FILE the indexes for the predecessors of basic_block BB. */ static void print_pred_bbs (FILE *file, basic_block bb) @@ -4511,11 +4511,11 @@ print_pred_bbs (FILE *file, basic_block bb) edge_iterator ei; FOR_EACH_EDGE (e, ei, bb->preds) - fprintf (file, "bb_%d", e->src->index); + fprintf (file, "bb_%d ", e->src->index); } -/* Print the successors indexes of edge E on FILE. */ +/* Print on FILE the indexes for the successors of basic_block BB. */ static void print_succ_bbs (FILE *file, basic_block bb) @@ -4524,7 +4524,7 @@ print_succ_bbs (FILE *file, basic_block bb) edge_iterator ei; FOR_EACH_EDGE (e, ei, bb->succs) - fprintf (file, "bb_%d", e->src->index); + fprintf (file, "bb_%d ", e->dest->index); } diff --git a/gcc/tree-chrec.c b/gcc/tree-chrec.c index b48813c8daf..1a229225f80 100644 --- a/gcc/tree-chrec.c +++ b/gcc/tree-chrec.c @@ -1110,9 +1110,24 @@ chrec_convert (tree type, tree chrec, tree at_stmt) if (evolution_function_is_affine_p (chrec)) { - tree step = convert_step (current_loops->parray[CHREC_VARIABLE (chrec)], - type, CHREC_LEFT (chrec), CHREC_RIGHT (chrec), - at_stmt); + tree step; + bool dummy; + + /* Avoid conversion of (signed char) {(uchar)1, +, (uchar)1}_x + when it is not possible to prove that the scev does not wrap. + See PR22236, where a sequence 1, 2, ..., 255 has to be + converted to signed char, but this would wrap: + 1, 2, ..., 127, -128, ... The result should not be + {(schar)1, +, (schar)1}_x, but instead, we should keep the + conversion: (schar) {(uchar)1, +, (uchar)1}_x. */ + if (scev_probably_wraps_p (type, CHREC_LEFT (chrec), CHREC_RIGHT (chrec), + at_stmt, + current_loops->parray[CHREC_VARIABLE (chrec)], + &dummy, &dummy)) + return fold_convert (type, chrec); + + step = convert_step (current_loops->parray[CHREC_VARIABLE (chrec)], type, + CHREC_LEFT (chrec), CHREC_RIGHT (chrec), at_stmt); if (!step) return fold_convert (type, chrec); diff --git a/gcc/tree-data-ref.c b/gcc/tree-data-ref.c index 05361a2fc16..7fc64242bf3 100644 --- a/gcc/tree-data-ref.c +++ b/gcc/tree-data-ref.c @@ -731,23 +731,6 @@ dump_ddrs (FILE *file, varray_type ddrs) -/* Initialize LOOP->ESTIMATED_NB_ITERATIONS with the lowest safe - approximation of the number of iterations for LOOP. */ - -static void -compute_estimated_nb_iterations (struct loop *loop) -{ - struct nb_iter_bound *bound; - - for (bound = loop->bounds; bound; bound = bound->next) - if (TREE_CODE (bound->bound) == INTEGER_CST - /* Update only when there is no previous estimation. */ - && (chrec_contains_undetermined (loop->estimated_nb_iterations) - /* Or when the current estimation is smaller. */ - || tree_int_cst_lt (bound->bound, loop->estimated_nb_iterations))) - loop->estimated_nb_iterations = bound->bound; -} - /* Estimate the number of iterations from the size of the data and the access functions. */ @@ -830,7 +813,7 @@ analyze_array_indexes (struct loop *loop, set to true when REF is in the right hand side of an assignment. */ -static struct data_reference * +struct data_reference * analyze_array (tree stmt, tree ref, bool is_read) { struct data_reference *res; @@ -3644,9 +3627,6 @@ find_data_references_in_loop (struct loop *loop, varray_type *datarefs) if (!ZERO_SSA_OPERANDS (stmt, SSA_OP_VIRTUAL_DEFS)) loop->parallel_p = false; } - - if (chrec_contains_undetermined (loop->estimated_nb_iterations)) - compute_estimated_nb_iterations (loop); } free (bbs); diff --git a/gcc/tree-data-ref.h b/gcc/tree-data-ref.h index d0e670149d3..f845fcd4dcf 100644 --- a/gcc/tree-data-ref.h +++ b/gcc/tree-data-ref.h @@ -264,6 +264,7 @@ extern void free_dependence_relation (struct data_dependence_relation *); extern void free_dependence_relations (varray_type); extern void free_data_refs (varray_type); extern void compute_subscript_distance (struct data_dependence_relation *); +extern struct data_reference *analyze_array (tree, tree, bool); diff --git a/gcc/tree-flow-inline.h b/gcc/tree-flow-inline.h index 299e1b51240..b0f5e888551 100644 --- a/gcc/tree-flow-inline.h +++ b/gcc/tree-flow-inline.h @@ -1191,7 +1191,7 @@ op_iter_init_must_and_may_def (ssa_op_iter *ptr, tree stmt, /* If there is a single operand in STMT matching FLAGS, return it. Otherwise - return NULL. PTR is the iterator to use. */ + return NULL. */ static inline tree single_ssa_tree_operand (tree stmt, int flags) { @@ -1209,7 +1209,7 @@ single_ssa_tree_operand (tree stmt, int flags) /* If there is a single operand in STMT matching FLAGS, return it. Otherwise - return NULL. PTR is the iterator to use. */ + return NULL. */ static inline use_operand_p single_ssa_use_operand (tree stmt, int flags) { @@ -1228,7 +1228,7 @@ single_ssa_use_operand (tree stmt, int flags) /* If there is a single operand in STMT matching FLAGS, return it. Otherwise - return NULL. PTR is the iterator to use. */ + return NULL. */ static inline def_operand_p single_ssa_def_operand (tree stmt, int flags) { @@ -1246,7 +1246,7 @@ single_ssa_def_operand (tree stmt, int flags) /* If there is a single operand in STMT matching FLAGS, return it. Otherwise - return NULL. PTR is the iterator to use. */ + return NULL. */ static inline bool zero_ssa_operands (tree stmt, int flags) { diff --git a/gcc/tree-flow.h b/gcc/tree-flow.h index bfbceba9f27..fa5c1d692df 100644 --- a/gcc/tree-flow.h +++ b/gcc/tree-flow.h @@ -728,7 +728,8 @@ tree find_loop_niter (struct loop *, edge *); tree loop_niter_by_eval (struct loop *, edge); tree find_loop_niter_by_eval (struct loop *, edge *); void estimate_numbers_of_iterations (struct loops *); -bool scev_probably_wraps_p (tree, tree, tree, tree, struct loop *, bool *); +bool scev_probably_wraps_p (tree, tree, tree, tree, struct loop *, bool *, + bool *); tree convert_step (struct loop *, tree, tree, tree, tree); void free_numbers_of_iterations_estimates (struct loops *); void rewrite_into_loop_closed_ssa (bitmap, unsigned); diff --git a/gcc/tree-scalar-evolution.c b/gcc/tree-scalar-evolution.c index 65348d08402..db798c025e2 100644 --- a/gcc/tree-scalar-evolution.c +++ b/gcc/tree-scalar-evolution.c @@ -1939,11 +1939,8 @@ instantiate_parameters_1 (struct loop *loop, tree chrec, basic_block def_bb; struct loop *def_loop; - if (chrec == NULL_TREE - || automatically_generated_chrec_p (chrec)) - return chrec; - - if (is_gimple_min_invariant (chrec)) + if (automatically_generated_chrec_p (chrec) + || is_gimple_min_invariant (chrec)) return chrec; switch (TREE_CODE (chrec)) diff --git a/gcc/tree-ssa-loop-ivcanon.c b/gcc/tree-ssa-loop-ivcanon.c index 4ac20039f4c..1788c33e037 100644 --- a/gcc/tree-ssa-loop-ivcanon.c +++ b/gcc/tree-ssa-loop-ivcanon.c @@ -267,7 +267,7 @@ try_unroll_loop_completely (struct loops *loops ATTRIBUTE_UNUSED, } /* Adds a canonical induction variable to LOOP if suitable. LOOPS is the loops - tree. CREATE_IV is true if we may create a new iv. UL determines what + tree. CREATE_IV is true if we may create a new iv. UL determines which loops we are allowed to completely unroll. If TRY_EVAL is true, we try to determine the number of iterations of a loop by direct evaluation. Returns true if cfg is changed. */ diff --git a/gcc/tree-ssa-loop-ivopts.c b/gcc/tree-ssa-loop-ivopts.c index e5a8d85504d..d0a186a166a 100644 --- a/gcc/tree-ssa-loop-ivopts.c +++ b/gcc/tree-ssa-loop-ivopts.c @@ -1443,6 +1443,8 @@ idx_find_step (tree base, tree *idx, void *data) /* The step for pointer arithmetics already is 1 byte. */ step = build_int_cst (sizetype, 1); + /* FIXME: convert_step should not be used outside chrec_convert: fix + this by calling chrec_convert. */ iv_step = convert_step (dta->ivopts_data->current_loop, sizetype, iv->base, iv->step, dta->stmt); diff --git a/gcc/tree-ssa-loop-niter.c b/gcc/tree-ssa-loop-niter.c index 21dc9eca8ae..8399faf30e2 100644 --- a/gcc/tree-ssa-loop-niter.c +++ b/gcc/tree-ssa-loop-niter.c @@ -1381,6 +1381,128 @@ record_estimate (struct loop *loop, tree bound, tree additional, tree at_stmt) loop->bounds = elt; } +/* Initialize LOOP->ESTIMATED_NB_ITERATIONS with the lowest safe + approximation of the number of iterations for LOOP. */ + +static void +compute_estimated_nb_iterations (struct loop *loop) +{ + struct nb_iter_bound *bound; + + for (bound = loop->bounds; bound; bound = bound->next) + if (TREE_CODE (bound->bound) == INTEGER_CST + /* Update only when there is no previous estimation. */ + && (chrec_contains_undetermined (loop->estimated_nb_iterations) + /* Or when the current estimation is smaller. */ + || tree_int_cst_lt (bound->bound, loop->estimated_nb_iterations))) + loop->estimated_nb_iterations = bound->bound; +} + +/* The following analyzers are extracting informations on the bounds + of LOOP from the following undefined behaviors: + + - data references should not access elements over the statically + allocated size, + + - signed variables should not overflow when flag_wrapv is not set. +*/ + +static void +infer_loop_bounds_from_undefined (struct loop *loop) +{ + unsigned i; + basic_block bb, *bbs; + block_stmt_iterator bsi; + + bbs = get_loop_body (loop); + + for (i = 0; i < loop->num_nodes; i++) + { + bb = bbs[i]; + + for (bsi = bsi_start (bb); !bsi_end_p (bsi); bsi_next (&bsi)) + { + tree stmt = bsi_stmt (bsi); + + switch (TREE_CODE (stmt)) + { + case MODIFY_EXPR: + { + tree op0 = TREE_OPERAND (stmt, 0); + tree op1 = TREE_OPERAND (stmt, 1); + + /* For each array access, analyze its access function + and record a bound on the loop iteration domain. */ + if (TREE_CODE (op1) == ARRAY_REF) + analyze_array (stmt, op1, true); + + if (TREE_CODE (op0) == ARRAY_REF) + analyze_array (stmt, op0, false); + + /* For each signed type variable in LOOP, analyze its + scalar evolution and record a bound of the loop + based on the type's ranges. */ + else if (!flag_wrapv && TREE_CODE (op0) == SSA_NAME) + { + tree init, step, diff, estimation; + tree scev = instantiate_parameters + (loop, analyze_scalar_evolution (loop, op0)); + tree type = chrec_type (scev); + tree utype; + + if (chrec_contains_undetermined (scev) + || TYPE_UNSIGNED (type)) + break; + + init = initial_condition_in_loop_num (scev, loop->num); + step = evolution_part_in_loop_num (scev, loop->num); + + if (init == NULL_TREE + || step == NULL_TREE + || TREE_CODE (init) != INTEGER_CST + || TREE_CODE (step) != INTEGER_CST) + break; + + utype = unsigned_type_for (type); + if (tree_int_cst_lt (step, integer_zero_node)) + diff = fold (build2 (MINUS_EXPR, utype, init, + TYPE_MIN_VALUE (type))); + else + diff = fold (build2 (MINUS_EXPR, utype, + TYPE_MAX_VALUE (type), init)); + + estimation = fold (build2 (CEIL_DIV_EXPR, utype, diff, + step)); + record_estimate (loop, estimation, boolean_true_node, stmt); + } + + break; + } + + case CALL_EXPR: + { + tree args; + + for (args = TREE_OPERAND (stmt, 1); args; + args = TREE_CHAIN (args)) + if (TREE_CODE (TREE_VALUE (args)) == ARRAY_REF) + analyze_array (stmt, TREE_VALUE (args), true); + + break; + } + + default: + break; + } + } + + if (chrec_contains_undetermined (loop->estimated_nb_iterations)) + compute_estimated_nb_iterations (loop); + } + + free (bbs); +} + /* Records estimates on numbers of iterations of LOOP. */ static void @@ -1419,14 +1541,8 @@ estimate_numbers_of_iterations_loop (struct loop *loop) } free (exits); - /* Analyzes the bounds of arrays accessed in the loop. */ if (chrec_contains_undetermined (loop->estimated_nb_iterations)) - { - varray_type datarefs; - VARRAY_GENERIC_PTR_INIT (datarefs, 3, "datarefs"); - find_data_references_in_loop (loop, &datarefs); - free_data_refs (datarefs); - } + infer_loop_bounds_from_undefined (loop); } /* Records estimates on numbers of iterations of LOOPS. */ @@ -1645,6 +1761,43 @@ convert_step_widening (struct loop *loop, tree new_type, tree base, tree step, return NULL_TREE; } +/* Returns true when VAR is used in pointer arithmetics. DEPTH is + used for limiting the search. */ + +static bool +used_in_pointer_arithmetic_p (tree var, int depth) +{ + use_operand_p use_p; + imm_use_iterator iter; + + if (depth == 0 + || TREE_CODE (var) != SSA_NAME + || !has_single_use (var)) + return false; + + FOR_EACH_IMM_USE_FAST (use_p, iter, var) + { + tree stmt = USE_STMT (use_p); + + if (stmt && TREE_CODE (stmt) == MODIFY_EXPR) + { + tree rhs = TREE_OPERAND (stmt, 1); + + if (TREE_CODE (rhs) == NOP_EXPR + || TREE_CODE (rhs) == CONVERT_EXPR) + { + if (POINTER_TYPE_P (TREE_TYPE (rhs))) + return true; + return false; + } + else + return used_in_pointer_arithmetic_p (TREE_OPERAND (stmt, 0), + depth - 1); + } + } + return false; +} + /* Return false only when the induction variable BASE + STEP * I is known to not overflow: i.e. when the number of iterations is small enough with respect to the step and initial condition in order to @@ -1652,19 +1805,60 @@ convert_step_widening (struct loop *loop, tree new_type, tree base, tree step, iv is known to overflow or when the property is not computable. Initialize INIT_IS_MAX to true when the evolution goes from - INIT_IS_MAX to LOWER_BOUND_IN_TYPE, false in the contrary case, not - defined when the function returns true. */ + INIT_IS_MAX to LOWER_BOUND_IN_TYPE, false in the contrary case. + When this property cannot be determined, UNKNOWN_MAX is set to + true. */ bool scev_probably_wraps_p (tree type, tree base, tree step, tree at_stmt, struct loop *loop, - bool *init_is_max) + bool *init_is_max, bool *unknown_max) { struct nb_iter_bound *bound; tree delta, step_abs; tree unsigned_type, valid_niter; - tree base_plus_step = fold_build2 (PLUS_EXPR, type, base, step); + tree base_plus_step; + + /* FIXME: The following code will not be used anymore once + http://gcc.gnu.org/ml/gcc-patches/2005-06/msg02025.html is + committed. + + If AT_STMT is a cast to unsigned that is later used for + referencing a memory location, it is followed by a pointer + conversion just after. Because pointers do not wrap, the + sequences that reference the memory do not wrap either. In the + following example, sequences corresponding to D_13 and to D_14 + can be proved to not wrap because they are used for computing a + memory access: + + D.1621_13 = (long unsigned intD.4) D.1620_12; + D.1622_14 = D.1621_13 * 8; + D.1623_15 = (doubleD.29 *) D.1622_14; + */ + if (at_stmt && TREE_CODE (at_stmt) == MODIFY_EXPR) + { + tree op0 = TREE_OPERAND (at_stmt, 0); + tree op1 = TREE_OPERAND (at_stmt, 1); + tree type_op1 = TREE_TYPE (op1); + + if ((TYPE_UNSIGNED (type_op1) + && used_in_pointer_arithmetic_p (op0, 2)) + || POINTER_TYPE_P (type_op1)) + { + *unknown_max = true; + return false; + } + } + if (TREE_CODE (base) == REAL_CST + || TREE_CODE (step) == REAL_CST) + { + *unknown_max = true; + return true; + } + + *unknown_max = false; + base_plus_step = fold_build2 (PLUS_EXPR, type, base, step); switch (compare_trees (base_plus_step, base)) { case -1: @@ -1691,6 +1885,7 @@ scev_probably_wraps_p (tree type, tree base, tree step, don't know as in the default case. */ default: + *unknown_max = true; return true; } @@ -1709,7 +1904,7 @@ scev_probably_wraps_p (tree type, tree base, tree step, i_2 to wrap around, but not i.0_6, because it is of a signed type. This causes VRP to erroneously fold the predicate above because it thinks that i.0_6 cannot be negative. */ - if (TREE_CODE (at_stmt) == MODIFY_EXPR) + if (at_stmt && TREE_CODE (at_stmt) == MODIFY_EXPR) { tree rhs = TREE_OPERAND (at_stmt, 1); tree outer_t = TREE_TYPE (rhs); @@ -1725,7 +1920,10 @@ scev_probably_wraps_p (tree type, tree base, tree step, if (TYPE_UNSIGNED (inner_t) && (TYPE_SIZE (inner_t) <= TYPE_SIZE (outer_t) || TYPE_PRECISION (inner_t) <= TYPE_PRECISION (outer_t))) - return true; + { + *unknown_max = true; + return true; + } } } @@ -1746,11 +1944,13 @@ scev_probably_wraps_p (tree type, tree base, tree step, /* At this point we still don't have a proof that the iv does not overflow: give up. */ + *unknown_max = true; return true; } /* Return the conversion to NEW_TYPE of the STEP of an induction - variable BASE + STEP * I at AT_STMT. */ + variable BASE + STEP * I at AT_STMT. When it fails, return + NULL_TREE. */ tree convert_step (struct loop *loop, tree new_type, tree base, tree step, diff --git a/gcc/tree-vect-analyze.c b/gcc/tree-vect-analyze.c index 6786161312b..8f18d89f9a4 100644 --- a/gcc/tree-vect-analyze.c +++ b/gcc/tree-vect-analyze.c @@ -1173,7 +1173,7 @@ vect_analyze_data_refs (loop_vec_info loop_vinfo) if (!dr || !DR_REF (dr)) { if (vect_print_dump_info (REPORT_UNVECTORIZED_LOOPS)) - fprintf (vect_dump, "not vectorized: unhandled data-ref "); + fprintf (vect_dump, "not vectorized: unhandled data-ref "); return false; } diff --git a/gcc/tree-vrp.c b/gcc/tree-vrp.c index 81fda73e7fd..0f7f6710433 100644 --- a/gcc/tree-vrp.c +++ b/gcc/tree-vrp.c @@ -1534,29 +1534,31 @@ adjust_range_with_scev (value_range_t *vr, struct loop *loop, tree stmt, tree var) { tree init, step, chrec; - bool init_is_max; + bool init_is_max, unknown_max; /* TODO. Don't adjust anti-ranges. An anti-range may provide better opportunities than a regular range, but I'm not sure. */ if (vr->type == VR_ANTI_RANGE) return; - chrec = analyze_scalar_evolution (loop, var); + chrec = instantiate_parameters (loop, analyze_scalar_evolution (loop, var)); if (TREE_CODE (chrec) != POLYNOMIAL_CHREC) return; - init = CHREC_LEFT (chrec); - step = CHREC_RIGHT (chrec); + init = initial_condition_in_loop_num (chrec, loop->num); + step = evolution_part_in_loop_num (chrec, loop->num); /* If STEP is symbolic, we can't know whether INIT will be the minimum or maximum value in the range. */ - if (!is_gimple_min_invariant (step)) + if (step == NULL_TREE + || !is_gimple_min_invariant (step)) return; /* Do not adjust ranges when chrec may wrap. */ if (scev_probably_wraps_p (chrec_type (chrec), init, step, stmt, cfg_loops->parray[CHREC_VARIABLE (chrec)], - &init_is_max)) + &init_is_max, &unknown_max) + || unknown_max) return; if (!POINTER_TYPE_P (TREE_TYPE (init)) |