mirror of
git://gcc.gnu.org/git/gcc.git
synced 2025-04-06 05:20:26 +08:00
Partially revert recent H8 patch for conditional branches
So I'd forgotten an important tidbit on the H8 port. Specifically for a branch instruction, the target label must be operand 0 for the length computations. This really only affects the main conditional branch pattern. The other conditional branch patterns are split and ultimately funnel into the main pattern. This patch fixes the issue by partially reverting an earlier change. This issue didn't show up until late in the optimization work on cc0 removal of the H8 port, but was caught by the testsuite. So there's no new test. Built and regression tested H8 with this change, with and without the cc0 removal patches. gcc/ * config/h8300/jumpcall.md (branch_true, branch_false): Revert recent change. Ensure operand[0] is always the target label.
This commit is contained in:
parent
3c52cd517a
commit
fdd2fb1729
@ -37,46 +37,62 @@
|
||||
DONE;
|
||||
})
|
||||
|
||||
(define_insn "branch"
|
||||
(define_insn "branch_true"
|
||||
[(set (pc)
|
||||
(if_then_else (match_operator 2 "comparison_operator"
|
||||
(if_then_else (match_operator 1 "comparison_operator"
|
||||
[(cc0) (const_int 0)])
|
||||
(match_operand 0 "pc_or_label_operand" "")
|
||||
(match_operand 1 "pc_or_label_operand" "")))]
|
||||
"operands[0] == pc_rtx || operands[1] == pc_rtx"
|
||||
(label_ref (match_operand 0 "" ""))
|
||||
(pc)))]
|
||||
""
|
||||
{
|
||||
if ((cc_status.flags & CC_OVERFLOW_UNUSABLE) != 0
|
||||
&& (GET_CODE (operands[2]) == GT
|
||||
|| GET_CODE (operands[2]) == GE
|
||||
|| GET_CODE (operands[2]) == LE
|
||||
|| GET_CODE (operands[2]) == LT))
|
||||
&& (GET_CODE (operands[1]) == GT
|
||||
|| GET_CODE (operands[1]) == GE
|
||||
|| GET_CODE (operands[1]) == LE
|
||||
|| GET_CODE (operands[1]) == LT))
|
||||
{
|
||||
cc_status.flags &= ~CC_OVERFLOW_UNUSABLE;
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (operands[0] != pc_rtx)
|
||||
{
|
||||
if (get_attr_length (insn) == 2)
|
||||
return "b%j2 %l0";
|
||||
else if (get_attr_length (insn) == 4)
|
||||
return "b%j2 %l0:16";
|
||||
else
|
||||
return "b%k2 .Lh8BR%=\;jmp @%l0\\n.Lh8BR%=:";
|
||||
}
|
||||
if (get_attr_length (insn) == 2)
|
||||
return "b%j1 %l0";
|
||||
else if (get_attr_length (insn) == 4)
|
||||
return "b%j1 %l0:16";
|
||||
else
|
||||
{
|
||||
if (get_attr_length (insn) == 2)
|
||||
return "b%k2 %l1";
|
||||
else if (get_attr_length (insn) == 4)
|
||||
return "b%k2 %l1:16";
|
||||
else
|
||||
return "b%j2 .Lh8BR%=\;jmp @%l1\\n.Lh8BR%=:";
|
||||
}
|
||||
return "b%k1 .Lh8BR%=\;jmp @%l0\\n.Lh8BR%=:";
|
||||
}
|
||||
[(set_attr "type" "branch")
|
||||
(set_attr "cc" "none")])
|
||||
|
||||
(define_insn "branch_false"
|
||||
[(set (pc)
|
||||
(if_then_else (match_operator 1 "comparison_operator"
|
||||
[(cc0) (const_int 0)])
|
||||
(pc)
|
||||
(label_ref (match_operand 0 "" ""))))]
|
||||
""
|
||||
{
|
||||
if ((cc_status.flags & CC_OVERFLOW_UNUSABLE) != 0
|
||||
&& (GET_CODE (operands[1]) == GT
|
||||
|| GET_CODE (operands[1]) == GE
|
||||
|| GET_CODE (operands[1]) == LE
|
||||
|| GET_CODE (operands[1]) == LT))
|
||||
{
|
||||
cc_status.flags &= ~CC_OVERFLOW_UNUSABLE;
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (get_attr_length (insn) == 2)
|
||||
return "b%k1 %l0";
|
||||
else if (get_attr_length (insn) == 4)
|
||||
return "b%k1 %l0:16";
|
||||
else
|
||||
return "b%j1 .Lh8BR%=\;jmp @%l0\\n.Lh8BR%=:";
|
||||
}
|
||||
[(set_attr "type" "branch")
|
||||
(set_attr "cc" "none")])
|
||||
|
||||
;; The brabc/brabs patterns have been disabled because their length computation
|
||||
;; is horribly broken. When we call out to a function via a SYMBOL_REF we get
|
||||
;; bogus default and minimum lengths. The trick used by the PA port seems to
|
||||
|
Loading…
x
Reference in New Issue
Block a user