On Tue, 15 Nov 2011, Andrey Belevantsev wrote: > > diff --git a/gcc/sel-sched.c b/gcc/sel-sched.c > > index f11faca..91fb0fe 100644 > > --- a/gcc/sel-sched.c > > +++ b/gcc/sel-sched.c > > @@ -5234,6 +5234,7 @@ move_exprs_to_boundary (bnd_t bnd, expr_t expr_vliw, > > > > b = move_op (BND_TO (bnd), expr_seq, expr_vliw, > > get_dest_from_orig_ops (expr_seq), c_expr,&should_move); > > + free_history_vect (&EXPR_HISTORY_OF_CHANGES (c_expr)); > Why do we need this? Upon returning from move_exprs_to_boundary (in > schedule_expr_on_boundary), we immediately call clear_expr on c_expr which in > turn calls free_history_vect. So the history vector would be cleared anyways, > no? In this case we don't have to export free_history_vect, too.
Indeed, thanks for noticing this. Updated patch below (retested on amd64-linux). 2011-11-17 Alexander Monakov <amona...@ispras.ru> * sel-sched-ir.c (free_history_vect): Remove unneeded declaration. (insert_in_history_vect): Rename argument spec_ds to old_spec_ds, add new argument (new_spec_ds). Record it in the history vector. Update all callers. (init_expr): Assert that initial history is empty. Simplify code. (copy_history_of_changes): Split out from copy_expr. * sel-sched-ir.h (struct expr_history_def_1): Rename spec_ds to old_spec_ds. Add new member (new_spec_ds). (copy_history_of_changes): Declare. (insert_in_history_vect): Adjust prototype. * sel-sched.c (undo_transformations): Adjust. (move_op_orig_expr_found): Populate history vector of c_expr for use in redo_transformations. (redo_transformations): New. (move_op_ascend): Use redo_transformations. diff --git a/gcc/sel-sched-ir.c b/gcc/sel-sched-ir.c index a4fb9ac..d394019 100644 --- a/gcc/sel-sched-ir.c +++ b/gcc/sel-sched-ir.c @@ -149,7 +149,6 @@ static expr_t set_insn_init (expr_t, vinsn_t, int); static void cfg_preds (basic_block, insn_t **, int *); static void prepare_insn_expr (insn_t, int); -static void free_history_vect (VEC (expr_history_def, heap) **); static void move_bb_info (basic_block, basic_block); static void remove_empty_bb (basic_block, bool); @@ -1503,13 +1502,13 @@ find_in_history_vect (VEC(expr_history_def, heap) *vect, rtx insn, /* Insert new element in a sorted history vector pointed to by PVECT, if it is not there already. The element is searched using - UID/NEW_EXPR_VINSN pair. TYPE, OLD_EXPR_VINSN and SPEC_DS save - the history of a transformation. */ + UID/NEW_EXPR_VINSN pair. TYPE, OLD_EXPR_VINSN, OLD_SPEC_DS and + NEW_SPEC_DS save the history of a transformation. */ void insert_in_history_vect (VEC (expr_history_def, heap) **pvect, unsigned uid, enum local_trans_type type, vinsn_t old_expr_vinsn, vinsn_t new_expr_vinsn, - ds_t spec_ds) + ds_t old_spec_ds, ds_t new_spec_ds) { VEC(expr_history_def, heap) *vect = *pvect; expr_history_def temp; @@ -1525,15 +1524,16 @@ insert_in_history_vect (VEC (expr_history_def, heap) **pvect, /* It is possible that speculation types of expressions that were propagated through different paths will be different here. In this case, merge the status to get the correct check later. */ - if (phist->spec_ds != spec_ds) - phist->spec_ds = ds_max_merge (phist->spec_ds, spec_ds); + if (phist->old_spec_ds != old_spec_ds) + phist->old_spec_ds = ds_max_merge (phist->old_spec_ds, old_spec_ds); return; } temp.uid = uid; temp.old_expr_vinsn = old_expr_vinsn; temp.new_expr_vinsn = new_expr_vinsn; - temp.spec_ds = spec_ds; + temp.old_spec_ds = old_spec_ds; + temp.new_spec_ds = new_spec_ds; temp.type = type; vinsn_attach (old_expr_vinsn); @@ -1576,7 +1576,7 @@ merge_history_vect (VEC (expr_history_def, heap) **pvect, for (i = 0; VEC_iterate (expr_history_def, from, i, phist); i++) insert_in_history_vect (pvect, phist->uid, phist->type, phist->old_expr_vinsn, phist->new_expr_vinsn, - phist->spec_ds); + phist->old_spec_ds, phist->new_spec_ds); } /* Compare two vinsns as rhses if possible and as vinsns otherwise. */ @@ -1632,10 +1632,8 @@ init_expr (expr_t expr, vinsn_t vi, int spec, int use, int priority, EXPR_SPEC_DONE_DS (expr) = spec_done_ds; EXPR_SPEC_TO_CHECK_DS (expr) = spec_to_check_ds; - if (history) - EXPR_HISTORY_OF_CHANGES (expr) = history; - else - EXPR_HISTORY_OF_CHANGES (expr) = NULL; + gcc_assert (!history); + EXPR_HISTORY_OF_CHANGES (expr) = history; EXPR_TARGET_AVAILABLE (expr) = target_available; EXPR_WAS_SUBSTITUTED (expr) = was_substituted; @@ -1644,35 +1642,39 @@ init_expr (expr_t expr, vinsn_t vi, int spec, int use, int priority, EXPR_CANT_MOVE (expr) = cant_move; } -/* Make a copy of the expr FROM into the expr TO. */ +/* Copy history vector of the expr FROM to the expr TO. */ void -copy_expr (expr_t to, expr_t from) +copy_history_of_changes (expr_t to, expr_t from) { - VEC(expr_history_def, heap) *temp = NULL; - if (EXPR_HISTORY_OF_CHANGES (from)) { unsigned i; expr_history_def *phist; - temp = VEC_copy (expr_history_def, heap, EXPR_HISTORY_OF_CHANGES (from)); - for (i = 0; - VEC_iterate (expr_history_def, temp, i, phist); - i++) + EXPR_HISTORY_OF_CHANGES (to) + = VEC_copy (expr_history_def, heap, EXPR_HISTORY_OF_CHANGES (from)); + FOR_EACH_VEC_ELT (expr_history_def, EXPR_HISTORY_OF_CHANGES (to), i, + phist) { vinsn_attach (phist->old_expr_vinsn); vinsn_attach (phist->new_expr_vinsn); } } +} +/* Make a copy of the expr FROM into the expr TO. */ +void +copy_expr (expr_t to, expr_t from) +{ init_expr (to, EXPR_VINSN (from), EXPR_SPEC (from), EXPR_USEFULNESS (from), EXPR_PRIORITY (from), EXPR_SCHED_TIMES (from), EXPR_ORIG_BB_INDEX (from), EXPR_SPEC_DONE_DS (from), EXPR_SPEC_TO_CHECK_DS (from), - EXPR_ORIG_SCHED_CYCLE (from), temp, + EXPR_ORIG_SCHED_CYCLE (from), NULL, EXPR_TARGET_AVAILABLE (from), EXPR_WAS_SUBSTITUTED (from), EXPR_WAS_RENAMED (from), EXPR_NEEDS_SPEC_CHECK_P (from), EXPR_CANT_MOVE (from)); + copy_history_of_changes (to, from); } /* Same, but the final expr will not ever be in av sets, so don't copy @@ -1803,7 +1805,7 @@ update_speculative_bits (expr_t to, expr_t from, insn_t split_point) insert_in_history_vect (&EXPR_HISTORY_OF_CHANGES (to), INSN_UID (split_point), TRANS_SPECULATION, EXPR_VINSN (from), EXPR_VINSN (to), - record_ds); + record_ds, EXPR_SPEC_DONE_DS (to)); } } } diff --git a/gcc/sel-sched-ir.h b/gcc/sel-sched-ir.h index ede08e4..0a428f0 100644 --- a/gcc/sel-sched-ir.h +++ b/gcc/sel-sched-ir.h @@ -89,8 +89,11 @@ struct expr_history_def_1 /* How the expression looks after the transformation. */ vinsn_t new_expr_vinsn; - /* And its speculative status. */ - ds_t spec_ds; + /* Its previous speculative status. */ + ds_t old_spec_ds; + + /* Its new speculative status. */ + ds_t new_spec_ds; /* Type of the transformation. */ enum local_trans_type type; @@ -1538,6 +1541,7 @@ extern bool vinsn_equal_p (vinsn_t, vinsn_t); /* EXPR functions. */ extern void copy_expr (expr_t, expr_t); +extern void copy_history_of_changes (expr_t, expr_t); extern void copy_expr_onside (expr_t, expr_t); extern void merge_expr_data (expr_t, expr_t, insn_t); extern void merge_expr (expr_t, expr_t, insn_t); @@ -1548,7 +1552,7 @@ extern int find_in_history_vect (VEC(expr_history_def, heap) *, rtx, vinsn_t, bool); extern void insert_in_history_vect (VEC(expr_history_def, heap) **, unsigned, enum local_trans_type, - vinsn_t, vinsn_t, ds_t); + vinsn_t, vinsn_t, ds_t, ds_t); extern void mark_unavailable_targets (av_set_t, av_set_t, regset); extern int speculate_expr (expr_t, ds_t); diff --git a/gcc/sel-sched.c b/gcc/sel-sched.c index 2af01ae..3cabfdd 100644 --- a/gcc/sel-sched.c +++ b/gcc/sel-sched.c @@ -1959,7 +1959,7 @@ undo_transformations (av_set_t *av_ptr, rtx insn) and in the case of merging different probabilities of the same speculative status along different paths we do not record this in the history vector. */ - old_ds = phist->spec_ds; + old_ds = phist->old_spec_ds; new_ds = EXPR_SPEC_DONE_DS (expr); old_ds &= SPECULATIVE; @@ -2403,7 +2403,7 @@ try_transformation_cache (expr_t expr, insn_t insn, insert_in_history_vect (&EXPR_HISTORY_OF_CHANGES (expr), INSN_UID (insn), pti->type, pti->vinsn_old, pti->vinsn_new, - EXPR_SPEC_DONE_DS (expr)); + EXPR_SPEC_DONE_DS (expr), pti->ds); if (INSN_IN_STREAM_P (VINSN_INSN_RTX (pti->vinsn_new))) pti->vinsn_new = vinsn_copy (pti->vinsn_new, true); @@ -2558,7 +2558,7 @@ moveup_expr_cached (expr_t expr, insn_t insn, bool inside_insn_group) insert_in_history_vect (&EXPR_HISTORY_OF_CHANGES (expr), INSN_UID (insn), trans_type, expr_old_vinsn, EXPR_VINSN (expr), - expr_old_spec_ds); + expr_old_spec_ds, EXPR_SPEC_DONE_DS (expr)); update_transformation_cache (expr, insn, inside_insn_group, trans_type, expr_old_vinsn); if (sched_verbose >= 6) @@ -5993,6 +5993,7 @@ move_op_orig_expr_found (insn_t insn, expr_t expr, moveop_static_params_p params = (moveop_static_params_p) static_params; copy_expr_onside (params->c_expr, INSN_EXPR (insn)); + copy_history_of_changes (params->c_expr, expr); track_scheduled_insns_and_blocks (insn); insn_emitted = handle_emitting_transformations (insn, expr, params); only_disconnect = (params->uid == INSN_UID (insn) @@ -6143,19 +6144,78 @@ fur_at_first_insn (insn_t insn, || AV_LEVEL (insn) == -1); } +/* Redo the transformation that moveup_expr performed on EXPR for INSN, as + recorded in EXPR's history vector. */ +static void +redo_transformations (expr_t expr, insn_t insn) +{ + unsigned i, insn_uid = INSN_UID (insn); + rtx orig_expr_lhs = EXPR_LHS (expr); + expr_history_def *hist; + VEC(expr_history_def, heap) *vect = EXPR_HISTORY_OF_CHANGES (expr); + + if (sched_verbose >= 6) + { + sel_print ("Moving "); + dump_expr (expr); + sel_print (" through %d: ", INSN_UID (insn)); + } + + FOR_EACH_VEC_ELT (expr_history_def, vect, i, hist) + if (hist->uid == insn_uid) + { + if (vinsn_equal_p (EXPR_VINSN (expr), hist->old_expr_vinsn)) + { + switch (hist->type) + { + case TRANS_SPECULATION: + EXPR_SPEC_DONE_DS (expr) = hist->new_spec_ds; + /* Fallthru. */ + case TRANS_SUBSTITUTION: + change_vinsn_in_expr (expr, hist->new_expr_vinsn); + break; + default: + gcc_unreachable (); + } + + if (EXPR_LHS (expr) != orig_expr_lhs && expr_dest_reg (expr)) + replace_dest_with_reg_in_expr (expr, orig_expr_lhs); + + if (sched_verbose >= 6) + { + sel_print ("changed (redoing): "); + dump_expr (expr); + sel_print ("\n"); + } + return; + } + else + { + /* We cannot assert the following; consider: + r1 = r3; + r2 = r3; + r4 = r1; + r5 = r2; + We choose to schedule r4 = r3, find it at r4 = r1, and + attempt to redo transformation on r2 = r3, which was recorded, + because r5 = r2 was moved through it and unified afterwards. */ + /* gcc_assert (vinsn_equal_p (EXPR_VINSN (expr), hist->new_expr_vinsn)); */ + break; + } + } + if (sched_verbose >= 6) + sel_print ("unchanged (redoing)\n"); +} + /* Called on the backward stage of recursion to call moveup_expr for insn and sparams->c_expr. */ static void move_op_ascend (insn_t insn, void *static_params) { - enum MOVEUP_EXPR_CODE res; moveop_static_params_p sparams = (moveop_static_params_p) static_params; if (! INSN_NOP_P (insn)) - { - res = moveup_expr_cached (sparams->c_expr, insn, false); - gcc_assert (res != MOVEUP_EXPR_NULL); - } + redo_transformations (sparams->c_expr, insn); /* Update liveness for this insn as it was invalidated. */ update_liveness_on_insn (insn);