diff options
author | nickc <nickc@138bc75d-0d04-0410-961f-82ee72b054a4> | 2004-11-19 09:47:23 +0000 |
---|---|---|
committer | nickc <nickc@138bc75d-0d04-0410-961f-82ee72b054a4> | 2004-11-19 09:47:23 +0000 |
commit | 0784ecd8586f31b393f42dc863d22e839133dac8 (patch) | |
tree | 8ca78644de58813f1f589e1ff2f999bc240d9ec5 /gcc | |
parent | 4d5cb40dc930d158298d8b2a2b3fb648f682cd8b (diff) | |
download | gcc-0784ecd8586f31b393f42dc863d22e839133dac8.tar.gz |
config/stormy16/stormy16.c (combine_bnp): Add code to handle zero_extension and
lshiftrt insns.
Replace uses of XEXP(...) with either SET_DEST() or SET_SRC() where
appropriate.
config/stormy16.stormy16.md (bclrx3, bsetx3): New patterns to handle bp and bn
insn patterns with a zero_extend rtx inside them.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@90898 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc')
-rw-r--r-- | gcc/ChangeLog | 10 | ||||
-rw-r--r-- | gcc/config/stormy16/stormy16.c | 114 | ||||
-rw-r--r-- | gcc/config/stormy16/stormy16.md | 26 |
3 files changed, 124 insertions, 26 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog index b451f4fb8bb..aef41d64e49 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,13 @@ +2004-11-19 Nick Clifton <nickc@redhat.com> + + * config/stormy16/stormy16.c (combine_bnp): Add code to handle + zero_extension and lshiftrt insns. + Replace uses of XEXP(...) with either SET_DEST() or SET_SRC() + where appropriate. + * config/stormy16.stormy16.md (bclrx3, bsetx3): New patterns + to handle bp and bn insn patterns with a zero_extend rtx + inside them. + 2004-11-19 Ian Lance Taylor <ian@wasabisystems.com> * config/arm/arm.md (generic_sched): Make const. diff --git a/gcc/config/stormy16/stormy16.c b/gcc/config/stormy16/stormy16.c index 06a2f0d3fb6..ec40cd916b2 100644 --- a/gcc/config/stormy16/stormy16.c +++ b/gcc/config/stormy16/stormy16.c @@ -2570,9 +2570,12 @@ xstormy16_expand_builtin(tree exp, rtx target, static void combine_bnp (rtx insn) { - int insn_code, regno, need_extend, mask; + int insn_code, regno, need_extend; + unsigned int mask; rtx cond, reg, and, load, qireg, mem; enum machine_mode load_mode = QImode; + enum machine_mode and_mode = QImode; + rtx shift = NULL_RTX; insn_code = recog_memoized (insn); if (insn_code != CODE_FOR_cbranchhi @@ -2613,15 +2616,14 @@ combine_bnp (rtx insn) for (and = prev_real_insn (insn); and; and = prev_real_insn (and)) { int and_code = recog_memoized (and); + if (and_code == CODE_FOR_extendqihi2 - && rtx_equal_p (XEXP (PATTERN (and), 0), reg) - && rtx_equal_p (XEXP (XEXP (PATTERN (and), 1), 0), qireg)) - { - break; - } + && rtx_equal_p (SET_DEST (PATTERN (and)), reg) + && rtx_equal_p (XEXP (SET_SRC (PATTERN (and)), 0), qireg)) + break; if (and_code == CODE_FOR_movhi_internal - && rtx_equal_p (XEXP (PATTERN (and), 0), reg)) + && rtx_equal_p (SET_DEST (PATTERN (and)), reg)) { /* This is for testing bit 15. */ and = insn; @@ -2630,6 +2632,7 @@ combine_bnp (rtx insn) if (reg_mentioned_p (reg, and)) return; + if (GET_CODE (and) != NOTE && GET_CODE (and) != INSN) return; @@ -2641,44 +2644,84 @@ combine_bnp (rtx insn) for (and = prev_real_insn (insn); and; and = prev_real_insn (and)) { if (recog_memoized (and) == CODE_FOR_andhi3 - && rtx_equal_p (XEXP (PATTERN (and), 0), reg) - && rtx_equal_p (XEXP (XEXP (PATTERN (and), 1), 0), reg)) - { - break; - } + && rtx_equal_p (SET_DEST (PATTERN (and)), reg) + && rtx_equal_p (XEXP (SET_SRC (PATTERN (and)), 0), reg)) + break; if (reg_mentioned_p (reg, and)) return; + if (GET_CODE (and) != NOTE && GET_CODE (and) != INSN) return; } + + if (and) + { + /* Some mis-optimisations by GCC can generate a RIGHT-SHIFT + followed by an AND like this: + + (parallel [(set (reg:HI r7) (lshiftrt:HI (reg:HI r7) (const_int 3))) + (clobber (reg:BI carry))] + + (set (reg:HI r7) (and:HI (reg:HI r7) (const_int 1))) + + Attempt to detect this here. */ + for (shift = prev_real_insn (and); shift; shift = prev_real_insn (shift)) + { + if (recog_memoized (shift) == CODE_FOR_lshrhi3 + && rtx_equal_p (SET_DEST (XVECEXP (PATTERN (shift), 0, 0)), reg) + && rtx_equal_p (XEXP (SET_SRC (XVECEXP (PATTERN (shift), 0, 0)), 0), reg)) + break; + + if (reg_mentioned_p (reg, shift) + || (GET_CODE (shift) != NOTE + && GET_CODE (shift) != INSN)) + { + shift = NULL_RTX; + break; + } + } + } } if (!and) return; - for (load = prev_real_insn (and); load; load = prev_real_insn (load)) + for (load = shift ? prev_real_insn (shift) : prev_real_insn (and); + load; + load = prev_real_insn (load)) { int load_code = recog_memoized (load); + if (load_code == CODE_FOR_movhi_internal - && rtx_equal_p (XEXP (PATTERN (load), 0), reg) - && xstormy16_below100_operand (XEXP (PATTERN (load), 1), HImode) - && ! MEM_VOLATILE_P (XEXP (PATTERN (load), 1))) + && rtx_equal_p (SET_DEST (PATTERN (load)), reg) + && xstormy16_below100_operand (SET_SRC (PATTERN (load)), HImode) + && ! MEM_VOLATILE_P (SET_SRC (PATTERN (load)))) { load_mode = HImode; break; } if (load_code == CODE_FOR_movqi_internal - && rtx_equal_p (XEXP (PATTERN (load), 0), qireg) - && xstormy16_below100_operand (XEXP (PATTERN (load), 1), QImode)) + && rtx_equal_p (SET_DEST (PATTERN (load)), qireg) + && xstormy16_below100_operand (SET_SRC (PATTERN (load)), QImode)) { load_mode = QImode; break; } - + + if (load_code == CODE_FOR_zero_extendqihi2 + && rtx_equal_p (SET_DEST (PATTERN (load)), reg) + && xstormy16_below100_operand (XEXP (SET_SRC (PATTERN (load)), 0), QImode)) + { + load_mode = QImode; + and_mode = HImode; + break; + } + if (reg_mentioned_p (reg, load)) return; + if (GET_CODE (load) != NOTE && GET_CODE (load) != INSN) return; @@ -2686,19 +2729,33 @@ combine_bnp (rtx insn) if (!load) return; - if (!need_extend) + mem = SET_SRC (PATTERN (load)); + + if (need_extend) { - if (!xstormy16_onebit_set_operand (XEXP (XEXP (PATTERN (and), 1), 1), load_mode)) - return; - mask = (int) INTVAL (XEXP (XEXP (PATTERN (and), 1), 1)); + mask = (load_mode == HImode) ? 0x8000 : 0x80; + + /* If the mem includes a zero-extend operation and we are + going to generate a sign-extend operation then move the + mem inside the zero-extend. */ + if (GET_CODE (mem) == ZERO_EXTEND) + mem = XEXP (mem, 0); } else - mask = (load_mode == HImode) ? 0x8000 : 0x80; + { + if (!xstormy16_onebit_set_operand (XEXP (SET_SRC (PATTERN (and)), 1), load_mode)) + return; + + mask = (int) INTVAL (XEXP (SET_SRC (PATTERN (and)), 1)); + + if (shift) + mask <<= INTVAL (XEXP (SET_SRC (XVECEXP (PATTERN (shift), 0, 0)), 1)); + } - mem = XEXP (PATTERN (load), 1); if (load_mode == HImode) { rtx addr = XEXP (mem, 0); + if (! (mask & 0xff)) { addr = plus_constant (addr, 1); @@ -2710,11 +2767,16 @@ combine_bnp (rtx insn) if (need_extend) XEXP (cond, 0) = gen_rtx_SIGN_EXTEND (HImode, mem); else - XEXP (cond, 0) = gen_rtx_AND (QImode, mem, GEN_INT (mask)); + XEXP (cond, 0) = gen_rtx_AND (and_mode, mem, GEN_INT (mask)); + INSN_CODE (insn) = -1; delete_insn (load); + if (and != insn) delete_insn (and); + + if (shift != NULL_RTX) + delete_insn (shift); } static void diff --git a/gcc/config/stormy16/stormy16.md b/gcc/config/stormy16/stormy16.md index 13474bee032..14dd7902188 100644 --- a/gcc/config/stormy16/stormy16.md +++ b/gcc/config/stormy16/stormy16.md @@ -1268,6 +1268,19 @@ [(set_attr "length" "4") (set_attr "psw_operand" "nop")]) +(define_insn "*bclrx3" + [(set (pc) + (if_then_else (eq:HI (and:HI (zero_extend:HI (match_operand:QI 1 "xstormy16_below100_operand" "W")) + (match_operand:HI 2 "immediate_operand" "i")) + (const_int 0)) + (label_ref (match_operand 0 "" "")) + (pc))) + (clobber (match_operand:BI 3 "" "=y"))] + "" + "bn %1,%B2,%l0" + [(set_attr "length" "4") + (set_attr "psw_operand" "nop")]) + (define_insn "*bclr7" [(set (pc) (if_then_else (xor:HI (lshiftrt:HI (subreg:HI @@ -1320,6 +1333,19 @@ [(set_attr "length" "4") (set_attr "psw_operand" "nop")]) +(define_insn "*bsetx3" + [(set (pc) + (if_then_else (ne:HI (and:HI (zero_extend:HI (match_operand:QI 1 "xstormy16_below100_operand" "W")) + (match_operand:HI 2 "immediate_operand" "i")) + (const_int 0)) + (label_ref (match_operand 0 "" "")) + (pc))) + (clobber (match_operand:BI 3 "" "=y"))] + "" + "bp %1,%B2,%l0" + [(set_attr "length" "4") + (set_attr "psw_operand" "nop")]) + (define_insn "*bset7" [(set (pc) (if_then_else (lshiftrt:HI (subreg:HI (match_operand:QI 1 "xstormy16_below100_operand" "W") 0) |