1554c2c6e9
From-SVN: r436
2730 lines
82 KiB
Markdown
2730 lines
82 KiB
Markdown
;;- Machine description for ROMP chip for GNU C compiler
|
||
;; Copyright (C) 1988, 1991 Free Software Foundation, Inc.
|
||
;; Contributed by Richard Kenner (kenner@nyu.edu)
|
||
|
||
;; This file is part of GNU CC.
|
||
|
||
;; GNU CC is free software; you can redistribute it and/or modify
|
||
;; it under the terms of the GNU General Public License as published by
|
||
;; the Free Software Foundation; either version 2, or (at your option)
|
||
;; any later version.
|
||
|
||
;; GNU CC is distributed in the hope that it will be useful,
|
||
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
;; 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.
|
||
|
||
|
||
;;- See file "rtl.def" for documentation on define_insn, match_*, et. al.
|
||
|
||
;; Define the attributes for the ROMP.
|
||
|
||
;; Insn type. Used to default other attribute values.
|
||
|
||
(define_attr "type"
|
||
"branch,return,fp,load,loadz,store,call,address,arith,compare,multi,misc"
|
||
(const_string "arith"))
|
||
|
||
;; Length in bytes.
|
||
|
||
(define_attr "length" ""
|
||
(cond [(eq_attr "type" "branch")
|
||
(if_then_else (and (ge (minus (pc) (match_dup 0))
|
||
(const_int -256))
|
||
(le (minus (pc) (match_dup 0))
|
||
(const_int 254)))
|
||
(const_int 2)
|
||
(const_int 4))
|
||
(eq_attr "type" "return") (const_int 2)
|
||
(eq_attr "type" "fp") (const_int 10)
|
||
(eq_attr "type" "call") (const_int 4)
|
||
(eq_attr "type" "load")
|
||
(cond [(match_operand 1 "short_memory_operand" "") (const_int 2)
|
||
(match_operand 1 "symbolic_memory_operand" "") (const_int 8)]
|
||
(const_int 4))
|
||
(eq_attr "type" "loadz")
|
||
(cond [(match_operand 1 "zero_memory_operand" "") (const_int 2)
|
||
(match_operand 1 "symbolic_memory_operand" "") (const_int 8)]
|
||
(const_string "4"))
|
||
(eq_attr "type" "store")
|
||
(cond [(match_operand 0 "short_memory_operand" "") (const_int 2)
|
||
(match_operand 0 "symbolic_memory_operand" "") (const_int 8)]
|
||
(const_int 4))]
|
||
(const_int 4)))
|
||
|
||
;; Whether insn can be placed in a delay slot.
|
||
|
||
(define_attr "in_delay_slot" "yes,no"
|
||
(cond [(eq_attr "length" "8,10,38") (const_string "no")
|
||
(eq_attr "type" "branch,return,call,multi") (const_string "no")]
|
||
(const_string "yes")))
|
||
|
||
;; Whether insn needs a delay slot.
|
||
(define_attr "needs_delay_slot" "yes,no"
|
||
(if_then_else (eq_attr "type" "branch,return,call")
|
||
(const_string "yes") (const_string "no")))
|
||
|
||
;; What insn does to the condition code.
|
||
|
||
(define_attr "cc"
|
||
"clobber,none,sets,change0,copy1to0,compare,tbit"
|
||
(cond [(eq_attr "type" "load,loadz") (const_string "change0")
|
||
(eq_attr "type" "store") (const_string "none")
|
||
(eq_attr "type" "fp,call") (const_string "clobber")
|
||
(eq_attr "type" "branch,return") (const_string "none")
|
||
(eq_attr "type" "address") (const_string "change0")
|
||
(eq_attr "type" "compare") (const_string "compare")
|
||
(eq_attr "type" "arith") (const_string "sets")]
|
||
(const_string "clobber")))
|
||
|
||
;; Define attributes for `asm' insns.
|
||
|
||
(define_asm_attributes [(set_attr "type" "misc")
|
||
(set_attr "length" "8")
|
||
(set_attr "in_delay_slot" "no")
|
||
(set_attr "cc" "clobber")])
|
||
|
||
;; Define the delay slot requirements for branches and calls. We don't have
|
||
;; any annulled insns.
|
||
;;
|
||
(define_delay (eq_attr "needs_delay_slot" "yes")
|
||
[(eq_attr "in_delay_slot" "yes") (nil) (nil)])
|
||
|
||
;; We cannot give a floating-point comparison a delay slot, even though it
|
||
;; could make use of it. This is because it would confuse next_cc0_user
|
||
;; to do so. Other fp insns can't get a delay slow because they set their
|
||
;; result and use their input after the delay slot insn is executed. This
|
||
;; isn't what reorg.c expects.
|
||
|
||
;; Define load & store delays. These were obtained by measurements done by
|
||
;; jfc@athena.mit.edu.
|
||
;;
|
||
;; In general, the memory unit can support at most two simultaneous operations.
|
||
;;
|
||
;; Loads take 5 cycles to return the data and can be pipelined up to the
|
||
;; limit of two simultaneous operations.
|
||
(define_function_unit "memory" 1 2 (eq_attr "type" "load,loadz") 5 0)
|
||
|
||
;; Stores do not return data, but tie up the memory unit for 2 cycles if the
|
||
;; next insn is also a store.
|
||
(define_function_unit "memory" 1 2 (eq_attr "type" "store") 1 2
|
||
[(eq_attr "type" "store")])
|
||
|
||
;; Move word instructions.
|
||
;;
|
||
;; If destination is memory but source is not register, force source to
|
||
;; register.
|
||
;;
|
||
;; If source is a constant that is too large to load in a single insn, build
|
||
;; it in two pieces.
|
||
;;
|
||
;; If destination is memory and source is a register, a temporary register
|
||
;; will be needed. In that case, make a PARALLEL of the SET and a
|
||
;; CLOBBER of a SCRATCH to allocate the required temporary.
|
||
;;
|
||
;; This temporary is ACTUALLY only needed when the destination is a
|
||
;; relocatable expression. For generating RTL, however, we always
|
||
;; place the CLOBBER. In insns where it is not needed, the SCRATCH will
|
||
;; not be allocated to a register.
|
||
;;
|
||
;; Also, avoid creating pseudo-registers or SCRATCH rtx's during reload as
|
||
;; they will not be correctly handled. We never need pseudos for that
|
||
;; case anyway.
|
||
;;
|
||
;; We do not use DEFINE_SPLIT for loading constants because the number
|
||
;; of cases in the resulting unsplit insn would be too high to deal
|
||
;; with practically.
|
||
(define_expand "movsi"
|
||
[(set (match_operand:SI 0 "general_operand" "")
|
||
(match_operand:SI 1 "general_operand" ""))]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
rtx op1 = operands[1];
|
||
|
||
if (GET_CODE (op1) == REG && REGNO (op1) == 16)
|
||
DONE;
|
||
|
||
if (GET_CODE (op0) == REG && REGNO (op0) == 16)
|
||
DONE;
|
||
|
||
if (GET_CODE (op0) == MEM && ! reload_in_progress)
|
||
{
|
||
emit_insn (gen_storesi (operands[0], force_reg (SImode, operands[1])));
|
||
DONE;
|
||
}
|
||
else if (GET_CODE (op1) == CONST_INT)
|
||
{
|
||
int const_val = INTVAL (op1);
|
||
|
||
/* Try a number of cases to see how to best load the constant. */
|
||
if ((const_val & 0xffff) == 0
|
||
|| (const_val & 0xffff0000) == 0
|
||
|| (unsigned) (const_val + 0x8000) < 0x10000)
|
||
/* Can do this in one insn, so generate it. */
|
||
;
|
||
else if (((- const_val) & 0xffff) == 0
|
||
|| ((- const_val) & 0xffff0000) == 0
|
||
|| (unsigned) ((- const_val) + 0x8000) < 0x10000)
|
||
{
|
||
/* Can do this by loading the negative constant and then negating. */
|
||
emit_move_insn (operands[0],
|
||
gen_rtx (CONST_INT, VOIDmode, - const_val));
|
||
emit_insn (gen_negsi2 (operands[0], operands[0]));
|
||
DONE;
|
||
}
|
||
else
|
||
/* Do this the long way. */
|
||
{
|
||
unsigned int high_part = const_val & 0xffff0000;
|
||
unsigned int low_part = const_val & 0xffff;
|
||
int i;
|
||
|
||
if (low_part >= 0x10 && exact_log2 (low_part) >= 0)
|
||
i = high_part, high_part = low_part, low_part = i;
|
||
|
||
emit_move_insn (operands[0],
|
||
gen_rtx (CONST_INT, VOIDmode, low_part));
|
||
emit_insn (gen_iorsi3 (operands[0], operands[0],
|
||
gen_rtx (CONST_INT, VOIDmode, high_part)));
|
||
DONE;
|
||
}
|
||
}
|
||
}")
|
||
|
||
;; Move from a symbolic memory location to a register is special. In this
|
||
;; case, we know in advance that the register cannot be r0, so we can improve
|
||
;; register allocation by treating it separately.
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=b")
|
||
(match_operand:SI 1 "symbolic_memory_operand" "m"))]
|
||
""
|
||
"load %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
;; Generic single-word move insn. We avoid the case where the destination is
|
||
;; a symbolic address, as that needs a temporary register.
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "reg_or_nonsymb_mem_operand" "=r,r,r,r,r,r,r,r,b,Q")
|
||
(match_operand:SI 1 "romp_operand" "rR,I,K,L,M,S,s,Q,m,r"))]
|
||
"register_operand (operands[0], SImode)
|
||
|| register_operand (operands[1], SImode)"
|
||
"@
|
||
cas %0,%1,r0
|
||
lis %0,%1
|
||
cal %0,%1(r0)
|
||
cal16 %0,%1(r0)
|
||
cau %0,%H1(r0)
|
||
ail %0,r14,%C1
|
||
get %0,$%1
|
||
l%M1 %0,%1
|
||
load %0,%1
|
||
st%M0 %1,%0"
|
||
[(set_attr "type" "address,address,address,address,address,arith,misc,load,load,store")
|
||
(set_attr "length" "2,2,4,4,4,4,8,*,*,*")])
|
||
|
||
(define_insn "storesi"
|
||
[(set (match_operand:SI 0 "memory_operand" "=Q,m")
|
||
(match_operand:SI 1 "register_operand" "r,r"))
|
||
(clobber (match_scratch:SI 2 "=X,&b"))]
|
||
""
|
||
"@
|
||
st%M0 %1,%0
|
||
store %1,%0,%2"
|
||
[(set_attr "type" "store")])
|
||
|
||
;; This pattern is used by reload when we store into a symbolic address. It
|
||
;; provides the temporary register required. This pattern is only used
|
||
;; when SECONDARY_OUTPUT_RELOAD_CLASS returns something other than
|
||
;; NO_REGS, so we need not have any predicates here.
|
||
|
||
(define_expand "reload_outsi"
|
||
[(parallel [(set (match_operand:SI 0 "symbolic_memory_operand" "=m")
|
||
(match_operand:SI 1 "" "r"))
|
||
(clobber (match_operand:SI 2 "" "=&b"))])]
|
||
""
|
||
"")
|
||
|
||
;; Now do the same for the QI move instructions.
|
||
(define_expand "movqi"
|
||
[(set (match_operand:QI 0 "general_operand" "")
|
||
(match_operand:QI 1 "general_operand" ""))]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
|
||
if (GET_CODE (op0) == MEM && ! reload_in_progress)
|
||
{
|
||
emit_insn (gen_storeqi (operands[0], force_reg (QImode, operands[1])));
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:QI 0 "register_operand" "=b")
|
||
(match_operand:QI 1 "symbolic_memory_operand" "m"))]
|
||
""
|
||
"loadc %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:QI 0 "reg_or_nonsymb_mem_operand" "=r,r,r,r,r,b,Q")
|
||
(match_operand:QI 1 "romp_operand" "r,I,n,s,Q,m,r"))]
|
||
"register_operand (operands[0], QImode)
|
||
|| register_operand (operands[1], QImode)"
|
||
"@
|
||
cas %0,%1,r0
|
||
lis %0,%1
|
||
cal %0,%L1(r0)
|
||
get %0,$%1
|
||
lc%M1 %0,%1
|
||
loadc %0,%1
|
||
stc%M0 %1,%0"
|
||
[(set_attr "type" "address,address,address,misc,load,load,store")
|
||
(set_attr "length" "2,2,4,8,*,*,*")])
|
||
|
||
(define_insn "storeqi"
|
||
[(set (match_operand:QI 0 "memory_operand" "=Q,m")
|
||
(match_operand:QI 1 "register_operand" "r,r"))
|
||
(clobber (match_scratch:SI 2 "=X,&b"))]
|
||
""
|
||
"@
|
||
stc%M0 %1,%0
|
||
storec %1,%0,%2"
|
||
[(set_attr "type" "store")])
|
||
|
||
(define_expand "reload_outqi"
|
||
[(set (match_operand:QI 0 "symbolic_memory_operand" "=m")
|
||
(match_operand:QI 1 "" "r"))
|
||
(match_operand:SI 2 "" "=&b")]
|
||
""
|
||
"")
|
||
|
||
;; Finally, the HI instructions.
|
||
(define_expand "movhi"
|
||
[(set (match_operand:HI 0 "general_operand" "")
|
||
(match_operand:HI 1 "general_operand" ""))]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
|
||
if (GET_CODE (op0) == MEM && ! reload_in_progress)
|
||
{
|
||
emit_insn (gen_storehi (operands[0], force_reg (HImode, operands[1])));
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:HI 0 "register_operand" "=b")
|
||
(match_operand:HI 1 "symbolic_memory_operand" "m"))]
|
||
""
|
||
"loadha %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:HI 0 "reg_or_nonsymb_mem_operand" "=r,r,r,r,r,b,Q")
|
||
(match_operand:HI 1 "romp_operand" "r,I,n,s,Q,m,r"))]
|
||
"register_operand (operands[0], HImode)
|
||
|| register_operand (operands[1], HImode)"
|
||
"@
|
||
cas %0,%1,r0
|
||
lis %0,%1
|
||
cal %0,%L1(r0)
|
||
get %0,$%1
|
||
lh%N1 %0,%1
|
||
loadh %0,%1
|
||
sth%M0 %1,%0"
|
||
[(set_attr "type" "address,address,address,misc,loadz,loadz,store")
|
||
(set_attr "length" "2,2,4,8,*,*,*")])
|
||
|
||
(define_insn "storehi"
|
||
[(set (match_operand:HI 0 "memory_operand" "=Q,m")
|
||
(match_operand:HI 1 "register_operand" "r,r"))
|
||
(clobber (match_scratch:SI 2 "=X,&b"))]
|
||
""
|
||
"@
|
||
sth%M0 %1,%0
|
||
storeh %1,%0,%2"
|
||
[(set_attr "type" "store")])
|
||
|
||
(define_expand "reload_outhi"
|
||
[(set (match_operand:HI 0 "symbolic_memory_operand" "=m")
|
||
(match_operand:HI 1 "" "r"))
|
||
(match_operand:SI 2 "" "=&b")]
|
||
""
|
||
"")
|
||
|
||
;; For DI move, if we have a constant, break the operation apart into
|
||
;; two SImode moves because the optimizer may be able to do a better job
|
||
;; with the resulting code.
|
||
;;
|
||
;; For memory stores, make the required pseudo for a temporary in case we
|
||
;; are storing into an absolute address.
|
||
;;
|
||
;; We need to be careful about the cases where the output is a register that is
|
||
;; the second register of the input.
|
||
|
||
(define_expand "movdi"
|
||
[(set (match_operand:DI 0 "general_operand" "")
|
||
(match_operand:DI 1 "general_operand" ""))]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
rtx op1 = operands[1];
|
||
|
||
if (CONSTANT_P (op1))
|
||
{
|
||
rtx insns;
|
||
|
||
start_sequence ();
|
||
emit_move_insn (operand_subword (op0, 0, 1, DImode),
|
||
operand_subword (op1, 0, 1, DImode));
|
||
emit_move_insn (operand_subword (op0, 1, 1, DImode),
|
||
operand_subword (op1, 1, 1, DImode));
|
||
insns = get_insns ();
|
||
end_sequence ();
|
||
|
||
emit_no_conflict_block (insns, op0, op1, 0, op1);
|
||
DONE;
|
||
}
|
||
|
||
if (GET_CODE (op0) == MEM && ! reload_in_progress)
|
||
{
|
||
emit_insn (gen_storedi (operands[0], force_reg (DImode, operands[1])));
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:DI 0 "reg_or_nonsymb_mem_operand" "=r,r,r,Q")
|
||
(match_operand:DI 1 "reg_or_mem_operand" "r,Q,m,r"))]
|
||
"register_operand (operands[0], DImode)
|
||
|| register_operand (operands[1], DImode)"
|
||
"*
|
||
{
|
||
switch (which_alternative)
|
||
{
|
||
case 0:
|
||
if (REGNO (operands[0]) == REGNO (operands[1]) + 1)
|
||
return \"cas %O0,%O1,r0\;cas %0,%1,r0\";
|
||
else
|
||
return \"cas %0,%1,r0\;cas %O0,%O1,r0\";
|
||
case 1:
|
||
/* Here we must see which word to load first. We default to the
|
||
low-order word unless it occurs in the address. */
|
||
if (refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1], 0))
|
||
return \"l%M1 %O0,%O1\;l%M1 %0,%1\";
|
||
else
|
||
return \"l%M1 %0,%1\;l%M1 %O0,%O1\";
|
||
case 2:
|
||
return \"get %O0,$%1\;ls %0,0(%O0)\;ls %O0,4(%O0)\";
|
||
case 3:
|
||
return \"st%M0 %1,%0\;st%M0 %O1,%O0\";
|
||
}
|
||
}"
|
||
[(set_attr "type" "multi")
|
||
(set_attr "cc" "change0,change0,change0,none")
|
||
(set_attr "length" "4,12,8,8")])
|
||
|
||
(define_insn "storedi"
|
||
[(set (match_operand:DI 0 "memory_operand" "=Q,m")
|
||
(match_operand:DI 1 "register_operand" "r,r"))
|
||
(clobber (match_scratch:SI 2 "=X,&b"))]
|
||
""
|
||
"@
|
||
st%M0 %1,%0\;st%M0 %O1,%O0
|
||
get %2,$%0\;sts %1,0(%2)\;sts %O1,4(%2)"
|
||
[(set_attr "type" "multi,multi")
|
||
(set_attr "cc" "none,none")
|
||
(set_attr "length" "8,12")])
|
||
|
||
(define_expand "reload_outdi"
|
||
[(set (match_operand:DI 0 "symbolic_memory_operand" "=m")
|
||
(match_operand:DI 1 "" "r"))
|
||
(match_operand:SI 2 "" "=&b")]
|
||
""
|
||
"")
|
||
|
||
;; Split symbolic memory operands differently. We first load the address
|
||
;; into a register and then do the two loads or stores. We can only do
|
||
;; this if operand_subword won't produce a SUBREG, which is only when
|
||
;; operands[0] is a hard register. Thus, these won't be used during the
|
||
;; first insn scheduling pass.
|
||
(define_split
|
||
[(set (match_operand:DI 0 "register_operand" "")
|
||
(match_operand:DI 1 "symbolic_memory_operand" ""))]
|
||
"GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) < FIRST_PSEUDO_REGISTER"
|
||
[(set (match_dup 2) (match_dup 3))
|
||
(set (match_dup 4) (match_dup 5))
|
||
(set (match_dup 6) (match_dup 7))]
|
||
"
|
||
{ operands[2] = operand_subword (operands[0], 1, 0, DImode);
|
||
operands[3] = XEXP (operands[1], 0);
|
||
operands[4] = operand_subword (operands[0], 0, 0, DImode);
|
||
operands[5] = gen_rtx (MEM, SImode, operands[2]);
|
||
operands[6] = operands[2];
|
||
operands[7] = gen_rtx (MEM, SImode,
|
||
gen_rtx (PLUS, SImode, operands[2],
|
||
gen_rtx (CONST_INT, VOIDmode, 4)));
|
||
|
||
if (operands[2] == 0 || operands[4] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
(define_split
|
||
[(set (match_operand:DI 0 "symbolic_memory_operand" "")
|
||
(match_operand:DI 1 "register_operand" ""))
|
||
(clobber (match_operand:SI 2 "register_operand" ""))]
|
||
"GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) < FIRST_PSEUDO_REGISTER"
|
||
[(set (match_dup 2) (match_dup 3))
|
||
(set (match_dup 4) (match_dup 5))
|
||
(set (match_dup 6) (match_dup 7))]
|
||
"
|
||
{ operands[3] = XEXP (operands[0], 0);
|
||
operands[4] = gen_rtx (MEM, SImode, operands[2]);
|
||
operands[5] = operand_subword (operands[1], 0, 0, DImode);
|
||
operands[6] = gen_rtx (MEM, SImode,
|
||
gen_rtx (PLUS, SImode, operands[2],
|
||
gen_rtx (CONST_INT, VOIDmode, 4)));
|
||
operands[7] = operand_subword (operands[1], 1, 0, DImode);
|
||
|
||
if (operands[5] == 0 || operands[7] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
;; If the output is a register and the input is memory, we have to be careful
|
||
;; and see which word needs to be loaded first.
|
||
;;
|
||
;; Note that this case doesn't have a CLOBBER. Therefore, we must either
|
||
;; be after reload or operand[0] must not be a MEM. So we don't need a
|
||
;; CLOBBER on the new insns either.
|
||
;;
|
||
;; Due to a bug in sched.c, we do not want to split this insn if both
|
||
;; operands are registers and they overlap unless reload has completed.
|
||
(define_split
|
||
[(set (match_operand:DI 0 "general_operand" "")
|
||
(match_operand:DI 1 "general_operand" ""))]
|
||
"! symbolic_memory_operand (operands[0], DImode)
|
||
&& ! symbolic_memory_operand (operands[1], DImode)
|
||
&& ! (GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) >= FIRST_PSEUDO_REGISTER)
|
||
&& ! (GET_CODE (operands[1]) == REG
|
||
&& REGNO (operands[1]) >= FIRST_PSEUDO_REGISTER)
|
||
&& ! (GET_CODE (operands[0]) == REG && GET_CODE (operands[1]) == REG
|
||
&& ! reload_completed
|
||
&& reg_overlap_mentioned_p (operands[0], operands[1]))"
|
||
[(set (match_dup 2) (match_dup 3))
|
||
(set (match_dup 4) (match_dup 5))]
|
||
"
|
||
{ if (GET_CODE (operands[0]) != REG
|
||
|| ! refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1], 0))
|
||
{
|
||
operands[2] = operand_subword (operands[0], 0, 0, DImode);
|
||
operands[3] = operand_subword (operands[1], 0, 0, DImode);
|
||
operands[4] = operand_subword (operands[0], 1, 0, DImode);
|
||
operands[5] = operand_subword (operands[1], 1, 0, DImode);
|
||
}
|
||
else
|
||
{
|
||
operands[2] = operand_subword (operands[0], 1, 0, DImode);
|
||
operands[3] = operand_subword (operands[1], 1, 0, DImode);
|
||
operands[4] = operand_subword (operands[0], 0, 0, DImode);
|
||
operands[5] = operand_subword (operands[1], 0, 0, DImode);
|
||
}
|
||
|
||
if (operands[2] == 0 || operands[3] == 0
|
||
|| operands[4] == 0 || operands[5] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
(define_split
|
||
[(set (match_operand:DI 0 "general_operand" "")
|
||
(match_operand:DI 1 "general_operand" ""))
|
||
(clobber (match_operand:SI 6 "register_operand" ""))]
|
||
"! symbolic_memory_operand (operands[0], DImode)
|
||
&& ! symbolic_memory_operand (operands[1], DImode)
|
||
&& ! (GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) >= FIRST_PSEUDO_REGISTER)
|
||
&& ! (GET_CODE (operands[1]) == REG
|
||
&& REGNO (operands[1]) >= FIRST_PSEUDO_REGISTER)"
|
||
[(parallel [(set (match_dup 2) (match_dup 3))
|
||
(clobber (match_dup 7))])
|
||
(parallel [(set (match_dup 4) (match_dup 5))
|
||
(clobber (match_dup 8))])]
|
||
"
|
||
{ if (GET_CODE (operands[0]) != REG
|
||
|| ! refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1], 0))
|
||
{
|
||
operands[2] = operand_subword (operands[0], 0, 0, DImode);
|
||
operands[3] = operand_subword (operands[1], 0, 0, DImode);
|
||
operands[4] = operand_subword (operands[0], 1, 0, DImode);
|
||
operands[5] = operand_subword (operands[1], 1, 0, DImode);
|
||
}
|
||
else
|
||
{
|
||
operands[2] = operand_subword (operands[0], 1, 0, DImode);
|
||
operands[3] = operand_subword (operands[1], 1, 0, DImode);
|
||
operands[4] = operand_subword (operands[0], 0, 0, DImode);
|
||
operands[5] = operand_subword (operands[1], 0, 0, DImode);
|
||
}
|
||
|
||
if (operands[2] == 0 || operands[3] == 0
|
||
|| operands[4] == 0 || operands[5] == 0)
|
||
FAIL;
|
||
|
||
/* We must be sure to make two different SCRATCH operands, since they
|
||
are not allowed to be shared. After reload, however, we only have
|
||
a SCRATCH if we won't use the operand, so it is allowed to share it
|
||
then. */
|
||
if (reload_completed || GET_CODE (operands[6]) != SCRATCH)
|
||
operands[7] = operands[8] = operands[6];
|
||
else
|
||
{
|
||
operands[7] = gen_rtx (SCRATCH, SImode);
|
||
operands[8] = gen_rtx (SCRATCH, SImode);
|
||
}
|
||
}")
|
||
|
||
;; Define move insns for SF, and DF.
|
||
;;
|
||
;; For register-register copies or a copy of something to itself, emit a
|
||
;; single SET insn since it will likely be optimized away.
|
||
;;
|
||
;; Otherwise, emit a floating-point move operation unless both input and
|
||
;; output are either constant, memory, or a non-floating-point hard register.
|
||
(define_expand "movdf"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(match_operand:DF 1 "general_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
rtx op1 = operands[1];
|
||
|
||
if (op0 == op1)
|
||
{
|
||
emit_insn (gen_rtx (SET, VOIDmode, op0, op1));
|
||
DONE;
|
||
}
|
||
|
||
if ((GET_CODE (op0) == MEM
|
||
|| (GET_CODE (op0) == REG && REGNO (op0) < FIRST_PSEUDO_REGISTER
|
||
&& ! FP_REGNO_P (REGNO (op0))))
|
||
&& (GET_CODE (op1) == MEM
|
||
|| GET_CODE (op1) == CONST_DOUBLE
|
||
|| (GET_CODE (op1) == REG && REGNO (op1) < FIRST_PSEUDO_REGISTER
|
||
&& ! FP_REGNO_P (REGNO (op1)) && ! rtx_equal_p (op0, op1))))
|
||
{
|
||
rtx insns;
|
||
|
||
if (GET_CODE (op1) == CONST_DOUBLE)
|
||
op1 = force_const_mem (DFmode, op1);
|
||
|
||
start_sequence ();
|
||
if (GET_CODE (operands[0]) != REG
|
||
|| ! refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1]), 0)
|
||
{
|
||
emit_move_insn (operand_subword (op0, 0, 1, DFmode),
|
||
operand_subword_force (op1, 0, DFmode));
|
||
emit_move_insn (operand_subword (op0, 1, 1, DFmode),
|
||
operand_subword_force (op1, 1, DFmode));
|
||
}
|
||
else
|
||
{
|
||
emit_move_insn (operand_subword (op0, 1, 1, DFmode),
|
||
operand_subword_force (op1, 1, DFmode));
|
||
emit_move_insn (operand_subword (op0, 0, 1, DFmode),
|
||
operand_subword_force (op1, 0, DFmode));
|
||
}
|
||
|
||
insns = get_insns ();
|
||
end_sequence ();
|
||
|
||
emit_no_conflict_block (insns, op0, op1, 0, op1);
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_expand "movsf"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(match_operand:SF 1 "general_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"
|
||
{ rtx op0 = operands[0];
|
||
rtx op1 = operands[1];
|
||
|
||
if (op0 == op1)
|
||
{
|
||
emit_insn (gen_rtx (SET, VOIDmode, op0, op1));
|
||
DONE;
|
||
}
|
||
|
||
if ((GET_CODE (op0) == MEM
|
||
|| (GET_CODE (op0) == REG && REGNO (op0) < FIRST_PSEUDO_REGISTER
|
||
&& ! FP_REGNO_P (REGNO (op0))))
|
||
&& (GET_CODE (op1) == MEM
|
||
|| GET_CODE (op1) == CONST_DOUBLE
|
||
|| (GET_CODE (op1) == REG && REGNO (op1) < FIRST_PSEUDO_REGISTER
|
||
&& ! FP_REGNO_P (REGNO (op1)))))
|
||
{
|
||
rtx last;
|
||
|
||
if (GET_CODE (op1) == CONST_DOUBLE)
|
||
op1 = force_const_mem (SFmode, op1);
|
||
|
||
last = emit_move_insn (operand_subword (op0, 0, 1, SFmode),
|
||
operand_subword_force (op1, 0, SFmode));
|
||
|
||
REG_NOTES (last) = gen_rtx (EXPR_LIST, REG_EQUAL, op1, REG_NOTES (last));
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
;; Define the move insns for SF and DF. Check for all general regs
|
||
;; in the FP insns and make them non-FP if so. Do the same if the input and
|
||
;; output are the same (the insn will be deleted in this case and we don't
|
||
;; want to think there are FP insns when there might not be).
|
||
(define_insn ""
|
||
[(set (match_operand:SF 0 "general_operand" "=*frg")
|
||
(match_dup 0))]
|
||
""
|
||
"nopr r0"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "2")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SF 0 "general_operand" "=r,*fr,r,r,Q,m,frg")
|
||
(match_operand:SF 1 "general_operand" "r,0,Q,m,r,r,frg"))
|
||
(clobber (match_operand:SI 2 "reg_0_operand" "=&z,z,z,z,z,z,z"))
|
||
(clobber (match_operand:SI 3 "reg_15_operand" "=&t,t,t,t,t,t,t"))]
|
||
""
|
||
"*
|
||
{ switch (which_alternative)
|
||
{
|
||
case 0:
|
||
return \"cas %0,%1,r0\";
|
||
case 1:
|
||
return \"nopr r0\";
|
||
case 2:
|
||
return \"l%M1 %0,%1\";
|
||
case 3:
|
||
return \"load %0,%1\";
|
||
case 4:
|
||
return \"st%M0 %1,%0\";
|
||
case 5:
|
||
return \"store %1,%0,%3\";
|
||
default:
|
||
return output_fpop (SET, operands[0], operands[1], 0, insn);
|
||
}
|
||
}"
|
||
[(set_attr "type" "address,address,load,load,store,store,fp")
|
||
(set_attr "length" "2,2,*,*,*,*,*")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:DF 0 "general_operand" "=*frg")
|
||
(match_dup 0))]
|
||
""
|
||
"nopr r0"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "2")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:DF 0 "general_operand" "=r,*fr,r,r,Q,m,frg")
|
||
(match_operand:DF 1 "general_operand" "r,0,Q,m,r,r,*frg"))
|
||
(clobber (match_operand:SI 2 "reg_0_operand" "=&z,z,z,z,z,z,z"))
|
||
(clobber (match_operand:SI 3 "reg_15_operand" "=&t,t,t,t,t,t,t"))]
|
||
""
|
||
"*
|
||
{ switch (which_alternative)
|
||
{
|
||
case 0:
|
||
if (REGNO (operands[0]) == REGNO (operands[1]) + 1)
|
||
return \"cas %O0,%O1,r0\;cas %0,%1,r0\";
|
||
else
|
||
return \"cas %0,%1,r0\;cas %O0,%O1,r0\";
|
||
case 1:
|
||
return \"nopr r0\";
|
||
case 2:
|
||
/* Here we must see which word to load first. We default to the
|
||
low-order word unless it occurs in the address. */
|
||
if (refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1], 0))
|
||
return \"l%M1 %O0,%O1\;l%M1 %0,%1\";
|
||
else
|
||
return \"l%M1 %0,%1\;l%M1 %O0,%O1\";
|
||
case 3:
|
||
return \"get %3,$%1\;ls %0,0(%3)\;ls %O0,4(%3)\";
|
||
case 4:
|
||
return \"st%M0 %1,%0\;st%M0 %O1,%O0\";
|
||
case 5:
|
||
return \"get %3,$%0\;sts %1,0(%3)\;sts %O1,4(%3)\";
|
||
default:
|
||
return output_fpop (SET, operands[0], operands[1], 0, insn);
|
||
}
|
||
}"
|
||
[(set_attr "type" "address,multi,multi,multi,multi,multi,fp")
|
||
(set_attr "length" "2,4,*,*,*,*,*")])
|
||
|
||
;; Split all the above cases that involve multiple insns and no floating-point
|
||
;; data block. If before reload, we can make a SCRATCH. Otherwise, use
|
||
;; register 15.
|
||
|
||
(define_split
|
||
[(set (match_operand:DF 0 "register_operand" "")
|
||
(match_operand:DF 1 "symbolic_memory_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
"GET_CODE (operands[0]) == REG && REGNO (operands[0]) < 16"
|
||
[(set (reg:SI 15) (match_dup 2))
|
||
(set (match_dup 3) (match_dup 4))
|
||
(set (match_dup 5) (match_dup 6))]
|
||
"
|
||
{ operands[2] = XEXP (operands[1], 0);
|
||
operands[3] = operand_subword (operands[0], 0, 0, DFmode);
|
||
operands[4] = gen_rtx (MEM, SImode, gen_rtx (REG, SImode, 15));
|
||
operands[5] = operand_subword (operands[0], 0, 1, DFmode);
|
||
operands[6] = gen_rtx (MEM, SImode,
|
||
gen_rtx (PLUS, SImode, gen_rtx (REG, SImode, 15),
|
||
gen_rtx (CONST_INT, VOIDmode, 4)));
|
||
|
||
if (operands[3] == 0 || operands[5] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
(define_split
|
||
[(set (match_operand:DF 0 "symbolic_memory_operand" "")
|
||
(match_operand:DF 1 "register_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
"GET_CODE (operands[1]) == REG && REGNO (operands[1]) < 16"
|
||
[(set (reg:SI 15) (match_dup 2))
|
||
(set (match_dup 3) (match_dup 4))
|
||
(set (match_dup 5) (match_dup 6))]
|
||
"
|
||
{ operands[2] = XEXP (operands[0], 0);
|
||
operands[3] = gen_rtx (MEM, SImode, gen_rtx (REG, SImode, 15));
|
||
operands[4] = operand_subword (operands[1], 0, 0, DFmode);
|
||
operands[5] = gen_rtx (MEM, SImode,
|
||
gen_rtx (PLUS, SImode, gen_rtx (REG, SImode, 15),
|
||
gen_rtx (CONST_INT, VOIDmode, 4)));
|
||
operands[6] = operand_subword (operands[1], 1, 0, DFmode);
|
||
|
||
if (operands[4] == 0 || operands[6] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
;; If the output is a register and the input is memory, we have to be careful
|
||
;; and see which word needs to be loaded first. We also cannot to the
|
||
;; split if the input is a constant because it would result in invalid
|
||
;; insns. When the output is a MEM, we must put a CLOBBER on each of the
|
||
;; resulting insn, when it is not a MEM, we must not.
|
||
(define_split
|
||
[(set (match_operand:DF 0 "memory_operand" "")
|
||
(match_operand:DF 1 "register_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
"GET_CODE (operands[1]) == REG && REGNO (operands[1]) < 15"
|
||
[(parallel [(set (match_dup 2) (match_dup 3))
|
||
(clobber (match_dup 6))])
|
||
(parallel [(set (match_dup 4) (match_dup 5))
|
||
(clobber (match_dup 7))])]
|
||
"
|
||
{ operands[2] = operand_subword (operands[0], 0, 0, DFmode);
|
||
operands[3] = operand_subword (operands[1], 0, 0, DFmode);
|
||
operands[4] = operand_subword (operands[0], 1, 0, DFmode);
|
||
operands[5] = operand_subword (operands[1], 1, 0, DFmode);
|
||
|
||
if (operands[2] == 0 || operands[3] == 0
|
||
|| operands[4] == 0 || operands[5] == 0)
|
||
FAIL;
|
||
|
||
if (reload_completed)
|
||
operands[6] = operands[7] = gen_rtx (REG, SImode, 15);
|
||
else
|
||
{
|
||
operands[6] = gen_rtx (SCRATCH, SImode);
|
||
operands[7] = gen_rtx (SCRATCH, SImode);
|
||
}
|
||
}")
|
||
|
||
(define_split
|
||
[(set (match_operand:DF 0 "nonmemory_operand" "")
|
||
(match_operand:DF 1 "general_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
"! symbolic_memory_operand (operands[1], DFmode)
|
||
&& GET_CODE (operands[1]) != CONST_DOUBLE
|
||
&& (GET_CODE (operands[0]) != REG || REGNO (operands[0]) < 15)
|
||
&& (GET_CODE (operands[1]) != REG || REGNO (operands[1]) < 15)
|
||
&& (GET_CODE (operands[0]) == REG || GET_CODE (operands[1]) == REG)"
|
||
[(set (match_dup 2) (match_dup 3))
|
||
(set (match_dup 4) (match_dup 5))]
|
||
"
|
||
{ if (GET_CODE (operands[0]) != REG
|
||
|| ! refers_to_regno_p (REGNO (operands[0]), REGNO (operands[0]) + 1,
|
||
operands[1], 0))
|
||
{
|
||
operands[2] = operand_subword (operands[0], 0, 0, DFmode);
|
||
operands[3] = operand_subword (operands[1], 0, 0, DFmode);
|
||
operands[4] = operand_subword (operands[0], 1, 0, DFmode);
|
||
operands[5] = operand_subword (operands[1], 1, 0, DFmode);
|
||
}
|
||
else
|
||
{
|
||
operands[2] = operand_subword (operands[0], 1, 0, DFmode);
|
||
operands[3] = operand_subword (operands[1], 1, 0, DFmode);
|
||
operands[4] = operand_subword (operands[0], 0, 0, DFmode);
|
||
operands[5] = operand_subword (operands[1], 0, 0, DFmode);
|
||
}
|
||
|
||
if (operands[2] == 0 || operands[3] == 0
|
||
|| operands[4] == 0 || operands[5] == 0)
|
||
FAIL;
|
||
}")
|
||
|
||
;; Conversions from one integer mode to another.
|
||
;; It is possible sometimes to sign- or zero-extend while fetching from memory.
|
||
;;
|
||
;; First, sign-extensions:
|
||
(define_expand "extendhisi2"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(sign_extend:SI (match_operand:HI 1 "register_operand" "")))]
|
||
""
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=b")
|
||
(sign_extend:SI (match_operand:HI 1 "symbolic_memory_operand" "m")))]
|
||
""
|
||
"loadha %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,b")
|
||
(sign_extend:SI (match_operand:HI 1 "nonimmediate_operand" "r,Q,m")))]
|
||
""
|
||
"@
|
||
exts %0,%1
|
||
lha%M1 %0,%1
|
||
loadha %0,%1"
|
||
[(set_attr "type" "arith,load,load")
|
||
(set_attr "length" "2,*,*")])
|
||
|
||
(define_expand "extendqisi2"
|
||
[(set (match_dup 2)
|
||
(ashift:SI (match_operand:QI 1 "register_operand" "")
|
||
(const_int 24)))
|
||
(set (match_operand:SI 0 "register_operand" "")
|
||
(ashiftrt:SI (match_dup 2)
|
||
(const_int 24)))]
|
||
""
|
||
"
|
||
{ operands[1] = gen_lowpart (SImode, operands[1]);
|
||
operands[2] = gen_reg_rtx (SImode); }")
|
||
|
||
(define_expand "extendqihi2"
|
||
[(set (match_dup 2)
|
||
(ashift:SI (match_operand:QI 1 "register_operand" "")
|
||
(const_int 24)))
|
||
(set (match_operand:HI 0 "register_operand" "")
|
||
(ashiftrt:SI (match_dup 2)
|
||
(const_int 24)))]
|
||
""
|
||
"
|
||
{ operands[0] = gen_lowpart (SImode, operands[0]);
|
||
operands[1] = gen_lowpart (SImode, operands[1]);
|
||
operands[2] = gen_reg_rtx (SImode); }")
|
||
|
||
;; Define peepholes to eliminate an instruction when we are doing a sign
|
||
;; extension but cannot clobber the input.
|
||
;;
|
||
;; In this case we will shift left 24 bits, but need a copy first. The shift
|
||
;; can be replaced by a "mc03" instruction, but this can only be done if
|
||
;; followed by the right shift of 24 or more bits.
|
||
(define_peephole
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(subreg:SI (match_operand:QI 1 "register_operand" "") 0))
|
||
(set (match_dup 0)
|
||
(ashift:SI (match_dup 0)
|
||
(const_int 24)))
|
||
(set (match_dup 0)
|
||
(ashiftrt:SI (match_dup 0)
|
||
(match_operand:SI 2 "const_int_operand" "")))]
|
||
"INTVAL (operands[2]) >= 24"
|
||
"mc03 %0,%1\;sari16 %0,%S2"
|
||
[(set_attr "type" "multi")
|
||
(set_attr "length" "4")
|
||
(set_attr "cc" "sets")])
|
||
|
||
;; Now zero extensions:
|
||
(define_expand "zero_extendhisi2"
|
||
[(set (match_operand:SI 0 "register_operand" "b")
|
||
(zero_extend:SI (match_operand:HI 1 "register_operand" "")))]
|
||
""
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=b")
|
||
(zero_extend:SI (match_operand:HI 1 "symbolic_memory_operand" "m")))]
|
||
""
|
||
"loadh %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,b")
|
||
(zero_extend:SI (match_operand:HI 1 "nonimmediate_operand" "r,Q,m")))]
|
||
""
|
||
"@
|
||
nilz %0,%1,65535
|
||
lh%N1 %0,%1
|
||
loadh %0,%1"
|
||
[(set_attr "type" "arith,loadz,load")])
|
||
|
||
(define_expand "zero_extendqisi2"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(zero_extend:SI (match_operand:QI 1 "register_operand" "")))]
|
||
""
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=b")
|
||
(zero_extend:SI (match_operand:QI 1 "symbolic_memory_operand" "m")))]
|
||
""
|
||
"loadc %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,b")
|
||
(zero_extend:SI (match_operand:QI 1 "nonimmediate_operand" "r,Q,m")))]
|
||
""
|
||
"@
|
||
nilz %0,%1,255
|
||
lc%M1 %0,%1
|
||
loadc %0,%1"
|
||
[(set_attr "type" "arith,load,load")])
|
||
|
||
(define_expand "zero_extendqihi2"
|
||
[(set (match_operand:HI 0 "register_operand" "")
|
||
(zero_extend:HI (match_operand:QI 1 "register_operand" "")))]
|
||
""
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:HI 0 "register_operand" "=b")
|
||
(zero_extend:HI (match_operand:QI 1 "symbolic_memory_operand" "m")))]
|
||
""
|
||
"loadc %0,%1"
|
||
[(set_attr "type" "load")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:HI 0 "register_operand" "=r,r,b")
|
||
(zero_extend:HI (match_operand:QI 1 "nonimmediate_operand" "r,Q,m")))]
|
||
""
|
||
"@
|
||
nilz %0,%1,255
|
||
lc%M1 %0,%1
|
||
loadc %0,%1"
|
||
[(set_attr "type" "arith,load,load")])
|
||
|
||
;; Various extract and insertion operations.
|
||
(define_expand "extzv"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(zero_extract:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "const_int_operand" "")
|
||
(match_operand:SI 3 "const_int_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) != CONST_INT || INTVAL (operands[2]) != 8)
|
||
FAIL;
|
||
|
||
if (GET_CODE (operands[3]) != CONST_INT)
|
||
FAIL;
|
||
|
||
if (INTVAL (operands[3]) != 0 && INTVAL (operands[3]) != 8
|
||
&& INTVAL (operands[3]) != 16 && INTVAL (operands[3]) != 24)
|
||
FAIL;
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=&r")
|
||
(zero_extract:SI (match_operand:SI 1 "register_operand" "r")
|
||
(const_int 8)
|
||
(match_operand:SI 2 "const_int_operand" "n")))]
|
||
"(INTVAL (operands[2]) & 7) == 0"
|
||
"lis %0,0\;mc3%B2 %0,%1"
|
||
[(set_attr "type" "multi")
|
||
(set_attr "cc" "change0")])
|
||
|
||
(define_split
|
||
[(set (match_operand:SI 0 "register_operand" "=&r")
|
||
(zero_extract:SI (match_operand:SI 1 "register_operand" "r")
|
||
(const_int 8)
|
||
(match_operand:SI 2 "const_int_operand" "n")))]
|
||
"(INTVAL (operands[2]) & 7) == 0"
|
||
[(set (match_dup 0) (const_int 0))
|
||
(set (zero_extract:SI (match_dup 0) (const_int 8) (const_int 24))
|
||
(zero_extract:SI (match_dup 1) (const_int 8) (match_dup 2)))]
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (zero_extract:SI (match_operand:SI 0 "register_operand" "+r")
|
||
(const_int 8)
|
||
(const_int 24))
|
||
(zero_extract:SI (match_operand:SI 1 "register_operand" "r")
|
||
(const_int 8)
|
||
(match_operand:SI 2 "const_int_operand" "n")))]
|
||
"(INTVAL (operands[2]) & 7) == 0"
|
||
"mc3%B2 %0,%1"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "2")])
|
||
|
||
(define_expand "insv"
|
||
[(set (zero_extract:SI (match_operand:SI 0 "register_operand" "")
|
||
(match_operand:SI 1 "const_int_operand" "")
|
||
(match_operand:SI 2 "const_int_operand" ""))
|
||
(match_operand:SI 3 "register_operand" ""))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) != CONST_INT)
|
||
FAIL;
|
||
|
||
if (GET_CODE (operands[1]) != CONST_INT)
|
||
FAIL;
|
||
|
||
if (INTVAL (operands[1]) == 1)
|
||
{
|
||
emit_insn (gen_bit_insv (operands[0], operands[1], operands[2],
|
||
operands[3]));
|
||
DONE;
|
||
}
|
||
else if (INTVAL (operands[1]) == 8
|
||
&& (INTVAL (operands[2]) % 8 == 0))
|
||
; /* Accept aligned byte-wide field. */
|
||
else
|
||
FAIL;
|
||
}")
|
||
|
||
;; For a single-bit insert, it is better to explicitly generate references
|
||
;; to the T bit. We will call the T bit "CC0" because it can be clobbered
|
||
;; by some CC0 sets (single-bit tests).
|
||
|
||
(define_expand "bit_insv"
|
||
[(set (cc0)
|
||
(zero_extract:SI (match_operand:SI 3 "register_operand" "")
|
||
(const_int 1)
|
||
(const_int 31)))
|
||
(parallel [(set (zero_extract:SI (match_operand:SI 0 "register_operand" "")
|
||
(match_operand:SI 1 "const_int_operand" "")
|
||
(match_operand:SI 2 "const_int_operand" ""))
|
||
(ne (cc0) (const_int 0)))
|
||
(clobber (match_scratch:SI 4 ""))])]
|
||
""
|
||
"")
|
||
|
||
(define_insn ""
|
||
[(set (zero_extract:SI (match_operand:SI 0 "register_operand" "+r")
|
||
(const_int 8)
|
||
(match_operand:SI 1 "const_int_operand" "n"))
|
||
(match_operand:SI 2 "register_operand" "r"))]
|
||
"(INTVAL (operands[1]) & 7) == 0"
|
||
"mc%B1%.3 %0,%2"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "2")])
|
||
|
||
;; This pattern cannot have any input reloads since if references CC0.
|
||
;; So we have to add code to support memory, which is the only other
|
||
;; thing that a "register_operand" can become. There is still a problem
|
||
;; if the address isn't valid and *it* needs a reload, but there is no
|
||
;; way to solve that problem, so let's hope it never happens.
|
||
|
||
(define_insn ""
|
||
[(set (zero_extract:SI (match_operand:SI 0 "register_operand" "+r,m")
|
||
(const_int 1)
|
||
(match_operand:SI 1 "const_int_operand" "n,m"))
|
||
(ne (cc0) (const_int 0)))
|
||
(clobber (match_scratch:SI 2 "=X,b"))]
|
||
""
|
||
"@
|
||
mftbi%t1 %0,%S1
|
||
l%M0 %2,%0\;mftb%t1 %2,%S1\;st%M0 %2,%0"
|
||
[(set_attr "type" "*,multi")
|
||
(set_attr "cc" "none,none")
|
||
(set_attr "length" "2,10")])
|
||
|
||
;; Arithmetic instructions. First, add and subtract.
|
||
;;
|
||
;; It may be that the second input is either large or small enough that
|
||
;; the operation cannot be done in a single insn. In that case, emit two.
|
||
(define_expand "addsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(plus:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "nonmemory_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) == CONST_INT
|
||
&& (unsigned) (INTVAL (operands[2]) + 0x8000) >= 0x10000
|
||
&& (INTVAL (operands[2]) & 0xffff) != 0)
|
||
{
|
||
int low = INTVAL (operands[2]) & 0xffff;
|
||
int high = (unsigned) INTVAL (operands[2]) >> 16;
|
||
|
||
if (low & 0x8000)
|
||
high++, low |= 0xffff0000;
|
||
|
||
emit_insn (gen_addsi3 (operands[0], operands[1],
|
||
gen_rtx (CONST_INT, VOIDmode, high << 16)));
|
||
operands[1] = operands[0];
|
||
operands[2] = gen_rtx (CONST_INT, VOIDmode, low);
|
||
}
|
||
}")
|
||
|
||
;; Put the insn to add a symbolic constant to a register separately to
|
||
;; improve register allocation since it has different register requirements.
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=b")
|
||
(plus:SI (match_operand:SI 1 "register_operand" "%b")
|
||
(match_operand:SI 2 "romp_symbolic_operand" "s")))]
|
||
""
|
||
"get %0,$%2(%1)"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "8")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,r,r,r,r,b")
|
||
(plus:SI (match_operand:SI 1 "reg_or_add_operand" "%0,0,r,b,0,r,b")
|
||
(match_operand:SI 2 "reg_or_add_operand" "I,J,K,M,r,b,s")))]
|
||
"register_operand (operands[1], SImode)
|
||
|| register_operand (operands[2], SImode)"
|
||
"@
|
||
ais %0,%2
|
||
sis %0,%n2
|
||
ail %0,%1,%2
|
||
cau %0,%H2(%1)
|
||
a %0,%2
|
||
cas %0,%1,%2
|
||
get %0,$%2(%1)"
|
||
[(set_attr "type" "arith,arith,arith,address,arith,address,misc")
|
||
(set_attr "length" "2,2,4,4,2,2,8")])
|
||
|
||
;; Now subtract.
|
||
;;
|
||
;; 1. If third operand is constant integer, convert it to add of the negative
|
||
;; of that integer.
|
||
;; 2. If the second operand is not a valid constant integer, force it into a
|
||
;; register.
|
||
(define_expand "subsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(minus:SI (match_operand:SI 1 "reg_or_any_cint_operand" "")
|
||
(match_operand:SI 2 "reg_or_any_cint_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands [2]) == CONST_INT)
|
||
{
|
||
emit_insn (gen_addsi3 (operands[0], operands[1],
|
||
gen_rtx (CONST_INT,
|
||
VOIDmode, - INTVAL (operands[2]))));
|
||
DONE;
|
||
}
|
||
else
|
||
operands[2] = force_reg (SImode, operands[2]);
|
||
|
||
if (GET_CODE (operands[1]) != CONST_INT
|
||
|| (unsigned) (INTVAL (operands[1]) + 0x8000) >= 0x10000)
|
||
operands[1] = force_reg (SImode, operands[1]);
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,r")
|
||
(minus:SI (match_operand:SI 1 "reg_or_D_operand" "K,0,r")
|
||
(match_operand:SI 2 "register_operand" "r,r,0")))]
|
||
""
|
||
"@
|
||
sfi %0,%2,%1
|
||
s %0,%2
|
||
sf %0,%1"
|
||
[(set_attr "length" "4,2,2")])
|
||
|
||
;; Multiply either calls a special RT routine or is done in-line, depending
|
||
;; on the value of a -m flag.
|
||
;;
|
||
;; First define the way we call the subroutine.
|
||
(define_expand "mulsi3_subr"
|
||
[(set (reg:SI 2) (match_operand:SI 1 "register_operand" ""))
|
||
(set (reg:SI 3) (match_operand:SI 2 "register_operand" ""))
|
||
(parallel [(set (reg:SI 2) (mult:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])
|
||
(set (match_operand:SI 0 "register_operand" "")
|
||
(reg:SI 2))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "mulsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(mult:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "register_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (! TARGET_IN_LINE_MUL)
|
||
{
|
||
emit_insn (gen_mulsi3_subr (operands[0], operands[1], operands[2]));
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
;; Define the patterns to match.
|
||
;; We would like to provide a delay slot for the insns that call internal
|
||
;; routines, but doing so is risky since reorg will think that the use of
|
||
;; r2 and r3 is completed in the insn needing the delay slot. Also, it
|
||
;; won't know that the cc will be clobbered. So take the safe approach
|
||
;; and don't give them delay slots.
|
||
(define_insn ""
|
||
[(set (reg:SI 2)
|
||
(mult:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
"! TARGET_IN_LINE_MUL"
|
||
"bali%# r15,lmul$$"
|
||
[(set_attr "type" "misc")
|
||
(set_attr "in_delay_slot" "no")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=&r")
|
||
(mult:SI (match_operand:SI 1 "register_operand" "%r")
|
||
(match_operand:SI 2 "register_operand" "r")))]
|
||
"TARGET_IN_LINE_MUL"
|
||
"*
|
||
{ return output_in_line_mul (); }"
|
||
[(set_attr "length" "38")
|
||
(set_attr "type" "multi")])
|
||
|
||
;; Handle divide and modulus. The same function returns both values,
|
||
;; so use divmodsi4. This divides arg 1 by arg 2 with quotient to go
|
||
;; into arg 0 and remainder in arg 3.
|
||
;;
|
||
;; We want to put REG_EQUAL notes for the two outputs. So we need a
|
||
;; function to do everything else.
|
||
(define_expand "divmodsi4_doit"
|
||
[(set (reg:SI 2)
|
||
(match_operand:SI 0 "register_operand" ""))
|
||
(set (reg:SI 3)
|
||
(match_operand:SI 1 "register_operand" ""))
|
||
(parallel [(set (reg:SI 2) (div:SI (reg:SI 2) (reg:SI 3)))
|
||
(set (reg:SI 3) (mod:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "divmodsi4"
|
||
[(parallel [(set (match_operand:SI 0 "register_operand" "")
|
||
(div:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "register_operand" "")))
|
||
(set (match_operand:SI 3 "register_operand" "")
|
||
(mod:SI (match_dup 1) (match_dup 2)))])]
|
||
""
|
||
"
|
||
{
|
||
rtx insn;
|
||
|
||
emit_insn (gen_divmodsi4_doit (operands[1], operands[2]));
|
||
insn = emit_move_insn (operands[0], gen_rtx (REG, SImode, 2));
|
||
REG_NOTES (insn) = gen_rtx (EXPR_LIST, REG_EQUAL,
|
||
gen_rtx (DIV, SImode, operands[1],
|
||
operands[2]),
|
||
REG_NOTES (insn));
|
||
insn = emit_move_insn (operands[3], gen_rtx (REG, SImode, 3));
|
||
REG_NOTES (insn) = gen_rtx (EXPR_LIST, REG_EQUAL,
|
||
gen_rtx (MOD, SImode, operands[1],
|
||
operands[2]),
|
||
REG_NOTES (insn));
|
||
DONE;
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (reg:SI 2)
|
||
(div:SI (reg:SI 2) (reg:SI 3)))
|
||
(set (reg:SI 3)
|
||
(mod:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
""
|
||
"bali%# r15,ldiv$$"
|
||
[(set_attr "type" "misc")
|
||
(set_attr "in_delay_slot" "no")])
|
||
|
||
;; Similarly for unsigned divide.
|
||
(define_expand "udivmodsi4_doit"
|
||
[(set (reg:SI 2)
|
||
(match_operand:SI 0 "register_operand" ""))
|
||
(set (reg:SI 3)
|
||
(match_operand:SI 1 "register_operand" ""))
|
||
(parallel [(set (reg:SI 2) (udiv:SI (reg:SI 2) (reg:SI 3)))
|
||
(set (reg:SI 3) (umod:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "udivmodsi4"
|
||
[(parallel [(set (match_operand:SI 0 "register_operand" "")
|
||
(udiv:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "register_operand" "")))
|
||
(set (match_operand:SI 3 "register_operand" "")
|
||
(umod:SI (match_dup 1) (match_dup 2)))])]
|
||
""
|
||
"
|
||
{
|
||
rtx insn;
|
||
|
||
emit_insn (gen_udivmodsi4_doit (operands[1], operands[2]));
|
||
insn = emit_move_insn (operands[0], gen_rtx (REG, SImode, 2));
|
||
REG_NOTES (insn) = gen_rtx (EXPR_LIST, REG_EQUAL,
|
||
gen_rtx (UDIV, SImode, operands[1],
|
||
operands[2]),
|
||
REG_NOTES (insn));
|
||
insn = emit_move_insn (operands[3], gen_rtx (REG, SImode, 3));
|
||
REG_NOTES (insn) = gen_rtx (EXPR_LIST, REG_EQUAL,
|
||
gen_rtx (UMOD, SImode, operands[1],
|
||
operands[2]),
|
||
REG_NOTES (insn));
|
||
DONE;
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (reg:SI 2)
|
||
(udiv:SI (reg:SI 2) (reg:SI 3)))
|
||
(set (reg:SI 3)
|
||
(umod:SI (reg:SI 2) (reg:SI 3)))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))]
|
||
""
|
||
"bali%# r15,uldiv$$"
|
||
[(set_attr "type" "misc")
|
||
(set_attr "in_delay_slot" "no")])
|
||
|
||
;; Define DImode arithmetic operations.
|
||
;;
|
||
;; It is possible to do certain adds and subtracts with constants in a single
|
||
;; insn, but it doesn't seem worth the trouble.
|
||
;;
|
||
;; Don't use DEFINE_SPLIT on these because the dependency on CC can't be
|
||
;; easily tracked in that case!
|
||
(define_insn "adddi3"
|
||
[(set (match_operand:DI 0 "register_operand" "=r")
|
||
(plus:DI (match_operand:DI 1 "register_operand" "%0")
|
||
(match_operand:DI 2 "register_operand" "r")))]
|
||
""
|
||
"a %O0,%O2\;ae %0,%2"
|
||
[(set_attr "type" "multi")])
|
||
|
||
(define_insn "subdi3"
|
||
[(set (match_operand:DI 0 "register_operand" "=r")
|
||
(minus:DI (match_operand:DI 1 "register_operand" "0")
|
||
(match_operand:DI 2 "register_operand" "r")))]
|
||
""
|
||
"s %O0,%O2\;se %0,%2"
|
||
[(set_attr "type" "multi")])
|
||
|
||
(define_insn "negdi2"
|
||
[(set (match_operand:DI 0 "register_operand" "=r,&r")
|
||
(neg:DI (match_operand:DI 1 "register_operand" "0,r")))]
|
||
""
|
||
"twoc %O0,%O1\;onec %0,%1\;aei %0,%0,0"
|
||
[(set_attr "type" "multi")
|
||
(set_attr "length" "8")])
|
||
|
||
;; Unary arithmetic operations.
|
||
(define_insn "abssi2"
|
||
[(set (match_operand:SI 0 "register_operand" "=r")
|
||
(abs:SI (match_operand:SI 1 "register_operand" "r")))]
|
||
""
|
||
"abs %0,%1"
|
||
[(set_attr "length" "2")])
|
||
|
||
(define_insn "negsi2"
|
||
[(set (match_operand:SI 0 "register_operand" "=r")
|
||
(neg:SI (match_operand:SI 1 "register_operand" "r")))]
|
||
""
|
||
"twoc %0,%1"
|
||
[(set_attr "length" "2")])
|
||
|
||
(define_insn "one_cmplsi2"
|
||
[(set (match_operand:SI 0 "register_operand" "=r")
|
||
(not:SI (match_operand:SI 1 "register_operand" "r")))]
|
||
""
|
||
"onec %0,%1"
|
||
[(set_attr "length" "2")])
|
||
|
||
|
||
;; Logical insns: AND, IOR, and XOR
|
||
;;
|
||
;; If the operation is being performed on a 32-bit constant such that
|
||
;; it cannot be done in one insn, do it in two. We may lose a bit on
|
||
;; CSE in pathological cases, but it seems better doing it this way.
|
||
(define_expand "andsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(and:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "reg_or_any_cint_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) == CONST_INT)
|
||
{
|
||
int top = (unsigned) INTVAL (operands[2]) >> 16;
|
||
int bottom = INTVAL (operands[2]) & 0xffff;
|
||
|
||
if (top != 0 && top != 0xffff && bottom != 0 && bottom != 0xffff)
|
||
{
|
||
emit_insn (gen_andsi3 (operands[0], operands[1],
|
||
gen_rtx (CONST_INT, VOIDmode,
|
||
(top << 16) | 0xffff)));
|
||
operands[1] = operands[0];
|
||
operands[2] = gen_rtx (CONST_INT, VOIDmode, 0xffff0000 | bottom);
|
||
}
|
||
}
|
||
}");
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,r")
|
||
(and:SI (match_operand:SI 1 "reg_or_and_operand" "%0,r,0")
|
||
(match_operand:SI 2 "reg_or_and_operand" "P,LMO,r")))]
|
||
"register_operand (operands[1], SImode)
|
||
|| register_operand (operands[2], SImode)"
|
||
"@
|
||
clrb%k2 %0,%b2
|
||
ni%z2 %0,%1,%Z2
|
||
n %0,%2"
|
||
[(set_attr "length" "2,4,2")])
|
||
|
||
;; logical OR (IOR)
|
||
(define_expand "iorsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(ior:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "reg_or_any_cint_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) == CONST_INT)
|
||
{
|
||
int top = (unsigned) INTVAL (operands[2]) >> 16;
|
||
int bottom = INTVAL (operands[2]) & 0xffff;
|
||
|
||
if (top != 0 && bottom != 0)
|
||
{
|
||
emit_insn (gen_iorsi3 (operands[0], operands[1],
|
||
gen_rtx (CONST_INT, VOIDmode, (top << 16))));
|
||
operands[1] = operands[0];
|
||
operands[2] = gen_rtx (CONST_INT, VOIDmode, bottom);
|
||
}
|
||
}
|
||
}");
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r,r")
|
||
(ior:SI (match_operand:SI 1 "reg_or_cint_operand" "%0,r,0")
|
||
(match_operand:SI 2 "reg_or_cint_operand" "N,LM,r")))]
|
||
"register_operand (operands[1], SImode)
|
||
|| register_operand (operands[2], SImode)"
|
||
"@
|
||
setb%h2 %0,%b2
|
||
oi%h2 %0,%1,%H2
|
||
o %0,%2"
|
||
[(set_attr "length" "2,4,2")])
|
||
|
||
;; exclusive-or (XOR)
|
||
(define_expand "xorsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "")
|
||
(xor:SI (match_operand:SI 1 "register_operand" "")
|
||
(match_operand:SI 2 "reg_or_any_cint_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[2]) == CONST_INT)
|
||
{
|
||
int top = (unsigned) INTVAL (operands[2]) >> 16;
|
||
int bottom = INTVAL (operands[2]) & 0xffff;
|
||
|
||
if (top == 0xffff && bottom == 0xffff)
|
||
{
|
||
emit_insn (gen_one_cmplsi2 (operands[0], operands[1]));
|
||
DONE;
|
||
}
|
||
else if (top != 0 && bottom != 0)
|
||
{
|
||
emit_insn (gen_xorsi3 (operands[0], operands[1],
|
||
gen_rtx (CONST_INT, VOIDmode, (top << 16))));
|
||
operands[1] = operands[0];
|
||
operands[2] = gen_rtx (CONST_INT, VOIDmode, bottom);
|
||
}
|
||
}
|
||
}");
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r")
|
||
(xor:SI (match_operand:SI 1 "reg_or_cint_operand" "%r,0")
|
||
(match_operand:SI 2 "reg_or_cint_operand" "LM,r")))]
|
||
"register_operand (operands[1], SImode)
|
||
|| register_operand (operands[2], SImode)"
|
||
"@
|
||
xi%h2 %0,%1,%H2
|
||
x %0,%2"
|
||
[(set_attr "length" "4,2")])
|
||
|
||
;; Various shift insns
|
||
(define_insn "ashrsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r")
|
||
(ashiftrt:SI (match_operand:SI 1 "register_operand" "0,0")
|
||
(match_operand:QI 2 "reg_or_cint_operand" "r,n")))]
|
||
""
|
||
"@
|
||
sar %0,%2
|
||
sari%s2 %0,%S2"
|
||
[(set_attr "length" "2")])
|
||
|
||
(define_insn "lshrsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r")
|
||
(lshiftrt:SI (match_operand:SI 1 "register_operand" "0,0")
|
||
(match_operand:QI 2 "reg_or_cint_operand" "r,n")))]
|
||
""
|
||
"@
|
||
sr %0,%2
|
||
sri%s2 %0,%S2"
|
||
[(set_attr "length" "2")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "register_operand" "=r")
|
||
(ashift:SI (match_operand:SI 1 "register_operand" "b")
|
||
(const_int 1)))]
|
||
""
|
||
"cas %0,%1,%1"
|
||
[(set_attr "length" "2")
|
||
(set_attr "type" "address")])
|
||
|
||
(define_insn "ashlsi3"
|
||
[(set (match_operand:SI 0 "register_operand" "=r,r")
|
||
(ashift:SI (match_operand:SI 1 "register_operand" "0,0")
|
||
(match_operand:QI 2 "reg_or_cint_operand" "r,n")))]
|
||
""
|
||
"@
|
||
sl %0,%2
|
||
sli%s2 %0,%S2"
|
||
[(set_attr "length" "2")])
|
||
|
||
;; Function call insns:
|
||
;;
|
||
;; On the ROMP, &fcn is actually a pointer to the data area, which is passed
|
||
;; to the function in r0. &.fcn is the actual starting address of the
|
||
;; function. Also, the word at &fcn contains &.fcn.
|
||
;;
|
||
;; For both functions that do and don't return values, there are two cases:
|
||
;; where the function's address is a constant, and where it isn't.
|
||
;;
|
||
;; Operand 1 (2 for `call_value') is the number of arguments and is not used.
|
||
(define_expand "call"
|
||
[(use (reg:SI 0))
|
||
(parallel [(call (mem:SI (match_operand:SI 0 "address_operand" ""))
|
||
(match_operand 1 "" ""))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[0]) != MEM || GET_CODE (operands[1]) != CONST_INT)
|
||
abort();
|
||
|
||
operands[0] = XEXP (operands[0], 0);
|
||
if (GET_CODE (operands[0]) == SYMBOL_REF)
|
||
{
|
||
extern rtx get_symref ();
|
||
char *real_fcnname =
|
||
(char *) alloca (strlen (XSTR (operands[0], 0)) + 2);
|
||
|
||
/* Copy the data area address to r0. */
|
||
emit_move_insn (gen_rtx (REG, SImode, 0),
|
||
force_reg (SImode, operands[0]));
|
||
strcpy (real_fcnname, \".\");
|
||
strcat (real_fcnname, XSTR (operands[0], 0));
|
||
operands[0] = get_symref (real_fcnname);
|
||
}
|
||
else
|
||
{
|
||
rtx data_access;
|
||
|
||
emit_move_insn (gen_rtx (REG, SImode, 0),
|
||
force_reg (SImode, operands[0]));
|
||
data_access = gen_rtx (MEM, SImode, operands[0]);
|
||
RTX_UNCHANGING_P (data_access) = 1;
|
||
operands[0] = copy_to_reg (data_access);
|
||
}
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(call (mem:SI (match_operand:SI 0 "register_operand" "b"))
|
||
(match_operand 1 "" "g"))
|
||
(clobber (reg:SI 15))]
|
||
""
|
||
"balr%# r15,%0"
|
||
[(set_attr "type" "call")
|
||
(set_attr "length" "2")])
|
||
|
||
(define_insn ""
|
||
[(call (mem:SI (match_operand:SI 0 "romp_symbolic_operand" "i"))
|
||
(match_operand 1 "" "g"))
|
||
(clobber (reg:SI 15))]
|
||
"GET_CODE (operands[0]) == SYMBOL_REF"
|
||
"bali%# r15,%0"
|
||
[(set_attr "type" "call")])
|
||
|
||
;; Call a function and return a value.
|
||
(define_expand "call_value"
|
||
[(use (reg:SI 0))
|
||
(parallel [(set (match_operand 0 "" "=fg")
|
||
(call (mem:SI (match_operand:SI 1 "address_operand" ""))
|
||
(match_operand 2 "" "")))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"
|
||
{
|
||
if (GET_CODE (operands[1]) != MEM || GET_CODE (operands[2]) != CONST_INT)
|
||
abort();
|
||
|
||
operands[1] = XEXP (operands[1], 0);
|
||
if (GET_CODE (operands[1]) == SYMBOL_REF)
|
||
{
|
||
extern rtx get_symref ();
|
||
char *real_fcnname =
|
||
(char *) alloca (strlen (XSTR (operands[1], 0)) + 2);
|
||
|
||
/* Copy the data area address to r0. */
|
||
emit_move_insn (gen_rtx (REG, SImode, 0),
|
||
force_reg (SImode, operands[1]));
|
||
strcpy (real_fcnname, \".\");
|
||
strcat (real_fcnname, XSTR (operands[1], 0));
|
||
operands[1] = get_symref (real_fcnname);
|
||
}
|
||
else
|
||
{
|
||
rtx data_access;
|
||
|
||
emit_move_insn (gen_rtx (REG, SImode, 0),
|
||
force_reg (SImode, operands[1]));
|
||
data_access = gen_rtx (MEM, SImode, operands[1]);
|
||
RTX_UNCHANGING_P (data_access) = 1;
|
||
operands[1] = copy_to_reg (data_access);
|
||
}
|
||
}")
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "" "=fg")
|
||
(call (mem:SI (match_operand:SI 1 "register_operand" "b"))
|
||
(match_operand 2 "" "g")))
|
||
(clobber (reg:SI 15))]
|
||
""
|
||
"balr%# r15,%1"
|
||
[(set_attr "length" "2")
|
||
(set_attr "type" "call")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "" "=fg")
|
||
(call (mem:SI (match_operand:SI 1 "romp_symbolic_operand" "i"))
|
||
(match_operand 2 "" "g")))
|
||
(clobber (reg:SI 15))]
|
||
"GET_CODE (operands[1]) == SYMBOL_REF"
|
||
"bali%# r15,%1"
|
||
[(set_attr "type" "call")])
|
||
|
||
;; No operation insn.
|
||
(define_insn "nop"
|
||
[(const_int 0)]
|
||
""
|
||
"nopr r0"
|
||
[(set_attr "type" "address")
|
||
(set_attr "length" "2")
|
||
(set_attr "cc" "none")])
|
||
|
||
;; Here are the floating-point operations.
|
||
;;
|
||
;; Start by providing DEFINE_EXPAND for each operation.
|
||
;; The insns will be handled with MATCH_OPERATOR; the methodology will be
|
||
;; discussed below.
|
||
|
||
;; First the conversion operations.
|
||
|
||
(define_expand "truncdfsf2"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(float_truncate:SF (match_operand:DF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "extendsfdf2"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(float_extend:DF (match_operand:SF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "floatsisf2"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(float:SF (match_operand:SI 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "floatsidf2"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(float:DF (match_operand:SI 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "fix_truncsfsi2"
|
||
[(parallel [(set (match_operand:SI 0 "general_operand" "")
|
||
(fix:SI (match_operand:SF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "fix_truncdfsi2"
|
||
[(parallel [(set (match_operand:SI 0 "general_operand" "")
|
||
(fix:SI (match_operand:DF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
;; Now the binary operations.
|
||
|
||
(define_expand "addsf3"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(plus:SF (match_operand:SF 1 "general_operand" "")
|
||
(match_operand:SF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "adddf3"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(plus:DF (match_operand:DF 1 "general_operand" "")
|
||
(match_operand:DF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "subsf3"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(minus:SF (match_operand:SF 1 "general_operand" "")
|
||
(match_operand:SF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "subdf3"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(minus:DF (match_operand:DF 1 "general_operand" "")
|
||
(match_operand:DF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "mulsf3"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(mult:SF (match_operand:SF 1 "general_operand" "")
|
||
(match_operand:SF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "muldf3"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(mult:DF (match_operand:DF 1 "general_operand" "")
|
||
(match_operand:DF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "divsf3"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(div:SF (match_operand:SF 1 "general_operand" "")
|
||
(match_operand:SF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "divdf3"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(div:DF (match_operand:DF 1 "general_operand" "")
|
||
(match_operand:DF 2 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
;; Unary floating-point operations.
|
||
;;
|
||
;; Negations can be done without floating-point, since this is IEEE.
|
||
;; But we cannot do this if an operand is a hard FP register, since
|
||
;; the SUBREG we create would not be valid.
|
||
(define_expand "negsf2"
|
||
[(set (match_operand:SF 0 "register_operand" "")
|
||
(neg:SF (match_operand:SF 1 "register_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (! (GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) < FIRST_PSEUDO_REGISTER
|
||
&& FP_REGNO_P (REGNO (operands[0])))
|
||
&& ! (GET_CODE (operands[1]) == REG
|
||
&& REGNO (operands[1]) < FIRST_PSEUDO_REGISTER
|
||
&& FP_REGNO_P (REGNO (operands[1]))))
|
||
{
|
||
rtx result;
|
||
rtx target = operand_subword (operands[0], 0, 1, SFmode);
|
||
|
||
result = expand_binop (SImode, xor_optab,
|
||
operand_subword_force (operands[1], 0, SFmode),
|
||
gen_rtx (CONST_INT, VOIDmode, 0x80000000),
|
||
target, 0, OPTAB_WIDEN);
|
||
if (result == 0)
|
||
abort ();
|
||
|
||
if (result != target)
|
||
emit_move_insn (result, target);
|
||
|
||
/* Make a place for REG_EQUAL. */
|
||
emit_move_insn (operands[0], operands[0]);
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_expand "negdf2"
|
||
[(set (match_operand:DF 0 "register_operand" "")
|
||
(neg:DF (match_operand:DF 1 "register_operand" "")))]
|
||
""
|
||
"
|
||
{
|
||
if (! (GET_CODE (operands[0]) == REG
|
||
&& REGNO (operands[0]) < FIRST_PSEUDO_REGISTER
|
||
&& FP_REGNO_P (REGNO (operands[0])))
|
||
&& ! (GET_CODE (operands[1]) == REG
|
||
&& REGNO (operands[1]) < FIRST_PSEUDO_REGISTER
|
||
&& FP_REGNO_P (REGNO (operands[1]))))
|
||
{
|
||
rtx result;
|
||
rtx target = operand_subword (operands[0], 0, 1, DFmode);
|
||
rtx insns;
|
||
|
||
start_sequence ();
|
||
result = expand_binop (SImode, xor_optab,
|
||
operand_subword_force (operands[1], 0, DFmode),
|
||
gen_rtx (CONST_INT, VOIDmode, 0x80000000),
|
||
target, 0, OPTAB_WIDEN);
|
||
if (result == 0)
|
||
abort ();
|
||
|
||
if (result != target)
|
||
emit_move_insn (result, target);
|
||
|
||
emit_move_insn (operand_subword (operands[0], 1, 1, DFmode),
|
||
operand_subword_force (operands[1], 1, DFmode));
|
||
|
||
insns = get_insns ();
|
||
end_sequence ();
|
||
|
||
emit_no_conflict_block (insns, operands[0], operands[1], 0, 0);
|
||
DONE;
|
||
}
|
||
}")
|
||
|
||
(define_expand "abssf2"
|
||
[(parallel [(set (match_operand:SF 0 "general_operand" "")
|
||
(abs:SF (match_operand:SF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "absdf2"
|
||
[(parallel [(set (match_operand:DF 0 "general_operand" "")
|
||
(abs:DF (match_operand:DF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
;; Any floating-point operation can be either SFmode or DFmode, and each
|
||
;; operand (including the output) can be either a normal operand or a
|
||
;; conversion from a normal operand.
|
||
;;
|
||
;; We use MATCH_OPERATOR to match a floating-point binary or unary operator
|
||
;; and input and output conversions. So we need 2^N patterns for each type
|
||
;; of operation, where N is the number of operands, including the output.
|
||
;; There are thus a total of 14 patterns, 8 for binary operations, 4 for
|
||
;; unary operations and two for conversion/move operations (only one
|
||
;; operand can have a conversion for move operations). In addition, we have
|
||
;; to be careful that a floating-point reload register doesn't get allocated
|
||
;; for an integer. We take care of this for inputs with PREFERRED_RELOAD_CLASS
|
||
;; but need to have two different constraints for outputs. This means that
|
||
;; we have to duplicate each pattern where the output could be an integer.
|
||
;; This adds another 7 patterns, for a total of 21.
|
||
|
||
;; Start with conversion operations (moves are done above).
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operand 2 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 3 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 4 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (SET, operands[0], operands[2], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operand 2 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 3 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 4 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (SET, operands[0], operands[2], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
;; Next, binary floating-point operations.
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_binary"
|
||
[(match_operand 2 "general_operand" "frg")
|
||
(match_operand 3 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 4 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 5 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[2], operands[3])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0],
|
||
operands[2], operands[3], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_binary"
|
||
[(match_operand 2 "general_operand" "frg")
|
||
(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[2], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0],
|
||
operands[2], operands[4], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_binary"
|
||
[(match_operator 2 "float_conversion"
|
||
[(match_operand 3 "general_operand" "frg")])
|
||
(match_operand 4 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[3], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0],
|
||
operands[3], operands[4], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_binary"
|
||
[(match_operator 2 "float_conversion"
|
||
[(match_operand 3 "general_operand" "frg")])
|
||
(match_operator 4 "float_conversion"
|
||
[(match_operand 5 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 6 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 7 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[3], operands[5])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0],
|
||
operands[3], operands[5], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operand 3 "general_operand" "frg")
|
||
(match_operand 4 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[3], operands[4], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operand 3 "general_operand" "frg")
|
||
(match_operand 4 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[3], operands[4], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operand 3 "general_operand" "frg")
|
||
(match_operator 4 "float_conversion"
|
||
[(match_operand 5 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 6 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 7 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[3], operands[5], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operand 3 "general_operand" "frg")
|
||
(match_operator 4 "float_conversion"
|
||
[(match_operand 5 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 6 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 7 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], operands[4])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[3], operands[5], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])
|
||
(match_operand 5 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 6 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 7 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], operands[5])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[4], operands[5], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])
|
||
(match_operand 5 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 6 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 7 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], operands[5])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[4], operands[5], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])
|
||
(match_operator 5 "float_conversion"
|
||
[(match_operand 6 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 7 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 8 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], operands[6])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[4], operands[6], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_binary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])
|
||
(match_operator 5 "float_conversion"
|
||
[(match_operand 6 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 7 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 8 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], operands[6])"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0],
|
||
operands[4], operands[6], insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
;; Unary floating-point operations.
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_unary"
|
||
[(match_operand 2 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 3 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 4 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[2], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0], operands[2],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_unary"
|
||
[(match_operator 2 "float_conversion"
|
||
[(match_operand 3 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 4 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 5 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[1]), operands[3], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[1]), operands[0], operands[3],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_unary"
|
||
[(match_operand 3 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 4 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 5 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0], operands[3],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_unary"
|
||
[(match_operand 3 "general_operand" "frg")])]))
|
||
(clobber (match_operand:SI 4 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 5 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[3], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0], operands[3],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand:SI 0 "general_operand" "=g")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_unary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0], operands[4],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
(define_insn ""
|
||
[(set (match_operand 0 "general_operand" "=frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operator 2 "float_unary"
|
||
[(match_operator 3 "float_conversion"
|
||
[(match_operand 4 "general_operand" "frg")])])]))
|
||
(clobber (match_operand:SI 5 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 6 "reg_15_operand" "=&t"))]
|
||
"check_precision (GET_MODE (operands[2]), operands[4], 0)"
|
||
"*
|
||
{ return output_fpop (GET_CODE (operands[2]), operands[0], operands[4],
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")])
|
||
|
||
;; Compare insns are next. Note that the ROMP has two types of compares,
|
||
;; signed & unsigned, and one type of branch. Use the routine
|
||
;; `next_insn_tests_no_unsigned' to see which type to use.
|
||
(define_expand "tstsi"
|
||
[(set (cc0)
|
||
(match_operand:SI 0 "register_operand" "r"))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "cmpsi"
|
||
[(set (cc0)
|
||
(compare (match_operand:SI 0 "register_operand" "")
|
||
(match_operand:SI 1 "reg_or_cint_operand" "")))]
|
||
""
|
||
"")
|
||
|
||
;; Signed compare, `test' first.
|
||
|
||
(define_insn ""
|
||
[(set (cc0)
|
||
(match_operand:SI 0 "register_operand" "r"))]
|
||
"next_insn_tests_no_unsigned (insn)"
|
||
"cis %0,0"
|
||
[(set_attr "length" "2")
|
||
(set_attr "type" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (match_operand:SI 0 "register_operand" "r,r,r"))
|
||
(set (match_operand:SI 1 "reg_or_nonsymb_mem_operand" "=0,r,Q")
|
||
(match_dup 0))]
|
||
"next_insn_tests_no_unsigned (insn)"
|
||
"@
|
||
cis %1,0
|
||
nilo %1,%0,65535
|
||
st%M1 %0,%1\;cis %0,0"
|
||
[(set_attr "type" "compare,compare,store")
|
||
(set_attr "length" "2,4,6")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0)
|
||
(compare (match_operand:SI 0 "register_operand" "r,r,r")
|
||
(match_operand:SI 1 "reg_or_cint_operand" "I,K,r")))]
|
||
"next_insn_tests_no_unsigned (insn)"
|
||
"@
|
||
cis %0,%1
|
||
cil %0,%1
|
||
c %0,%1"
|
||
[(set_attr "length" "2,4,2")
|
||
(set_attr "type" "compare")])
|
||
|
||
;; Unsigned comparisons, `test' first, again.
|
||
(define_insn ""
|
||
[(set (cc0)
|
||
(match_operand:SI 0 "register_operand" "r"))]
|
||
"! next_insn_tests_no_unsigned (insn)"
|
||
"clil %0,0"
|
||
[(set_attr "type" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0)
|
||
(compare (match_operand:SI 0 "register_operand" "r,r")
|
||
(match_operand:SI 1 "reg_or_cint_operand" "K,r")))]
|
||
"! next_insn_tests_no_unsigned (insn)"
|
||
"@
|
||
clil %0,%1
|
||
cl %0,%1"
|
||
[(set_attr "length" "4,2")
|
||
(set_attr "type" "compare")])
|
||
|
||
;; Bit test insn. Many cases are converted into this by combine. This
|
||
;; uses the ROMP test bit.
|
||
|
||
(define_insn ""
|
||
[(set (cc0)
|
||
(zero_extract (match_operand:SI 0 "register_operand" "r,r")
|
||
(const_int 1)
|
||
(match_operand:SI 1 "reg_or_any_cint_operand" "r,n")))]
|
||
"next_insn_tests_no_inequality (insn)"
|
||
"@
|
||
mttb %0,%1
|
||
mttbi%t1 %0,%S1"
|
||
[(set_attr "length" "2")
|
||
(set_attr "type" "compare")
|
||
(set_attr "cc" "tbit")])
|
||
|
||
;; Floating-point comparisons. There are two, equality and order.
|
||
;; The difference will be that a trap for NaN will be given on the orderr
|
||
;; comparisons only.
|
||
|
||
(define_expand "cmpsf"
|
||
[(parallel [(set (cc0) (compare (match_operand:SF 0 "general_operand" "")
|
||
(match_operand:SF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "cmpdf"
|
||
[(parallel [(set (cc0) (compare (match_operand:DF 0 "general_operand" "")
|
||
(match_operand:DF 1 "general_operand" "")))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "tstsf"
|
||
[(parallel [(set (cc0) (match_operand:SF 0 "general_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
(define_expand "tstdf"
|
||
[(parallel [(set (cc0) (match_operand:DF 0 "general_operand" ""))
|
||
(clobber (reg:SI 0))
|
||
(clobber (reg:SI 15))])]
|
||
""
|
||
"")
|
||
|
||
;; There are four cases for compare and two for test. These correspond
|
||
;; to each input having a floating-point conversion or not.
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (compare (match_operand 0 "general_operand" "frg")
|
||
(match_operand 1 "general_operand" "frg")))
|
||
(clobber (match_operand:SI 2 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 3 "reg_15_operand" "=&t"))]
|
||
"GET_MODE (operands[1]) == SFmode || GET_MODE (operands[1]) == DFmode"
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[0], operands[1], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (compare (match_operand 0 "general_operand" "frg")
|
||
(match_operator 1 "float_conversion"
|
||
[(match_operand 2 "general_operand" "frg")])))
|
||
(clobber (match_operand:SI 3 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 4 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[0], operands[2], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (compare (match_operator 0 "float_conversion"
|
||
[(match_operand 1 "general_operand" "frg")])
|
||
(match_operand 2 "general_operand" "frg")))
|
||
(clobber (match_operand:SI 3 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 4 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[1], operands[2], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (compare (match_operator 0 "float_conversion"
|
||
[(match_operand 1 "general_operand" "frg")])
|
||
(match_operator 2 "float_conversion"
|
||
[(match_operand 3 "general_operand" "frg")])))
|
||
(clobber (match_operand:SI 4 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 5 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[1], operands[3], 0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (match_operand 0 "general_operand" "frg"))
|
||
(clobber (match_operand:SI 1 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 2 "reg_15_operand" "=&t"))]
|
||
"GET_MODE (operands[0]) == SFmode || GET_MODE (operands[0]) == DFmode"
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[0], immed_real_const_1 (0, 0,
|
||
GET_MODE (operands[0])),
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
(define_insn ""
|
||
[(set (cc0) (match_operator 0 "float_conversion"
|
||
[(match_operand 1 "general_operand" "frg")]))
|
||
(clobber (match_operand:SI 2 "reg_0_operand" "=&z"))
|
||
(clobber (match_operand:SI 3 "reg_15_operand" "=&t"))]
|
||
""
|
||
"*
|
||
{ return output_fpop (next_insn_tests_no_inequality (insn) ? EQ : GE,
|
||
operands[1], immed_real_const_1 (0, 0,
|
||
GET_MODE (operands[1])),
|
||
0, insn);
|
||
}"
|
||
[(set_attr "type" "fp")
|
||
(set_attr "cc" "compare")])
|
||
|
||
;; Branch insns. Unsigned vs. signed have already
|
||
;; been taken care of. The only insns that need to be concerned about the
|
||
;; test bit are beq and bne because the rest are either always true,
|
||
;; always false, or converted to EQ or NE.
|
||
|
||
;; For conditional branches, we use `define_expand' and just have two patterns
|
||
;; that match them. Operand printing does most of the work.
|
||
|
||
(define_expand "beq"
|
||
[(set (pc)
|
||
(if_then_else (eq (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bne"
|
||
[(set (pc)
|
||
(if_then_else (ne (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bgt"
|
||
[(set (pc)
|
||
(if_then_else (gt (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bgtu"
|
||
[(set (pc)
|
||
(if_then_else (gtu (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "blt"
|
||
[(set (pc)
|
||
(if_then_else (lt (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bltu"
|
||
[(set (pc)
|
||
(if_then_else (ltu (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bge"
|
||
[(set (pc)
|
||
(if_then_else (ge (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bgeu"
|
||
[(set (pc)
|
||
(if_then_else (geu (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "ble"
|
||
[(set (pc)
|
||
(if_then_else (le (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
(define_expand "bleu"
|
||
[(set (pc)
|
||
(if_then_else (leu (cc0)
|
||
(const_int 0))
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"")
|
||
|
||
;; Define both directions of branch and return.
|
||
|
||
(define_insn ""
|
||
[(set (pc)
|
||
(if_then_else (match_operator 1 "comparison_operator"
|
||
[(cc0) (const_int 0)])
|
||
(label_ref (match_operand 0 "" ""))
|
||
(pc)))]
|
||
""
|
||
"*
|
||
{
|
||
if (restore_compare_p (operands[1]))
|
||
return 0;
|
||
else if (get_attr_length (insn) == 2)
|
||
return \"j%j1 %l0\";
|
||
else
|
||
return \"b%j1%# %l0\";
|
||
}"
|
||
[(set_attr "type" "branch")])
|
||
|
||
(define_insn ""
|
||
[(set (pc)
|
||
(if_then_else (match_operator 0 "comparison_operator"
|
||
[(cc0) (const_int 0)])
|
||
(return)
|
||
(pc)))]
|
||
"null_epilogue ()"
|
||
"*
|
||
{
|
||
if (restore_compare_p (operands[0]))
|
||
return 0;
|
||
else
|
||
return \"b%j0r%# r15\";
|
||
}"
|
||
[(set_attr "type" "return")])
|
||
|
||
(define_insn ""
|
||
[(set (pc)
|
||
(if_then_else (match_operator 1 "comparison_operator"
|
||
[(cc0) (const_int 0)])
|
||
(pc)
|
||
(label_ref (match_operand 0 "" ""))))]
|
||
""
|
||
"*
|
||
{
|
||
if (restore_compare_p (operands[1]))
|
||
return 0;
|
||
else if (get_attr_length (insn) == 2)
|
||
return \"j%J1 %l0\";
|
||
else
|
||
return \"b%J1%# %l0\";
|
||
}"
|
||
[(set_attr "type" "branch")])
|
||
|
||
(define_insn ""
|
||
[(set (pc)
|
||
(if_then_else (match_operator 0 "comparison_operator"
|
||
[(cc0) (const_int 0)])
|
||
(pc)
|
||
(return)))]
|
||
"null_epilogue ()"
|
||
"*
|
||
{
|
||
if (restore_compare_p (operands[0]))
|
||
return 0;
|
||
else
|
||
return \"b%J0r%# r15\";
|
||
}"
|
||
[(set_attr "type" "return")])
|
||
|
||
;; Unconditional branch and return.
|
||
|
||
(define_insn "jump"
|
||
[(set (pc)
|
||
(label_ref (match_operand 0 "" "")))]
|
||
""
|
||
"*
|
||
{
|
||
if (get_attr_length (insn) == 2)
|
||
return \"j %l0\";
|
||
else
|
||
return \"b%# %l0\";
|
||
}"
|
||
[(set_attr "type" "branch")])
|
||
|
||
(define_insn "return"
|
||
[(return)]
|
||
"null_epilogue ()"
|
||
"br%# r15"
|
||
[(set_attr "type" "return")])
|
||
|
||
(define_insn "indirect_jump"
|
||
[(set (pc) (match_operand:SI 0 "register_operand" "r"))]
|
||
""
|
||
"br%# %0"
|
||
[(set_attr "type" "branch")
|
||
(set_attr "length" "2")])
|
||
|
||
;; Table jump for switch statements:
|
||
(define_insn "tablejump"
|
||
[(set (pc)
|
||
(match_operand:SI 0 "register_operand" "r"))
|
||
(use (label_ref (match_operand 1 "" "")))]
|
||
""
|
||
"br%# %0"
|
||
[(set_attr "type" "branch")
|
||
(set_attr "length" "2")])
|
||
|
||
;;- Local variables:
|
||
;;- mode:emacs-lisp
|
||
;;- comment-start: ";;- "
|
||
;;- eval: (set-syntax-table (copy-sequence (syntax-table)))
|
||
;;- eval: (modify-syntax-entry ?[ "(]")
|
||
;;- eval: (modify-syntax-entry ?] ")[")
|
||
;;- eval: (modify-syntax-entry ?{ "(}")
|
||
;;- eval: (modify-syntax-entry ?} "){")
|
||
;;- End:
|