diff options
Diffstat (limited to 'contrib/gcc/regmove.c')
-rw-r--r-- | contrib/gcc/regmove.c | 555 |
1 files changed, 372 insertions, 183 deletions
diff --git a/contrib/gcc/regmove.c b/contrib/gcc/regmove.c index 605169a..81a3520 100644 --- a/contrib/gcc/regmove.c +++ b/contrib/gcc/regmove.c @@ -15,7 +15,8 @@ GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU CC; see the file COPYING. If not, write to -the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ +the Free Software Foundation, 59 Temple Place - Suite 330, +Boston, MA 02111-1307, USA. */ /* This module looks for cases where matching constraints would force @@ -24,16 +25,8 @@ the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ instruction to avoid the move instruction. */ #include "config.h" -#ifdef __STDC__ -#include <stdarg.h> -#else -#include <varargs.h> -#endif - -/* stdio.h must precede rtl.h for FFS. */ #include "system.h" - -#include "rtl.h" +#include "rtl.h" /* stdio.h must precede rtl.h for FFS. */ #include "insn-config.h" #include "recog.h" #include "output.h" @@ -50,7 +43,7 @@ static int optimize_reg_copy_1 PROTO((rtx, rtx, rtx)); static void optimize_reg_copy_2 PROTO((rtx, rtx, rtx)); static void optimize_reg_copy_3 PROTO((rtx, rtx, rtx)); static rtx gen_add3_insn PROTO((rtx, rtx, rtx)); -static void copy_src_to_dest PROTO((rtx, rtx, rtx, int)); +static void copy_src_to_dest PROTO((rtx, rtx, rtx, int, int)); static int *regmove_bb_head; struct match { @@ -60,16 +53,32 @@ struct match { int early_clobber[MAX_RECOG_OPERANDS]; }; -#ifdef AUTO_INC_DEC +static rtx discover_flags_reg PROTO((void)); +static void mark_flags_life_zones PROTO((rtx)); +static void flags_set_1 PROTO((rtx, rtx)); + static int try_auto_increment PROTO((rtx, rtx, rtx, rtx, HOST_WIDE_INT, int)); -#endif static int find_matches PROTO((rtx, struct match *)); static int fixup_match_1 PROTO((rtx, rtx, rtx, rtx, rtx, int, int, int, FILE *)) ; static int reg_is_remote_constant_p PROTO((rtx, rtx, rtx)); static int stable_but_for_p PROTO((rtx, rtx, rtx)); +static int regclass_compatible_p PROTO((int, int)); static int loop_depth; +/* Return non-zero if registers with CLASS1 and CLASS2 can be merged without + causing too much register allocation problems. */ +static int +regclass_compatible_p (class0, class1) + int class0, class1; +{ + return (class0 == class1 + || (reg_class_subset_p (class0, class1) + && ! CLASS_LIKELY_SPILLED_P (class0)) + || (reg_class_subset_p (class1, class0) + && ! CLASS_LIKELY_SPILLED_P (class1))); +} + /* Generate and return an insn body to add r1 and c, storing the result in r0. */ static rtx @@ -87,7 +96,6 @@ gen_add3_insn (r0, r1, c) return (GEN_FCN (icode) (r0, r1, c)); } -#ifdef AUTO_INC_DEC /* INC_INSN is an instruction that adds INCREMENT to REG. Try to fold INC_INSN as a post/pre in/decrement into INSN. @@ -111,18 +119,14 @@ try_auto_increment (insn, inc_insn, inc_insn_set, reg, increment, pre) { int size = GET_MODE_SIZE (GET_MODE (use)); if (0 -#ifdef HAVE_POST_INCREMENT - || (pre == 0 && (inc_code = POST_INC, increment == size)) -#endif -#ifdef HAVE_PRE_INCREMENT - || (pre == 1 && (inc_code = PRE_INC, increment == size)) -#endif -#ifdef HAVE_POST_DECREMENT - || (pre == 0 && (inc_code = POST_DEC, increment == -size)) -#endif -#ifdef HAVE_PRE_DECREMENT - || (pre == 1 && (inc_code = PRE_DEC, increment == -size)) -#endif + || (HAVE_POST_INCREMENT + && pre == 0 && (inc_code = POST_INC, increment == size)) + || (HAVE_PRE_INCREMENT + && pre == 1 && (inc_code = PRE_INC, increment == size)) + || (HAVE_POST_DECREMENT + && pre == 0 && (inc_code = POST_DEC, increment == -size)) + || (HAVE_PRE_DECREMENT + && pre == 1 && (inc_code = PRE_DEC, increment == -size)) ) { if (inc_insn_set) @@ -150,8 +154,175 @@ try_auto_increment (insn, inc_insn, inc_insn_set, reg, increment, pre) } return 0; } -#endif /* AUTO_INC_DEC */ + +/* Determine if the pattern generated by add_optab has a clobber, + such as might be issued for a flags hard register. To make the + code elsewhere simpler, we handle cc0 in this same framework. + + Return the register if one was discovered. Return NULL_RTX if + if no flags were found. Return pc_rtx if we got confused. */ + +static rtx +discover_flags_reg () +{ + rtx tmp; + tmp = gen_rtx_REG (word_mode, 10000); + tmp = gen_add3_insn (tmp, tmp, GEN_INT (2)); + + /* If we get something that isn't a simple set, or a + [(set ..) (clobber ..)], this whole function will go wrong. */ + if (GET_CODE (tmp) == SET) + return NULL_RTX; + else if (GET_CODE (tmp) == PARALLEL) + { + int found; + + if (XVECLEN (tmp, 0) != 2) + return pc_rtx; + tmp = XVECEXP (tmp, 0, 1); + if (GET_CODE (tmp) != CLOBBER) + return pc_rtx; + tmp = XEXP (tmp, 0); + + /* Don't do anything foolish if the md wanted to clobber a + scratch or something. We only care about hard regs. + Moreover we don't like the notion of subregs of hard regs. */ + if (GET_CODE (tmp) == SUBREG + && GET_CODE (SUBREG_REG (tmp)) == REG + && REGNO (SUBREG_REG (tmp)) < FIRST_PSEUDO_REGISTER) + return pc_rtx; + found = (GET_CODE (tmp) == REG && REGNO (tmp) < FIRST_PSEUDO_REGISTER); + + return (found ? tmp : NULL_RTX); + } + + return pc_rtx; +} + +/* It is a tedious task identifying when the flags register is live and + when it is safe to optimize. Since we process the instruction stream + multiple times, locate and record these live zones by marking the + mode of the instructions -- + + QImode is used on the instruction at which the flags becomes live. + + HImode is used within the range (exclusive) that the flags are + live. Thus the user of the flags is not marked. + + All other instructions are cleared to VOIDmode. */ +/* Used to communicate with flags_set_1. */ +static rtx flags_set_1_rtx; +static int flags_set_1_set; + +static void +mark_flags_life_zones (flags) + rtx flags; +{ + int flags_regno; + int flags_nregs; + int block; + +#ifdef HAVE_cc0 + /* If we found a flags register on a cc0 host, bail. */ + if (flags == NULL_RTX) + flags = cc0_rtx; + else if (flags != cc0_rtx) + flags = pc_rtx; +#endif + + /* Simple cases first: if no flags, clear all modes. If confusing, + mark the entire function as being in a flags shadow. */ + if (flags == NULL_RTX || flags == pc_rtx) + { + enum machine_mode mode = (flags ? HImode : VOIDmode); + rtx insn; + for (insn = get_insns(); insn; insn = NEXT_INSN (insn)) + PUT_MODE (insn, mode); + return; + } + +#ifdef HAVE_cc0 + flags_regno = -1; + flags_nregs = 1; +#else + flags_regno = REGNO (flags); + flags_nregs = HARD_REGNO_NREGS (flags_regno, GET_MODE (flags)); +#endif + flags_set_1_rtx = flags; + + /* Process each basic block. */ + for (block = n_basic_blocks - 1; block >= 0; block--) + { + rtx insn, end; + int live; + + insn = BLOCK_HEAD (block); + end = BLOCK_END (block); + + /* Look out for the (unlikely) case of flags being live across + basic block boundaries. */ + live = 0; +#ifndef HAVE_cc0 + { + int i; + for (i = 0; i < flags_nregs; ++i) + live |= REGNO_REG_SET_P (BASIC_BLOCK (block)->global_live_at_start, + flags_regno + i); + } +#endif + + while (1) + { + /* Process liveness in reverse order of importance -- + alive, death, birth. This lets more important info + overwrite the mode of lesser info. */ + + if (GET_RTX_CLASS (GET_CODE (insn)) == 'i') + { +#ifdef HAVE_cc0 + /* In the cc0 case, death is not marked in reg notes, + but is instead the mere use of cc0 when it is alive. */ + if (live && reg_mentioned_p (cc0_rtx, PATTERN (insn))) + live = 0; +#else + /* In the hard reg case, we watch death notes. */ + if (live && find_regno_note (insn, REG_DEAD, flags_regno)) + live = 0; +#endif + PUT_MODE (insn, (live ? HImode : VOIDmode)); + + /* In either case, birth is denoted simply by it's presence + as the destination of a set. */ + flags_set_1_set = 0; + note_stores (PATTERN (insn), flags_set_1); + if (flags_set_1_set) + { + live = 1; + PUT_MODE (insn, QImode); + } + } + else + PUT_MODE (insn, (live ? HImode : VOIDmode)); + + if (insn == end) + break; + insn = NEXT_INSN (insn); + } + } +} + +/* A subroutine of mark_flags_life_zones, called through note_stores. */ + +static void +flags_set_1 (x, pat) + rtx x, pat; +{ + if (GET_CODE (pat) == SET + && reg_overlap_mentioned_p (x, flags_set_1_rtx)) + flags_set_1_set = 1; +} + static int *regno_src_regno; /* Indicate how good a choice REG (which appears as a source) is to replace @@ -496,18 +667,6 @@ optimize_reg_copy_3 (insn, dest, src) rtx p, set, subreg; enum machine_mode old_mode; - /* This code has been disabled on the egcs-1.1 release branch due to - a potentially serious bug. - - In a nutshell, if we perform a series of substitutions, then have a - later substitution fail we will not be able to undo the previous - substitutions, leaving bogus RTL. - - A fix for this can be found in the mainline sources, but it did not - seem worth the trouble and potential problems to migrate the real - fix to the egcs-1.1 branch. */ - return; - if (src_no < FIRST_PSEUDO_REGISTER || dst_no < FIRST_PSEUDO_REGISTER || ! find_reg_note (insn, REG_DEAD, src_reg) @@ -536,23 +695,48 @@ optimize_reg_copy_3 (insn, dest, src) || GET_CODE (SET_SRC (set)) != MEM || SET_DEST (set) != src_reg) return; + + /* Be conserative: although this optimization is also valid for + volatile memory references, that could cause trouble in later passes. */ + if (MEM_VOLATILE_P (SET_SRC (set))) + return; + + /* Do not use a SUBREG to truncate from one mode to another if truncation + is not a nop. */ + if (GET_MODE_BITSIZE (GET_MODE (src_reg)) <= GET_MODE_BITSIZE (GET_MODE (src)) + && !TRULY_NOOP_TRUNCATION (GET_MODE_BITSIZE (GET_MODE (src)), + GET_MODE_BITSIZE (GET_MODE (src_reg)))) + return; + old_mode = GET_MODE (src_reg); PUT_MODE (src_reg, GET_MODE (src)); XEXP (src, 0) = SET_SRC (set); - if (! validate_change (p, &SET_SRC (set), src, 0)) - { - PUT_MODE (src_reg, old_mode); - XEXP (src, 0) = src_reg; - return; - } + + /* Include this change in the group so that it's easily undone if + one of the changes in the group is invalid. */ + validate_change (p, &SET_SRC (set), src, 1); + + /* Now walk forward making additional replacements. We want to be able + to undo all the changes if a later substitution fails. */ subreg = gen_rtx_SUBREG (old_mode, src_reg, 0); while (p = NEXT_INSN (p), p != insn) { if (GET_RTX_CLASS (GET_CODE (p)) != 'i') continue; - validate_replace_rtx (src_reg, subreg, p); + + /* Make a tenative change. */ + validate_replace_rtx_group (src_reg, subreg, p); + } + + validate_replace_rtx_group (src, src_reg, insn); + + /* Now see if all the changes are valid. */ + if (! apply_change_group ()) + { + /* One or more changes were no good. Back out everything. */ + PUT_MODE (src_reg, old_mode); + XEXP (src, 0) = src_reg; } - validate_replace_rtx (src, src_reg, insn); } @@ -560,11 +744,12 @@ optimize_reg_copy_3 (insn, dest, src) instead moving the value to dest directly before the operation. */ static void -copy_src_to_dest (insn, src, dest, loop_depth) +copy_src_to_dest (insn, src, dest, loop_depth, old_max_uid) rtx insn; rtx src; rtx dest; int loop_depth; + int old_max_uid; { rtx seq; rtx link; @@ -590,13 +775,25 @@ copy_src_to_dest (insn, src, dest, loop_depth) && REG_LIVE_LENGTH (REGNO (dest)) > 0 && (set = single_set (insn)) != NULL_RTX && !reg_mentioned_p (dest, SET_SRC (set)) - && validate_replace_rtx (src, dest, insn)) + && GET_MODE (src) == GET_MODE (dest)) { + int old_num_regs = reg_rtx_no; + /* Generate the src->dest move. */ start_sequence (); emit_move_insn (dest, src); seq = gen_sequence (); end_sequence (); + /* If this sequence uses new registers, we may not use it. */ + if (old_num_regs != reg_rtx_no + || ! validate_replace_rtx (src, dest, insn)) + { + /* We have to restore reg_rtx_no to its old value, lest + recompute_reg_usage will try to compute the usage of the + new regs, yet reg_n_info is not valid for them. */ + reg_rtx_no = old_num_regs; + return; + } emit_insn_before (seq, insn); move_insn = PREV_INSN (insn); p_move_notes = ®_NOTES (move_insn); @@ -624,11 +821,14 @@ copy_src_to_dest (insn, src, dest, loop_depth) /* Is the insn the head of a basic block? If so extend it */ insn_uid = INSN_UID (insn); move_uid = INSN_UID (move_insn); - bb = regmove_bb_head[insn_uid]; - if (bb >= 0) + if (insn_uid < old_max_uid) { - basic_block_head[bb] = move_insn; - regmove_bb_head[insn_uid] = -1; + bb = regmove_bb_head[insn_uid]; + if (bb >= 0) + { + BLOCK_HEAD (bb) = move_insn; + regmove_bb_head[insn_uid] = -1; + } } /* Update the various register tables. */ @@ -877,19 +1077,24 @@ regmove_optimize (f, nregs, regmove_dump_file) int nregs; FILE *regmove_dump_file; { + int old_max_uid = get_max_uid (); rtx insn; struct match match; int pass; - int maxregnum = max_reg_num (), i; + int i; rtx copy_src, copy_dst; - regno_src_regno = (int *)alloca (sizeof *regno_src_regno * maxregnum); - for (i = maxregnum; --i >= 0; ) regno_src_regno[i] = -1; + /* Find out where a potential flags register is live, and so that we + can supress some optimizations in those zones. */ + mark_flags_life_zones (discover_flags_reg ()); - regmove_bb_head = (int *)alloca (sizeof (int) * (get_max_uid () + 1)); - for (i = get_max_uid (); i >= 0; i--) regmove_bb_head[i] = -1; + regno_src_regno = (int *)alloca (sizeof *regno_src_regno * nregs); + for (i = nregs; --i >= 0; ) regno_src_regno[i] = -1; + + regmove_bb_head = (int *)alloca (sizeof (int) * (old_max_uid + 1)); + for (i = old_max_uid; i >= 0; i--) regmove_bb_head[i] = -1; for (i = 0; i < n_basic_blocks; i++) - regmove_bb_head[INSN_UID (basic_block_head[i])] = i; + regmove_bb_head[INSN_UID (BLOCK_HEAD (i))] = i; /* A forward/backward pass. Replace output operands with input operands. */ @@ -908,8 +1113,7 @@ regmove_optimize (f, nregs, regmove_dump_file) insn = pass ? PREV_INSN (insn) : NEXT_INSN (insn)) { rtx set; - int insn_code_number; - int operand_number, match_number; + int op_no, match_no; if (GET_CODE (insn) == NOTE) { @@ -954,11 +1158,11 @@ regmove_optimize (f, nregs, regmove_dump_file) } } } -#ifdef REGISTER_CONSTRAINTS - insn_code_number - = find_matches (insn, &match); + if (! flag_regmove) + continue; - if (insn_code_number < 0) +#ifdef REGISTER_CONSTRAINTS + if (! find_matches (insn, &match)) continue; /* Now scan through the operands looking for a source operand @@ -968,21 +1172,19 @@ regmove_optimize (f, nregs, regmove_dump_file) If it dies there, then replace the dest in both operands with the source operand. */ - for (operand_number = 0; - operand_number < insn_n_operands[insn_code_number]; - operand_number++) + for (op_no = 0; op_no < recog_n_operands; op_no++) { rtx src, dst, src_subreg; enum reg_class src_class, dst_class; - match_number = match.with[operand_number]; + match_no = match.with[op_no]; /* Nothing to do if the two operands aren't supposed to match. */ - if (match_number < 0) + if (match_no < 0) continue; - src = recog_operand[operand_number]; - dst = recog_operand[match_number]; + src = recog_operand[op_no]; + dst = recog_operand[match_no]; if (GET_CODE (src) != REG) continue; @@ -1003,7 +1205,7 @@ regmove_optimize (f, nregs, regmove_dump_file) if (REGNO (src) < FIRST_PSEUDO_REGISTER) { - if (match.commutative[operand_number] < operand_number) + if (match.commutative[op_no] < op_no) regno_src_regno[REGNO (dst)] = REGNO (src); continue; } @@ -1011,42 +1213,38 @@ regmove_optimize (f, nregs, regmove_dump_file) if (REG_LIVE_LENGTH (REGNO (src)) < 0) continue; - /* operand_number/src must be a read-only operand, and + /* op_no/src must be a read-only operand, and match_operand/dst must be a write-only operand. */ - if (match.use[operand_number] != READ - || match.use[match_number] != WRITE) + if (match.use[op_no] != READ + || match.use[match_no] != WRITE) continue; - if (match.early_clobber[match_number] + if (match.early_clobber[match_no] && count_occurrences (PATTERN (insn), src) > 1) continue; /* Make sure match_operand is the destination. */ - if (recog_operand[match_number] != SET_DEST (set)) + if (recog_operand[match_no] != SET_DEST (set)) continue; /* If the operands already match, then there is nothing to do. */ /* But in the commutative case, we might find a better match. */ if (operands_match_p (src, dst) - || (match.commutative[operand_number] >= 0 + || (match.commutative[op_no] >= 0 && operands_match_p (recog_operand[match.commutative - [operand_number]], dst) + [op_no]], dst) && (replacement_quality (recog_operand[match.commutative - [operand_number]]) + [op_no]]) >= replacement_quality (src)))) continue; src_class = reg_preferred_class (REGNO (src)); dst_class = reg_preferred_class (REGNO (dst)); - if (src_class != dst_class - && (! reg_class_subset_p (src_class, dst_class) - || CLASS_LIKELY_SPILLED_P (src_class)) - && (! reg_class_subset_p (dst_class, src_class) - || CLASS_LIKELY_SPILLED_P (dst_class))) + if (! regclass_compatible_p (src_class, dst_class)) continue; if (fixup_match_1 (insn, set, src, src_subreg, dst, pass, - operand_number, match_number, + op_no, match_no, regmove_dump_file)) break; } @@ -1071,11 +1269,10 @@ regmove_optimize (f, nregs, regmove_dump_file) } if (GET_RTX_CLASS (GET_CODE (insn)) == 'i') { - int insn_code_number = find_matches (insn, &match); - int operand_number, match_number; + int op_no, match_no; int success = 0; - - if (insn_code_number < 0) + + if (! find_matches (insn, &match)) continue; /* Now scan through the operands looking for a destination operand @@ -1086,9 +1283,7 @@ regmove_optimize (f, nregs, regmove_dump_file) copy_src = NULL_RTX; copy_dst = NULL_RTX; - for (operand_number = 0; - operand_number < insn_n_operands[insn_code_number]; - operand_number++) + for (op_no = 0; op_no < recog_n_operands; op_no++) { rtx set, p, src, dst; rtx src_note, dst_note; @@ -1096,14 +1291,14 @@ regmove_optimize (f, nregs, regmove_dump_file) enum reg_class src_class, dst_class; int length; - match_number = match.with[operand_number]; + match_no = match.with[op_no]; /* Nothing to do if the two operands aren't supposed to match. */ - if (match_number < 0) + if (match_no < 0) continue; - dst = recog_operand[match_number]; - src = recog_operand[operand_number]; + dst = recog_operand[match_no]; + src = recog_operand[op_no]; if (GET_CODE (src) != REG) continue; @@ -1115,26 +1310,26 @@ regmove_optimize (f, nregs, regmove_dump_file) /* If the operands already match, then there is nothing to do. */ if (operands_match_p (src, dst) - || (match.commutative[operand_number] >= 0 - && operands_match_p (recog_operand[match.commutative[operand_number]], dst))) + || (match.commutative[op_no] >= 0 + && operands_match_p (recog_operand[match.commutative[op_no]], dst))) continue; set = single_set (insn); if (! set) continue; - /* match_number/dst must be a write-only operand, and + /* match_no/dst must be a write-only operand, and operand_operand/src must be a read-only operand. */ - if (match.use[operand_number] != READ - || match.use[match_number] != WRITE) + if (match.use[op_no] != READ + || match.use[match_no] != WRITE) continue; - if (match.early_clobber[match_number] + if (match.early_clobber[match_no] && count_occurrences (PATTERN (insn), src) > 1) continue; - /* Make sure match_number is the destination. */ - if (recog_operand[match_number] != SET_DEST (set)) + /* Make sure match_no is the destination. */ + if (recog_operand[match_no] != SET_DEST (set)) continue; if (REGNO (src) < FIRST_PSEUDO_REGISTER) @@ -1150,11 +1345,7 @@ regmove_optimize (f, nregs, regmove_dump_file) } src_class = reg_preferred_class (REGNO (src)); dst_class = reg_preferred_class (REGNO (dst)); - if (src_class != dst_class - && (! reg_class_subset_p (src_class, dst_class) - || CLASS_LIKELY_SPILLED_P (src_class)) - && (! reg_class_subset_p (dst_class, src_class) - || CLASS_LIKELY_SPILLED_P (dst_class))) + if (! regclass_compatible_p (src_class, dst_class)) { if (!copy_src) { @@ -1206,11 +1397,11 @@ regmove_optimize (f, nregs, regmove_dump_file) if (regmove_dump_file) fprintf (regmove_dump_file, "Could fix operand %d of insn %d matching operand %d.\n", - operand_number, INSN_UID (insn), match_number); + op_no, INSN_UID (insn), match_no); /* Scan backward to find the first instruction that uses the input operand. If the operand is set here, then - replace it in both instructions with match_number. */ + replace it in both instructions with match_no. */ for (length = 0, p = PREV_INSN (insn); p; p = PREV_INSN (p)) { @@ -1258,7 +1449,7 @@ regmove_optimize (f, nregs, regmove_dump_file) validate_replace_rtx (dst, src, insn); /* Now make sure the dst is right. */ validate_change (insn, - recog_operand_loc[match_number], + recog_operand_loc[match_no], dst, 0); } } @@ -1338,7 +1529,7 @@ regmove_optimize (f, nregs, regmove_dump_file) if (regmove_dump_file) fprintf (regmove_dump_file, "Fixed operand %d of insn %d matching operand %d.\n", - operand_number, INSN_UID (insn), match_number); + op_no, INSN_UID (insn), match_no); break; } @@ -1347,16 +1538,30 @@ regmove_optimize (f, nregs, regmove_dump_file) /* If we weren't able to replace any of the alternatives, try an alternative appoach of copying the source to the destination. */ if (!success && copy_src != NULL_RTX) - copy_src_to_dest (insn, copy_src, copy_dst, loop_depth); + copy_src_to_dest (insn, copy_src, copy_dst, loop_depth, + old_max_uid); } } #endif /* REGISTER_CONSTRAINTS */ + + /* In fixup_match_1, some insns may have been inserted after basic block + ends. Fix that here. */ + for (i = 0; i < n_basic_blocks; i++) + { + rtx end = BLOCK_END (i); + rtx new = end; + rtx next = NEXT_INSN (new); + while (next != 0 && INSN_UID (next) >= old_max_uid + && (i == n_basic_blocks - 1 || BLOCK_HEAD (i + 1) != next)) + new = next, next = NEXT_INSN (new); + BLOCK_END (i) = new; + } } -/* Returns the INSN_CODE for INSN if its pattern has matching constraints for - any operand. Returns -1 if INSN can't be recognized, or if the alternative - can't be determined. +/* Returns nonzero if INSN's pattern has matching constraints for any operand. + Returns 0 if INSN can't be recognized, or if the alternative can't be + determined. Initialize the info in MATCHP based on the constraints. */ @@ -1366,39 +1571,34 @@ find_matches (insn, matchp) struct match *matchp; { int likely_spilled[MAX_RECOG_OPERANDS]; - int operand_number; - int insn_code_number = recog_memoized (insn); + int op_no; int any_matches = 0; - if (insn_code_number < 0) - return -1; - - insn_extract (insn); - if (! constrain_operands (insn_code_number, 0)) - return -1; + extract_insn (insn); + if (! constrain_operands (0)) + return 0; /* Must initialize this before main loop, because the code for the commutative case may set matches for operands other than the current one. */ - for (operand_number = insn_n_operands[insn_code_number]; - --operand_number >= 0; ) - matchp->with[operand_number] = matchp->commutative[operand_number] = -1; + for (op_no = recog_n_operands; --op_no >= 0; ) + matchp->with[op_no] = matchp->commutative[op_no] = -1; - for (operand_number = 0; operand_number < insn_n_operands[insn_code_number]; - operand_number++) + for (op_no = 0; op_no < recog_n_operands; op_no++) { - char *p, c; + const char *p; + char c; int i = 0; - p = insn_operand_constraint[insn_code_number][operand_number]; + p = recog_constraints[op_no]; - likely_spilled[operand_number] = 0; - matchp->use[operand_number] = READ; - matchp->early_clobber[operand_number] = 0; + likely_spilled[op_no] = 0; + matchp->use[op_no] = READ; + matchp->early_clobber[op_no] = 0; if (*p == '=') - matchp->use[operand_number] = WRITE; + matchp->use[op_no] = WRITE; else if (*p == '+') - matchp->use[operand_number] = READWRITE; + matchp->use[op_no] = READWRITE; for (;*p && i < which_alternative; p++) if (*p == ',') @@ -1412,32 +1612,32 @@ find_matches (insn, matchp) case '+': break; case '&': - matchp->early_clobber[operand_number] = 1; + matchp->early_clobber[op_no] = 1; break; case '%': - matchp->commutative[operand_number] = operand_number + 1; - matchp->commutative[operand_number + 1] = operand_number; + matchp->commutative[op_no] = op_no + 1; + matchp->commutative[op_no + 1] = op_no; break; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': c -= '0'; - if (c < operand_number && likely_spilled[(unsigned char) c]) + if (c < op_no && likely_spilled[(unsigned char) c]) break; - matchp->with[operand_number] = c; + matchp->with[op_no] = c; any_matches = 1; - if (matchp->commutative[operand_number] >= 0) - matchp->with[matchp->commutative[operand_number]] = c; + if (matchp->commutative[op_no] >= 0) + matchp->with[matchp->commutative[op_no]] = c; break; case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': case 'h': case 'j': case 'k': case 'l': case 'p': case 'q': case 't': case 'u': case 'v': case 'w': case 'x': case 'y': case 'z': case 'A': case 'B': case 'C': case 'D': case 'W': case 'Y': case 'Z': - if (CLASS_LIKELY_SPILLED_P (REG_CLASS_FROM_LETTER (c))) - likely_spilled[operand_number] = 1; + if (CLASS_LIKELY_SPILLED_P (REG_CLASS_FROM_LETTER ((unsigned char)c))) + likely_spilled[op_no] = 1; break; } } - return any_matches ? insn_code_number : -1; + return any_matches; } /* Try to replace output operand DST in SET, with input operand SRC. SET is @@ -1590,13 +1790,17 @@ fixup_match_1 (insn, set, src, src_subreg, dst, backward, operand_number, This also gives opportunities for subsequent optimizations in the backward pass, so do it there. */ if (code == PLUS && backward -#ifdef HAVE_cc0 - /* We may not emit an insn directly - after P if the latter sets CC0. */ - && ! sets_cc0_p (PATTERN (p)) -#endif - ) - + /* Don't do this if we can likely tie DST to SET_DEST + of P later; we can't do this tying here if we got a + hard register. */ + && ! (dst_note && ! REG_N_CALLS_CROSSED (REGNO (dst)) + && single_set (p) + && GET_CODE (SET_DEST (single_set (p))) == REG + && (REGNO (SET_DEST (single_set (p))) + < FIRST_PSEUDO_REGISTER)) + /* We may only emit an insn directly after P if we + are not in the shadow of a live flags register. */ + && GET_MODE (p) == VOIDmode) { search_end = q; q = insn; @@ -1670,11 +1874,10 @@ fixup_match_1 (insn, set, src, src_subreg, dst, backward, operand_number, if (code == MINUS) { post_inc = emit_insn_after (copy_rtx (PATTERN (insn)), p); -#if defined (HAVE_PRE_INCREMENT) || defined (HAVE_PRE_DECREMENT) - if (search_end + if ((HAVE_PRE_INCREMENT || HAVE_PRE_DECREMENT) + && search_end && try_auto_increment (search_end, post_inc, 0, src, newconst, 1)) post_inc = 0; -#endif validate_change (insn, &XEXP (SET_SRC (set), 1), GEN_INT (insn_const), 0); REG_N_SETS (REGNO (src))++; REG_N_REFS (REGNO (src)) += true_loop_depth; @@ -1687,12 +1890,11 @@ fixup_match_1 (insn, set, src, src_subreg, dst, backward, operand_number, rtx pat = PATTERN (insn); if (src_note) remove_note (overlap, src_note); -#if defined (HAVE_POST_INCREMENT) || defined (HAVE_POST_DECREMENT) - if (code == PLUS + if ((HAVE_POST_INCREMENT || HAVE_POST_DECREMENT) + && code == PLUS && try_auto_increment (overlap, insn, 0, src, insn_const, 0)) insn = overlap; else -#endif { rtx notes = REG_NOTES (insn); @@ -1775,35 +1977,23 @@ fixup_match_1 (insn, set, src, src_subreg, dst, backward, operand_number, } } - /* Don't remove this seemingly useless if, it is needed to pair with the - else in the next two conditionally included code blocks. */ - if (0) - {;} -#if defined (HAVE_PRE_INCREMENT) || defined (HAVE_PRE_DECREMENT) - else if ((code == PLUS || code == MINUS) && insn_const + if ((HAVE_PRE_INCREMENT || HAVE_PRE_DECREMENT) + && (code == PLUS || code == MINUS) && insn_const && try_auto_increment (p, insn, 0, src, insn_const, 1)) insn = p; -#endif -#if defined (HAVE_POST_INCREMENT) || defined (HAVE_POST_DECREMENT) - else if (post_inc + else if ((HAVE_POST_INCREMENT || HAVE_POST_DECREMENT) + && post_inc && try_auto_increment (p, post_inc, post_inc_set, src, newconst, 0)) post_inc = 0; -#endif -#if defined (HAVE_PRE_INCREMENT) || defined (HAVE_PRE_DECREMENT) /* If post_inc still prevails, try to find an insn where it can be used as a pre-in/decrement. If code is MINUS, this was already tried. */ if (post_inc && code == PLUS /* Check that newconst is likely to be usable in a pre-in/decrement before starting the search. */ - && (0 -#if defined (HAVE_PRE_INCREMENT) - || (newconst > 0 && newconst <= MOVE_MAX) -#endif -#if defined (HAVE_PRE_DECREMENT) - || (newconst < 0 && newconst >= -MOVE_MAX) -#endif - ) && exact_log2 (newconst)) + && ((HAVE_PRE_INCREMENT && newconst > 0 && newconst <= MOVE_MAX) + || (HAVE_PRE_DECREMENT && newconst < 0 && newconst >= -MOVE_MAX)) + && exact_log2 (newconst)) { rtx q, inc_dest; @@ -1840,7 +2030,6 @@ fixup_match_1 (insn, set, src, src_subreg, dst, backward, operand_number, } } } -#endif /* defined (HAVE_PRE_INCREMENT) || defined (HAVE_PRE_DECREMENT) */ /* Move the death note for DST to INSN if it is used there. */ if (reg_overlap_mentioned_p (dst, PATTERN (insn))) @@ -1973,7 +2162,7 @@ regmove_profitable_p () we find a machine where this occurs and regmove should be enabled. */ break; - if (find_matches (pat, &match) >= 0) + if (find_matches (pat, &match)) return 1; break; } |